socket编程TCP/IP

socket编程TCP/IP_第1张图片

服务器端:

int socket_fd, connect_fd;
struct sockaddr_in servaddr;

初始化:
socket_fd = sock(AF_INET, SOCK_STREAM, 0);

bzero(&serveraddr, sizeof(serveraddr));
servaddr.sin_family = AF_INET; /设置IPv4通信/
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//IP地址设置成INADDR_ANY,让系统自动获取本机的IP地址。
servaddr.sin_port = htons(SERV_PORT);//设置服务器端口为SERV_PORT

bind():
bind(socket_fd, (struct sockaddr*)&serveraddr, sizeof(serveraddr));

listen():
listen(socket_fd, 10); //10指的是连接队列的最大长度(第二个参数就是未完成队列的大小)

accept():

int accept(int sockfd, struct sockaddr* addr, socklen_t* len)
accept默认会阻塞进程,直到有一个客户连接建立后返回,它返回的是一个新可用的套接字,这个套接字是连接套接字。此时我们需要区分两种套接字,一种套接字正如accept的参数sockfd,它是监听套接字,在调用listen函数之后,一个套接字会从主动连接的套接字变身为一个监听套接字;而accept返回是一个连接套接字,它代表着一个网络已经存在的点点连接。自然要问的是:为什么要有两种套接字?原因很简单,如果使用一个描述字的话,那么它的功能太多,使得使用很不直观,同时在内核确实产生了一个这样的新的描述字。如果accept成功返回,则服务器与客户已经正确建立连接了,此时服务器通过accept返回的套接字来完成与客户的通信。

参数sockfd就是上面解释中的监听套接字
参数addr,这是一个结果参数,它用来接受一个返回值,这返回值指定客户端的地址,当然这个地址是通过某个地址结构来描述的,用户应该知道这一个什么样的地址结 构。如果对客户的地址不感兴趣,那么可以把这个值设置为NULL。

如果你给accept函数第2和第3个参数赋值后,就要记得“用之”,否则就会报错。

connect_fd = accept(socket_fd, (struct sockaddr*)NULL, NULL))
connect_fd = accept(socket_fd, (struct sockaddr *)&clientaddr, &addrlen);//返回的是客户端和服务端专用通道的socket描述符

read()/write()/send()/recv:

Ssize_t write(int fd,const void *buf,size_t nbytes);
Write函数将buf中的nbytes字节内容写入到文件描述符中,成功返回写的字节数,失败返回-1.
Ssize_t read(int fd,void *buf,size_t nbyte)
Read函数是负责从fd中读取内容,当读取成功时,read返回实际读取到的字节数,如果返回值是0,表示已经读取到文件的结束了,小于0表示是读取错误。

int recv(int sockfd,void *buf,int len,int flags)
int send(int sockfd,void *buf,int len,int flags)
ecv和send函数提供了和read和write差不多的功能.但是他们提供 了第四个参数来控制读写操作.前面的三个参数和read,write相同,第四个参数能够是0或是以下的组合.


| MSG_DONTROUTE | 不查找路由表 |
| MSG_OOB | 接受或发送带外数据 |
| MSG_PEEK | 查看数据,并不从系统缓冲区移走数据|
| MSG_WAITALL | 等待任何数据 |
|————————————————————–|
MSG_DONTROUTE:是send函数使用的标志.这个标志告诉IP协议.目的主机在本地网络上面,没有必要查找路由表.这个标志一般用网络诊断和路由程式里面.
MSG_OOB:表示能够接收和发送带外的数据.关于带外数据我们以后会解释的.
MSG_PEEK:是recv函数的使用标志,表示只是从系统缓冲区中读取内容,而不清楚系统缓冲区的内容.这样下次读的时候,仍然是相同的内容.一般在有多个进程读写数据时能够使用这个标志.
MSG_WAITALL是recv函数的使用标志,表示等到任何的信息到达时才返回.使用这个标志的时候recv回一直阻塞,直到指定的条件满足,或是发生了错误.1)当读到了指定的字节时,函数正常返回.返回值等于len 2)当读到了文档的结尾时,函数正常返回.返回值小于len 3)当操作发生错误时,返回-1,且配置错误为相应的错误号(errno)
假如flags为0,则和read,write相同的操作.更有其他的几个选项,但是我们实际上用的很少,能够查看Linux Programmer’s Manual得到周详解释.

len= recv(connect_fd, buf, MAXLINE, 0);
buf[len] = ‘\0’;

close():
close(connect_fd);
close(socket_fd);

标题

客户端

struct sockaddr_in servaddr;
int client_sockfd;
初始化:
client_sockfd=socket(AF_INET,SOCK_STREAM,0)
bzero(&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET;
//char ipstr[] = “192.168.6.254”;
inet_pton(AF_INET, ipstr, &serveraddr.sin_addr.s_addr);
//#define SERVER_PORT 8000
serveraddr.sin_port = htons(SERVER_PORT);

connect():
connect(confd, (struct sockaddr *)&serveraddr, sizeof(serveraddr));

向服务器端发送数据:
send(client_sockfd, sendbuf, strlen(sendbuf), 0)
接受服务器端传过来的数据:
rec_len = recv(client_sockfd,receivebuf, MAXLINE,0);

关闭:
close(client_sockfd);

例子:

服务器:

/*********************************************************************************
 *      Copyright:  (C) 2017 zoulei
 *                  All rights reserved.
 *
 *       Filename:  server.c
 *    Description:  This file
 *
 *        Version:  1.0.0(2017年06月18日)
 *         Author:  zoulei 
 *      ChangeLog:  1, Release initial version on "2017年06月18日 18时13分42秒"
 *
 ********************************************************************************/
 
#include
#include
#include
#include
#include
#include
#include
#define SERV_PORT 9998
#define MAXLINE 4096
 
int main(int argc, char** argv)
{
    int    socket_fd, connect_fd;
    struct sockaddr_in     servaddr; /* 服务器端网络地址结构体 */
    char    buf[MAXLINE],sendbuf[MAXLINE];
    int     len;
    /*创建服务器端套接字--IPv4协议,面向连接通信,TCP协议*/
    if( (socket_fd = socket(AF_INET, SOCK_STREAM, 0)) == -1 ){
    printf("create socket error: %s(errno: %d)\n",strerror(errno),errno);
    exit(0);
    }
    /*初始化*/
    memset(&servaddr, 0, sizeof(servaddr));/*数据初始化-清零 */
    servaddr.sin_family = AF_INET;  /*设置IPv4通信*/
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//IP地址设置成INADDR_ANY,让系统自动获取本机的IP地址。
    servaddr.sin_port = htons(SERV_PORT);//设置服务器端口为SERV_PORT
 
    /*将本地地址绑定到所创建的套接字上*/
    if( bind(socket_fd, (struct sockaddr*)&servaddr, sizeof(servaddr)) <0)
    {
         printf("bind socket error: %s(errno: %d)\n",strerror(errno),errno);
         exit(0);
    }
    /*开始监听是否有客户端连接*/
    if( listen(socket_fd, 10) <0)
    {
         printf("listen socket error: %s(errno: %d)\n",strerror(errno),errno);
         exit(0);
    }
    printf("waiting for client's connection......\n");
 
   /*阻塞直到有客户端连接,不然多浪费CPU资源*/
    if((connect_fd = accept(socket_fd, (struct sockaddr*)NULL, NULL)) <0)
     {
          printf("accept socket error: %s(errno: %d)",strerror(errno),errno);
          exit(1);
     }
        /*接受客户端传过来的数据*/
  while((len= recv(connect_fd, buf, MAXLINE, 0))>0)
     {
        buf[len] = '\0';
        printf("receive message from client: %s\n", buf);
       /*向客户端发送回应数据*/
        printf("send message to client: \n");
        fgets(sendbuf, 4096, stdin);
       if( send(connect_fd, sendbuf, strlen(sendbuf), 0) < 0)
         {
            printf("send messaeg error: %s(errno: %d)\n", strerror(errno), errno);
            exit(0);
         }
     }
  close(connect_fd);
  close(socket_fd);
}

客户端:

/*********************************************************************************
 *      Copyright:  (C) 2017 zoulei
 *                  All rights reserved.
 *
 *       Filename:  client.c
 *    Description:  This file
 *
 *        Version:  1.0.0(2017年06月12日)
 *         Author:  zoulei 
 *      ChangeLog:  1, Release initial version on "2017年06月12日 20时06分47秒"
 *
 ********************************************************************************/
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
 
#define MAXLINE    1024
#define SERV_PORT  9998
 
int main(int argc, char *argv[])
        {
            char sendbuf[MAXLINE],receivebuf[MAXLINE];
            struct sockaddr_in servaddr;
            int client_sockfd;
            int rec_len;
            /* 判断命令端输入的参数是否正确 */
             if( argc != 2)
             {
                 printf("usage: ./client \n");
                 exit(0);
              }
             /* 创建客户端套接字--IPv4协议,面向连接通信,TCP协议*/
             if((client_sockfd=socket(AF_INET,SOCK_STREAM,0))<0)
              {
                  perror("socket");
                  exit(0);
              }
             /* 初始化 */
            memset(&servaddr,0,sizeof(servaddr)); /* 数据初始化-清零 */
            servaddr.sin_family = AF_INET; /* 设置IPv4通信 */
            servaddr.sin_port = htons(SERV_PORT);/* 设置服务器端口号 */
            /* IP地址转换函数,将点分十进制转换为二进制 */
             if( inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0)
              {
                   printf("inet_pton error for %s\n",argv[1]);
                   exit(0);
              }
             /* 将套接字绑定到服务器的网络地址上*/
            if( connect(client_sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr))<0)
              {
                  perror("connected failed");
                  exit(0);
              }
            /* 循环发送接收数据,send发送数据,recv接收数据 */
       while(1)
           {
             printf("send msg to server: \n");
             fgets(sendbuf, 1024, stdin);
             /* 向服务器端发送数据 */
             if( send(client_sockfd, sendbuf, strlen(sendbuf), 0) < 0)
              {
                  printf("send msg error: %s(errno: %d)\n", strerror(errno), errno);
                  exit(0);
              }
             /* 接受服务器端传过来的数据 */
             if((rec_len = recv(client_sockfd,receivebuf, MAXLINE,0)) == -1)
              {
                   perror("recv error");
                   exit(1);
              }
             receivebuf[rec_len]='\0';
             printf("Response from server: %s\n",receivebuf);
           }
       /* 关闭套接字 */
    close(client_sockfd);
    return 0;
        }

你可能感兴趣的:(Linux)