基於UDP和TCP實現CS

1.我們應該先了解UDP和TCP協議

首先UDP和TCP是基於傳輸層的協議

我們需要了解UDP和TCP的特性

TCP協議的特性:

(1) 面向連接的服務;

(2) 可靠的數據傳輸服務;

(3) 面向字節流。

UDP協議的特性:

(1) 無連接服務;

(2) 不可靠的數據傳輸;

(3) 面向數據報;

以下是流行因特網應用層及其應用層協議和支撐的運輸協議:

應用

應用層協議

支撐的運輸層協議

電子郵件

SMTP

TCP

遠程終端訪問

Telnet

TCP

Web

HTTP

TCP

文件傳輸協議

FTP

TCP

流式媒體

HTTP

TCP

因特網電話

SIP

UDPTCP

用UDP實現的CS

server.c

#include<sys/types.h>
#include<sys/socket.h>
#include<string.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<fcntl.h>
#include<errno.h>


static void Usage()
{
    printf("Usage:%s ./server [ip] [port]\n");
}
int main(int argc,char* argv[])
{
    if(argc!=3)
    {
        Usage();
        return 1;
    }
    int sock=socket(AF_INET,SOCK_DGRAM,0);//udp使用SOCK_DGRAM
    if(sock<0)
    {
        perror("socket");
        return 2;
    }
    printf("%d\n",sock);
    //填充套接字
    struct sockaddr_in local;
    local.sin_family=AF_INET;
    local.sin_port=htons(atoi(argv[2]));
    local.sin_addr.s_addr=inet_addr(argv[1]);
    //綁定
    if(bind(sock,(struct sockaddr*)&local,sizeof(local))<0)
    {
        perror("bind");
        return 3;
    }
    //接收和發送
    char buf[1024];
    struct sockaddr_in client;
    while(1)
    {
        socklen_t len=sizeof(client);
        ssize_t s=recvfrom(sock,buf,sizeof(buf)-1,0,(struct sockaddr*)&client,&len);
        if(s>0)
        {
            buf[s]=0;
            printf("[%s:%d]: %s\n",inet_ntoa(client.sin_addr),
            ntohs(client.sin_port),buf);
            sendto(sock,buf,strlen(buf),0,(struct sockaddr*)&client,sizeof(client));
        }
    }
    return 0;
}

client.c

#include<stdlib.h>
#include<string.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<fcntl.h>
#include<errno.h>

static void Usage()
{
    printf("Usage:%s ./server [ip] [port]\n");
}
int main(int argc,char* argv[])
{
    if(argc!=3)
    {
        Usage();
        return 1;
    }
    int sock=socket(AF_INET,SOCK_DGRAM,0);//udp使用SOCK_DGRAM
    if(sock<0)
    {
        perror("socket");
        return 2;
    }
    printf("%d\n",sock);
    //填充套接字
    struct sockaddr_in server;
    server.sin_family=AF_INET;
    server.sin_port=htons(atoi(argv[2]));
    server.sin_addr.s_addr=inet_addr(argv[1]);
    
    //發送和接收
    char buf[1024];
    struct sockaddr_in peer;
    while(1)
    {
        socklen_t len=sizeof(peer);
        printf("please Enter#");
        fflush(stdout);
        ssize_t s=read(0,buf,sizeof(buf)-1);
        if(s>0)
        {
            buf[s-1]=0;
            sendto(sock,buf,strlen(buf),0,(struct sockaddr*)&server,sizeof(server));
            ssize_t _s=recvfrom(sock,buf,sizeof(buf)-1,0,(struct sockaddr*)&peer,&len);
            if(_s>0)
            {
                buf[_s]=0;
                printf("server echo %s\n",buf);
            }
        }
    }
    return 0;
}

基於TCP實現的CS(一對一的)

server.c

// ./server 192.168.x.x  8080//服務器這樣跑,命令行需要獲得192.168.x.x和端口號
static void Usage()
{
    printf("Usage:%s./server [ip] [port]\n");
}
int startup(const char *ip,int port)
{
    //1.創建套接字
    int fd=socket(AF_INET,SOCK_STREAM,0);
    if(fd<0)
    {
        perror("socket");
        exit(2);
    }

    //套接字絕對是3,因爲 0 1 2被佔
    printf("fd :%d\n",fd);

    //2.填充字節
    struct sockaddr_in local;
    //2.1 ipv4
    local.sin_family=AF_INET;
    //2.2 端口號
    local.sin_port=htons(port);//將主機序列轉化爲網絡序列。4個字節
    //2.3 對結構體可以整體初始化,但不可以整體賦值,要逐個賦值 ip(需要轉化)字符串轉爲4字節網絡序列
    local.sin_addr.s_addr=inet_addr(ip);

    //2.綁定
    //判斷綁定是否成功,將本地信息和文件信息綁定
    if(bind(fd,(struct sockaddr*)&local,sizeof(local))<0)
    {
        perror("bind");
        exit(3);
    }
    printf("bind ok!");
    //3.監聽套接字
    if(listen(fd,10)<0)
    {
        perror("listen");
        exit(4);
    }
    printf("listen ok!");
    return fd;
}
int main(int argc,char* argv[])
{
    if(argc!=3)
    {
        Usage();
        return 1;
    }

    int listen_fd=startup(argv[1],atoi(argv[2]));
    printf("bind and listen success,wait .....\n");
    while(1)//循環
    {
        //接受,要知道誰連我
        struct sockaddr_in client;
        socklen_t len=sizeof(client);
        int new_fd=accept(listen_fd,(struct sockaddr*)&client,&len);
        //判斷accept是否失敗
        if(new_fd<0)
        {
            perror("accept");
            continue;
        }
        printf("get new client[%s:%d]\n",inet_ntoa(client.sin_addr),ntohs(client.sin_port));

        //服務器read->read
        while(1)
        {
            char buf[1024]={0};
            ssize_t s=read(new_fd,buf,sizeof(buf)-1);
            if(s>0)
            {
                buf[s]=0;
                printf("client# %s\n",buf);
                write(new_fd,buf,strlen(buf));
            }
            else if(s==0)
            {
                printf("client close\n");
                break;
            }
            else
            {
                perror("read");
                break;
            }
        }
        close(new_fd);
    }
    return 0;
}

client.c

#include<stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<string.h>

//  ./client server_ip server_port

static void Usage()
{
    printf("Usage:%s ./client [ip][port]\n");
}
int main(int argc,char* argv[])
{
    if(argc!=3)
    {
        Usage();
        return 1;
    }
   // socket; 
   int sock=socket(AF_INET,SOCK_STREAM,0);
   if(sock<0)
   {
       perror("sock");
       return 2;
   }
   printf("%d\n",sock);
    //填充字節
    struct sockaddr_in server;
    server.sin_family=AF_INET;
    server.sin_port=htons(atoi(argv[2]));
    server.sin_addr.s_addr=inet_addr(argv[1]);
    printf("connect ok\n");
    //connect
    if(connect(sock,(struct sockaddr*)&server,sizeof(server))==-1)
    {
        perror("connect");
        return 3;
    }
    char buf[1024]={0};
    printf("ok\n");
    while(1)
    {
        printf("connect:ok2");
        printf("please Enter$");
        fflush(stdout);
        ssize_t s=read(0,buf,sizeof(buf)-1);
        if(s>0)
        {
            buf[s-1]=0;
            write(sock,buf,strlen(buf));
            ssize_t _s=read(sock,buf,sizeof(buf)-1);
            if(_s>0)
            {
                buf[_s]=0;
                printf("server echo$ %s\n",buf);
            }
        }

    }
    close(sock);
    return 0;
}
通過以上代碼我們可以更加了解tcp和udp通信






發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章