一、多线程概念
二、socket编程
socket 编程_socket程序_lqonlylove的博客-CSDN博客
三、多线程socket服务器实现
1、服务器代码
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <pthread.h>
#define BUF_LEN 1024
void *thread_worker(void *ctx);
int main(int argc, char* argv[])
{
pthread_t tid;
struct sockaddr_in addr;
int cfd=0; // 客户端 fd
int sfd=0; // 服务端 fd
// 1、创建服务器 socket
sfd = socket(AF_INET,SOCK_STREAM,0);
if(sfd == -1){
return -1;
}
// 2、初始化 sockaddr_un 结构体
memset(&addr,0,sizeof(struct sockaddr_in));
addr.sin_family = AF_INET;
inet_pton(AF_INET,"127.0.0.1",&addr.sin_addr);
addr.sin_port = htons(8888);
bzero(&(addr.sin_zero),8);
// 3、绑定 IP 地址和端口号
if(bind(sfd,(struct sockaddr*)&addr,sizeof(struct sockaddr_in)) == -1){
goto ERROTR_RETURN;
}
// 4、监听客户端连接
if(listen(sfd,5) == -1){
goto ERROTR_RETURN;
}
while(1){
// 5、接收客户端连接
cfd = accept(sfd,NULL,NULL);
if(cfd == -1){
printf("accept errno!\r\n");
exit(-1);
}
// 6、创建线程并处理业务(问题:tid会被覆盖,实际开发中需要进行管理)
if (pthread_create(&tid, NULL, thread_worker, (void*)&cfd)){
printf("Create thread failure!\r\n");
close(cfd);
}
}
return 0;
ERROTR_RETURN:
close(sfd);
return -1;
}
void *thread_worker(void *cfdx)
{
int cfd = *(int*)cfdx;
char buf[BUF_LEN] = "";
int buf_len = 0;
/* 防止僵尸线程产生 */
pthread_detach(pthread_self());
/* 数据收发 */
while(1){
memset(buf,0,BUF_LEN);
buf_len = 0;
buf_len = recv(cfd,buf,BUF_LEN,0);
if(buf_len == 0){
goto CLOSE_RETURN;
}
printf("%ld socket recv data length is %d: %s\n",(long)cfd,buf_len,buf);
}
CLOSE_RETURN:
printf("%ld socket clietn close!\n",(long)cfd);
close(cfd);
pthread_exit(0);
return NULL;
}
2、makefile
threadSocket: main.c
gcc main.c -o threadSocket -lpthread
.PHONY : clean
clean:
rm -rf threadSocket
3、客户端代码
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
int main(int argc, char* argv[])
{
struct sockaddr_in addr;
int cfd=0;
// 1、创建客户端 socket
cfd = socket(AF_INET,SOCK_STREAM,0);
// 2、初始化 sockaddr_un 结构体
memset(&addr,0,sizeof(struct sockaddr_in));
addr.sin_family = AF_INET;
inet_pton(AF_INET,"127.0.0.1",&addr.sin_addr);
addr.sin_port = htons(8888);
bzero(&(addr.sin_zero),8);
// 3、连接服务器
if(connect(cfd,(struct sockaddr*)&addr,sizeof(struct sockaddr_in)) == -1){
goto ERROTR_RETURN;
}
// 4、收发数据
while(1){
send(cfd,"cleintOne send data",19,0);
sleep(5);
}
return 0;
ERROTR_RETURN:
close(cfd);
return -1;
}