服务器端
这个步骤不知道算是TCP 还是UDP,希望大佬可以帮我解答一下…谢谢【鞠躬】
对于服务器,需要做的一些步骤:
- 创建套接字 socket
- 绑定 bind
- 监听 listen
- 接收、处理 accept
首先要创建一个套接字
int socket(int family,int type,int protocol);
family 是协议族,当前学习进度使用为AF_INET
rype为套接字类型
- 笔记1中记的SOCK_STREAM 流式套接字,代表着TCP;
- SOCK_DGRAM 数据报式套接字,代表着UDP
protocol为协议类别,使用写0 就行
一个或者几个端口代表一个进程,使用Socket创建的套接字不会分配端口
使用socket创建的套接字,是主动套接字,但是作为服务器需要被动的链接,所以需要进行处理。
创建完套接字需要进行绑定
绑定本机的数据。所使用的函数:
#include<sys/socket.h>
int bind(int sockfd,const struct sockaddr * myaddr,socklen_t addrlen);
它的功能是将本地协议地址与sockfd进行绑定。
其中sockfd是返回的创建的套接字的值
myaddr用于指向特定协议的地址结构体指针,是通用套接字地质结构体地址。
addrlen是这个地址结构的长度。
成功返回0,失败返回其它。
struct sockaddr_in myaddr;
unsigned short port = 8000;
bzero(&my_addr , sizeof(my_addr));
my_addr.sin_family = AF_INET;
my_addr.sin_port = htons(port);
my_addr.sin_addr.s_addr = htonl(INADDR_ANY);
err_log = bind(sockfd,(struct sockaddr *)&my_addr,sizeof(my_addr));
if(err_log != 0){
perror("bind");
close(sockfd);
exit(-1);
}
接下来就需要监听
int listen(int sockfd,int backlog);
1、监听函数的功能是将主动修改为被动,被动的监听才能是服务器的作用
2、使操作系统为这个套接字设置一个链接队列,用来记录所有连接到该套接字的链接。
其中sockfd 是监听的套接字
backlog 是链接队列的长度,意思就是同时可以缓存backlog个客户端,来依次完成任务
int err_log = 0;
err_log = listen(sockfd,10);
if(err_log != 0){
perror("listen");
close(sockfd);
exit(-1);
}
接收/处理队列的链接
已经监听到了很多的客户端,并且存到了队列中,那么这个函数就是用来取出队列中一个建立好的连接,如果没有任何连接可以用,就进入睡眠状态
#include<sys/socket.h>
int accept(int sockfd,struct sockaddr * cliaddr, socklen_t *addrlen);
sockfd : socket监听套接字
cliaddr : 用于存放客户端套接字地址结构
addrlen : 套接字地质结构体长度
注意accept返回的是一个已连接的套接字,这个套接字代表当前这个链接。
--这个套接字与刚开始socket创建的套接字不是一个套接字。
--使用socket函数创建的套接字是提供给服务器监听使用的。
--使用accept函数创建的套接字是新创建的一个套接字。
struct sockaddr_in client_addr; // 创建一个非通用套接字
char recv_buf[512] = ""; // 接收的
char cli_ip[INET_ADDRSTRLEN] = "";// 客户端IP 宏16
socklen_t client_addr_len = sizeof(client_addr);
int connfd = accept(sockfd,(struct sockaddr *)&client_addr,&client_addr_len);
if(connfd < 0){
perror("accept");
close(sockfd);
exit(-1);
}
inet_ntop(AF_INET, &client_addr.sin_addr,cli_ip,INET_ADDRSTRLEN);//字符串转换成点分十进制
printf("cli_ip is %s , port = %d \n",cli_ip, ntohs(client_addr.sin_port));
read(connfd,recv_buf,sizeof(recv_buf));// 从客户端读取一些信息,存放在recv_buff里。