socket编程TCP/IP

socket编程TCP/IP

服务器端:

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);