一、TCP一般
1.1、为例
理解:相当于你在服务器端建立好程序,然后监听客户端是否访问我,一旦访问我们握手成功,接收客户端的数据然后返回给客户端;同样在客户端有服务器的IP地址,尝试连接服务器端,连接成功后发送数据给服务器端,然后也接收服务器端的数据。
实现客户端和服务器端通讯的实现步骤
TCP服务器端的编写步骤:
1. 首先,你需要创建一个用于通讯的套接口,一般使用socket调用来实现。这等于你有了一个用于通讯的电话:)
2. 然后,你需要给你的套接口设定端口,相当于,你有了电话号码。这一步 一般通过设置网络套接口地址和调用bind函数来实现。
3. 调用listen函数使你的套接口成为一个监听套接字。 以上三个步骤是TCP服务器的常用步骤。
4. 调用accept函数来启动你的套接字,这时你的程序就可以等待客户端的连接了。
5. 处理客户端的连接请求。
6. 终止连接。
TCP编程的客户端一般步骤是:
1、创建一个socket,用函数socket();
2、设置socket属性,用函数setsockopt();* 可选
3、绑定IP地址、端口等信息到socket上,用函数bind();* 可选
4、设置要连接的对方的IP地址和端口等属性;
5、连接服务器,用函数connect()(相当于拨号);
6、收发数据,用函数send()和recv(),或者read()和write()(相当于通话);
7、关闭网络连接;
1.2、服务器端源代码
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <unistd.h>
#include <arpa/inet.h>
#define MAXBUF 1024
int main(int argc, char **argv)
{
int sockfd, new_fd;
socklen_t len;
struct sockaddr_in my_addr, their_addr;
unsigned int myport, lisnum;
char buf[MAXBUF + 1];
if (argv[1])
myport = atoi(argv[1]);
else
myport = 7838;
if (argv[2])
lisnum = atoi(argv[2]);
else
lisnum = 2;
if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
perror("socket");
exit(1);
}
else printf("socket created\n");
bzero(&my_addr, sizeof(my_addr));
my_addr.sin_family = PF_INET;
my_addr.sin_port = htons(myport);
if(argv[3]) my_addr.sin_addr.s_addr = inet_addr(argv[3]);
else my_addr.sin_addr.s_addr = INADDR_ANY;
if (bind(sockfd, (struct sockaddr *) &my_addr, sizeof(struct sockaddr)) == -1) {
perror("bind");
exit(1);
}
else printf("binded\n");
if (listen(sockfd, lisnum) == -1) {
perror("listen");
exit(1);
}
else printf("begin listen\n");
while(1) {
len = sizeof(struct sockaddr);
if ((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, &len)) == -1) {
perror("accept");
exit(errno);
}
else printf("server: got connection from %s, port %d, socket %d\n",inet_ntoa(their_addr.sin_addr), ntohs(their_addr.sin_port), new_fd);
/* 开始处理每个新连接上的数据收发 */
bzero(buf, MAXBUF + 1);
strcpy(buf, "这是在连接建立成功后向客户端发送的第一个消息\n只能向new_fd这个用accept函数新建立的socket发消息,不能向sockfd这个监听socket发送消息,监听socket不能用来接收或发送消息\n");
/* 发消息给客户端 */
len = send(new_fd, buf, strlen(buf), 0);
if(len < 0) {
printf("消息'%s'发送失败!错误代码是%d,错误信息是'%s'\n", buf, errno, strerror(errno));
}
else printf("消息'%s'发送成功,共发送了%d个字节!\n", buf, len);
bzero(buf, MAXBUF + 1);
/* 接收客户端的消息 */
len = recv(new_fd, buf, MAXBUF, 0);
if(len > 0) printf("接收消息成功:'%s',共%d个字节的数据\n", buf, len);
else printf("消息接收失败!错误代码是%d,错误信息是'%s'\n", errno, strerror(errno));
/* 处理每个新连接上的数据收发结束 */
}
close(sockfd);
return 0;
}
1.3、客户端源代码
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>
#include <resolv.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#define MAXBUF 1024
int main(int argc, char **argv)
{
int sockfd, len;
struct sockaddr_in dest;
char buffer[MAXBUF + 1];
if (argc != 3)
{printf ("参数格式错误!正确用法如下:\n\t\t%s IP地址 端口\n\t比如:\t%s 127.0.0.1 80\n此 程序用来从某个 IP 地址的服务器某个端口接收最多 MAXBUF 个字节的消息",argv[0], argv[0]);
exit(0);
}
/* 创建一个 socket 用于 tcp 通信 */
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
perror("Socket");
exit(errno);
}
printf("socket created\n");
/* 初始化服务器端(对方)的地址和端口信息 */
bzero(&dest, sizeof(dest));
dest.sin_family = AF_INET;
dest.sin_port = htons(atoi(argv[2]));
if (inet_aton(argv[1], (struct in_addr *) &dest.sin_addr.s_addr) == 0) {
perror(argv[1]);
exit(errno);
}
printf("address created\n");
/* 连接服务器 */
if (connect(sockfd, (struct sockaddr *) &dest, sizeof(dest)) != 0) {
perror("Connect ");
exit(errno);
}
printf("server connected\n");
/* 接收对方发过来的消息,最多接收 MAXBUF 个字节 */
bzero(buffer, MAXBUF + 1);
/* 接收服务器来的消息 */
len = recv(sockfd, buffer, MAXBUF, 0);
if(len > 0) printf("接收消息成功:'%s',共%d个字节的数据\n", buffer, len);
else printf("消息接收失败!错误代码是%d,错误信息是'%s'\n", errno, strerror(errno));
bzero(buffer, MAXBUF + 1);
strcpy(buffer, "这是客户端发给服务器端的消息\n");
/* 发消息给服务器 */
len = send(sockfd, buffer, strlen(buffer), 0);
if(len < 0) printf("消息'%s'发送失败!错误代码是%d,错误信息是'%s'\n", buffer, errno, strerror(errno));
else printf("消息'%s'发送成功,共发送了%d个字节!\n", buffer, len);
/* 关闭连接 */
close(sockfd);
return 0;
}
编译两个程序用下列命令:
gcc -Wall simple-server.c -o server
gcc -Wall simple-client.c -o client
启动服务端程序用如下命令:
./server 7838 1
启动客户端程序用如下命令:
./client 127.0.0.1 7838
就可以完成通讯功能。
1.4、参考网址
https://blog.csdn.net/u013457167/article/details/79582924
https://blog.csdn.net/lell3538/article/details/53335231
https://blog.csdn.net/weixin_37787043/article/details/78785495
二、select和setsockopt机制
2.1、简介
//Linux网络编程之TCP编程,select多路复用和超时检测,网络属性设置setsockopt之快速重启(经典)
//功能:一个服务器可以接收多个客户端的数据,并且每隔3秒提示阻塞在select前面超时提醒
//在服务器标准输入后,可以继续在同一个客户端发送数据到服务器
#if 0
-->int select(int nfds, fd_set *readfds, fd_set *writefds,
fd_set *exceptfds, struct timeval *timeout);
该函数用于监视的文件描述符的变化情况——读写或是异常。(统一管理)
返回值 >0 集合有响应
-1 出错
0 超时
参数1:所有监控的文件描述符中最大的那一个加1
参数2:所有要读的文件的文件描述符的集合
参数3:所有要写的文件的文件描述符的集合
参数4:异常集合的文件描述符
参数5:超时设置
NULL:一直阻塞,直到有文件描述符就绪或者出错
0:仅仅检测文件描述符集的状态,然后立即返回
其他:在指定时间内,如果没有事件发生,超时返回
void FD_SET(int fd, fd_set *set);
把指定的文件描述符加入集合中
void FD_CLR(int fd, fd_set *set);
从集合中清除指定的文件描述符
int FD_ISSET(int fd, fd_set *set);
判断哪个集合响应
void FD_ZERO(fd_set *set);
清空集合中所有的文件描述符
struct timeval {
long tv_sec; /* seconds */
long tv_usec; /* microseconds */
};
-->int setsockopt(int socket, int level, int option_name,
const void *option_value, socklen_t option_len);
参数:2:level 指定控制套接字的层次.可以取三种值:
1)SOL_SOCKET:通用套接字选项.
2)IPPROTO_IP:IP选项.
3)IPPROTO_TCP:TCP选项.
参数3:option_name 指定控制的方式(选项的名称),我们下面详细解释
选项名称 说明 数据类型
SOL_SOCKET
SO_BROADCAST 允许发送广播数据 int
SO_DEBUG 允许调试 int
SO_DONTROUTE 不查找路由 int
SO_ERROR 获得套接字错误 int
SO_KEEPALIVE 保持连接 int
SO_LINGER 延迟关闭连接 struct linger
SO_OOBINLINE 带外数据放入正常数据流 int
SO_RCVBUF 接收缓冲区大小 int
SO_SNDBUF 发送缓冲区大小 int
SO_RCVLOWAT 接收缓冲区下限 int
SO_SNDLOWAT 发送缓冲区下限 int
SO_RCVTIMEO 接收超时 struct timeval
SO_SNDTIMEO 发送超时 struct timeval
SO_REUSEADDR 允许重用本地地址和端口 int
SO_TYPE 获得套接字类型 int
SO_BSDCOMPAT 与BSD系统兼容 int
IPPROTO_IP
IP_HDRINCL 在数据包中包含IP首部 int
IP_OPTINOS IP首部选项 int
IP_TOS 服务类型
IP_TTL 生存时间 int
IPPRO_TCP
TCP_MAXSEG TCP最大数据段的大小 int
TCP_NODELAY 不使用Nagle算法 int
参数4: option_value 获得或者是设置套接字选项.根据选项名称的数据类型进行转换
#endif
2.2、服务器端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/select.h>
#include <sys/time.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#define SER_PORT 9999
#define SER_IP "192.168.7.115"
void sys_error(char * ch)
{
perror(ch);
exit(1);
}
int main(void)
{
char buf[128];
int serfd,ret;
//1.建立流式套接字
serfd = socket(AF_INET,SOCK_STREAM,0);
if(serfd < 0)
sys_error("socket failed");
//快速重启(在bind前面)
int on=1;//1表示打开
setsockopt(serfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on));
//2.绑定本地ip和端口
struct sockaddr_in ser;
bzero(&ser,sizeof(ser));
ser.sin_family = AF_INET;//选择IPV4
ser.sin_port = htons(SER_PORT);//填充端口
ser.sin_addr.s_addr = inet_addr(SER_IP);//填充IP
ret=bind(serfd,(struct sockaddr *)&ser,sizeof(ser));
if(ret < 0)
sys_error("bind failed");
//3.监听
ret=listen(serfd,5);
if(ret < 0)
sys_error("listen failed");
printf("listent ok\n");
//4.接收
struct sockaddr_in self;
bzero(&self,sizeof(self));
int len=sizeof(self);
int newfd;
fd_set r_ret;
int temp,maxfd;
while(1)
{
FD_SET(serfd,&r_ret);//把socket的文件描述符号加入读集合中
FD_SET(0,&r_ret);//把标准输入加入读集合中
maxfd = serfd;
if(maxfd < newfd)
{
FD_SET(newfd,&r_ret);//把客户端的文件描述符号加入读集合中
maxfd = newfd;
}
struct timeval st={3,0};
printf("select wait:\n");//超过3秒超时提醒
//超过3秒超时提醒
temp = select(maxfd+1,&r_ret,NULL,NULL,&st);
if(temp < 0)
sys_error("select failed");
else if(temp > 0)//集合有响应
{
if(FD_ISSET(0,&r_ret))//标准输入有响应
{
bzero(buf,128);
ret=read(0,buf,128);
if(ret < 0)
sys_error("read failed");
else if(ret == 0)
{
FD_CLR(0,&r_ret);
printf("stdin Bye-Bye\n");
}
else
printf("stdin input buf:%s\n",buf);
}
if(FD_ISSET(serfd,&r_ret))//客户端有响应
{
//接收
//返回建立好连接的套接字描述符,即客户端的文件描述符
newfd=accept(serfd,(struct sockaddr *)&self,&len);
if(newfd < 0)
sys_error("accept failed");
}
if(FD_ISSET(newfd,&r_ret))//判断新接入的客户端响应
{
bzero(buf,128);
ret = read(newfd,buf,128);
if(ret < 0)
sys_error("client read failed");
else if(ret == 0)
{
FD_CLR(newfd,&r_ret);
close(newfd);
}
else
printf("client buf:%s",buf);
}
}
#if 0
else
{
printf("time out\n");
}
#endif
}
//6.关闭
close(serfd);
close(newfd);
return 0;
}
2.3、客户端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define SER_PORT 9999
#define SER_IP "192.168.7.115"
void sys_error(char * ch)
{
perror(ch);
exit(1);
}
int main(void)
{
int clifd,ret;
char buf[128];
//1.建立流式套接字
clifd=socket(AF_INET,SOCK_STREAM,0);
if(clifd < 0)
sys_error("socket failed");
//2.主动发起连接
struct sockaddr_in ser;
bzero(&ser,sizeof(ser));
ser.sin_family = AF_INET;
ser.sin_port = htons(SER_PORT);
ser.sin_addr.s_addr = inet_addr(SER_IP);
ret = connect(clifd,(struct sockaddr *)&ser,sizeof(ser));
if(ret < 0)
sys_error("connect failed");
printf("connect ok\n");
//3.写数据
while(1)
{
bzero(buf,128);
fprintf(stderr,"please input:");
fgets(buf,128,stdin);
if(write(clifd,buf,strlen(buf)) < 0)
sys_error("write failed");
if(!strncmp(buf,"quit",4))
break;
}
//4.关闭
close(clifd);
return 0;
}
2.4、参考网址
https://blog.csdn.net/weixin_37787043/article/details/78791473