int main(int argc, const char *argv[])
{
//1 为通信创建一个端点
int sfd=socket(AF_INET,SOCK_STREAM,0);
//参数1 说明使用的是IPv4网络
//参数2 说明使用的是TCP面向连接的通讯方式
//参数3,由于参数2已经指定通信方式,直接为0
if(sfd ==-1)
{
perror("socket error");
return -1;
}
printf("socket success sfd=%d\n",sfd); // 3 (0 stdin ,stdout 1,stderr 2)
// 2 绑定IP地址和端口号
// 2.1 准备地址信息结构体
struct sockaddr_in sin;
sin.sin_family =AF_INET;
sin.sin_port =htons(SER_PORT);
sin.sin_addr.s_addr =inet_addr(SER_IP);//嵌入一个小的结构体
//绑定工作
if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)))
{
perror("bind error");
return -1;
}
printf("bind success\n");
// 3 将套接字设置成被动监听状态;
//
if(listen(sfd,128)==-1)
{
perror("listen error");
return -1;
}
printf("listen success\n");
// 4 阻塞等待客户端的连接
// 4.1 定义用于接受客户端信息的容器
struct sockaddr_in cin;
socklen_t addrlen = sizeof(cin);
int newfd =accept(sfd,(struct sockaddr*)&cin,&addrlen); // accept是阻塞函数
if(newfd ==-1)
{
perror("accept error");
return -1;
}
printf("accept success\n");
printf("[%s %d]发送来连接请求\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));
// 5 与客户端进行相互通信
char rbuf[128]={}; //用于读取消息内容的容器
while(1)
{
// 清空容器
bzero(rbuf,sizeof(rbuf));
//从套接字读取数据
int res =read(newfd,rbuf,sizeof(rbuf));
if(res == 0)
{
printf("客户端已经下线\n");
break ;
}
//将读取到的消息展示出来
printf("[%s %d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),rbuf );
//将收到的消息处理一下,回复给客户端
strcat(rbuf,"OK");
//将消息发送给客户端
write(newfd,rbuf,strlen(rbuf));
printf("发送成功\n");
}
// 6 关闭套接字
close(sfd);
return 0;
}
多线程实现TCP并发服务器
//定义信号处理函数
void handler (int signo)
{
//判断要处理的信号
if(signo == SIGCHLD)
{
while(waitpid(-1,NULL,WNOHANG) !=0);
}
}
int main(int argc, const char *argv[])
{
//将子进程的SIGCHLD(17)信号
//当子进程退出时,会向其子进程发送该信号
if(signal(SIGCHLD,handler) ==SIG_ERR)
{
perror("signal error");
return -1;
}
//1 为通信创建一个端点
int sfd=socket(AF_INET,SOCK_STREAM,0);
if(sfd ==-1)
{
perror("socket error");
return -1;
}
printf("socket success sfd=%d\n",sfd); // 3 (0 stdin ,stdout 1,stderr 2)
// 2 绑定IP地址和端口号
// 2.1 准备地址信息结构体
struct sockaddr_in sin;
sin.sin_family =AF_INET; //通信域
sin.sin_port =htons(SER_PORT); //端口号
sin.sin_addr.s_addr =inet_addr(SER_IP);//嵌入一个结构体,IP地址
//绑定工作
if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)) ==-1)
{
perror("bind error");
return -1;
}
printf("bind success\n");
// 3 将套接字设置成被动监听状态;
//
if(listen(sfd,128)==-1)
{
perror("listen error");
return -1;
}
printf("listen success\n");
// 4 阻塞等待客户端的连接
// 4.1 定义用于接受客户端信息的容器
struct sockaddr_in cin;
socklen_t addrlen = sizeof(cin);
int newfd =-1; // 客户端套接字变量
while(1)
{
//父进程
int newfd =accept(sfd,(struct sockaddr*)&cin,&addrlen);
if(newfd ==-1)
{
perror("accept error");
return -1;
}
printf("[%s %d]发送来连接请求\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));
pid_t pid =fork(); //创建子进程
pid_t pid =fork(); //创建子进程
if(pid>0)
{
//父进程体
//关闭newfd
close(newfd);
}
else if(pid ==0)
{
//关闭sfd
close(sfd);
//5与客户端进行相互通信、
char rbuf[128] ="";
while(1)
{
//清空容器
bzero(rbuf,sizeof(rbuf));
//从套接字中读取数据
//int res =read(newfd,rbuf,sizeof(rbuf)); read可以使用,但是不能用write写入
int res =recv(newfd,rbuf,sizeof(rbuf),0);
if(rbuf ==0)
{
printf("客户端已经举个下线\n");
break;
}
//将读取到的消息展示出来
printf("[%s : %d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),rbuf);
//将收到的消息处理一下,回复给客户端
//
strcat(rbuf,"HELLO WORLD");
//write(newfd,rbuf,strlen(rbuf));
send(newfd,rbuf,strlen(rbuf),0);
printf("发送成功\n");
}
// 6 关闭套接字
close(newfd);
//退出子进程
exit(EXIT_SUCCESS);
}
else
{
perror("fork error");
return -1;
}
}
close(sfd);
return 0;
}
多进程实现TC
//定义一个结构体类型,用于向线程体函数传递参数
struct Info
{
int newfd;
struct sockaddr_in cin;
};
// 定义线程体函数
//
void* deal_cli_msg(void *arg)
{
//解析传递过来的数据
int newfd =((struct Info*)arg)->newfd;
struct sockaddr_in cin=((struct Info*)arg)->cin;
//5与客户端进行相互通信、
char rbuf[128] ="";
while(1)
{
//清空容器
bzero(rbuf,sizeof(rbuf));
//从套接字中读取数据
//int res =read(newfd,rbuf,sizeof(rbuf)); read可以使用,但是不能用write写入
int res =recv(newfd,rbuf,sizeof(rbuf),0);
if(rbuf ==0)
{
printf("客户端已经举个下线\n");
break;
}
//将读取到的消息展示出来
printf("[%s : %d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),rbuf);
//将收到的消息处理一下,回复给客户端
//
strcat(rbuf,"HELLO WORLD");
//write(newfd,rbuf,strlen(rbuf));
send(newfd,rbuf,strlen(rbuf),0);
printf("发送成功\n");
}
// 6 关闭套接字
close(newfd);
//退出线程
pthread_exit(NULL);
}
int main(int argc, const char *argv[])
{
//1 为通信创建一个端点
int sfd=socket(AF_INET,SOCK_STREAM,0);
if(sfd ==-1)
{
perror("socket error");
return -1;
}
printf("socket success sfd=%d\n",sfd); // 3 (0 stdin ,stdout 1,stderr 2)
// 2 绑定IP地址和端口号
// 2.1 准备地址信息结构体
struct sockaddr_in sin;
sin.sin_family =AF_INET; //通信域
sin.sin_port =htons(SER_PORT); //端口号
sin.sin_addr.s_addr =inet_addr(SER_IP);//嵌入一个结构体,IP地址
//绑定工作
if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)) ==-1)
{
perror("bind error");
return -1;
}
printf("bind success\n");
// 3 将套接字设置成被动监听状态;
//
if(listen(sfd,128)==-1)
{
perror("listen error");
return -1;
}
printf("listen success\n");
// 4 阻塞等待客户端的连接
// 4.1 定义用于接受客户端信息的容器
struct sockaddr_in cin;
socklen_t addrlen = sizeof(cin);
int newfd =-1; // 客户端套接字变量
while(1)
{
//accept 函数会与选一个当前未分配的最小的文件描述,
//即使在阻塞过程中,有更小的文件描述符产生,本次操作的文件描述符也不会更改了
int newfd=accept(sfd,(struct sockaddr*)&cin,&addrlen);
if(newfd == -1)
{
perror("accept error");
return -1;
}
printf("[%s :%d]发来连接请求,newfd =%d\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd);
//定义要传递数据的结构体变量
struct Info buf ={newfd,cin};
//创建分支线程,用于通信
pthread_t tid= -1;
if(pthread_create(&tid,NULL,deal_cli_msg,&buf)== -1)
{
perror("pthread_creat error");
return -1;
}
//将线程设置成分离态
pthread_detach(tid);
}
close(sfd);
return 0;
}
P并发服务器