Linux下GNU C实现的简单聊天室程序
闲暇之余,利用linux下的多线程和网络知识,写了个简单多人聊天程序,意在说明用GNU C写网络类程序的步骤和需要注意的问题以及linux多线程的使用方法。
编译:
gcc -Wall -o server -lpthread server.c
gcc -Wall -o client -lpthread client.c
服务器端:
#include <sys/types.h> #include <sys/socket.h> #include <sys/wait.h> #include <stdio.h> #include <stdlib.h> #include <errno.h> #include <string.h> #include <sys/un.h> #include <sys/time.h> #include <sys/ioctl.h> #include <unistd.h> #include <netinet/in.h> #include <pthread.h> //定义同时聊天的人数 #define COUNT 5 //保存socket int socket_fd[COUNT]; //线程的入口函数 void pthread_function(int client_fd){ char message[1500]; char buf[1024]; int i,recvbytes; char name[20]; //首次连接时,接受并保存客户端名字 recvbytes = recv(client_fd, name, 20, 0); name[recvbytes]=':'; name[recvbytes+1]='\0'; while(1){ if((recvbytes = recv(client_fd, buf, 1024, 0))==-1){ perror("recv error"); exit(1); } if(recvbytes==0){ printf("%sbye!\n",name); break; } buf[recvbytes]='\0'; for(i=0;i<COUNT;i++){ if(socket_fd[i]==-1){ continue; }else{ message[0]='\0'; strcat(message,name); strcat(message,buf); if(send(socket_fd[i],message,strlen(message),0)==-1){ perror("send error"); exit(1); } } } } //断开时关闭socket,并将描述符值置为-1 close(client_fd); for(i=0;i<COUNT;i++){ if(socket_fd[i]==client_fd){ socket_fd[i]=-1; } } //退出线程 pthread_exit(NULL); } int main(){ //初始化socket数组 int i; for(i=0;i<COUNT;i++){ socket_fd[i]=-1; } pthread_t id; int sockfd,client_fd; socklen_t sin_size; struct sockaddr_in my_addr; struct sockaddr_in remote_addr; if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1){ perror("socket"); exit(1); } //配置信息 my_addr.sin_family=AF_INET; my_addr.sin_port=htons(12345); my_addr.sin_addr.s_addr=INADDR_ANY; bzero(&(my_addr.sin_zero),8); //绑定 if(bind(sockfd,(struct sockaddr *)&my_addr,sizeof(struct sockaddr))==-1){ perror("bind"); exit(1); } //监听 if(listen(sockfd,10)==-1){ perror("listen"); exit(1); } i=0; while(1){ sin_size=sizeof(struct sockaddr_in); if((client_fd=accept(sockfd,(struct sockaddr *)&remote_addr,&sin_size))==-1){ perror("accept"); exit(1); } //找到一个可用的socket位置 while(socket_fd[i]!=-1) i=(i+1)%COUNT; //保存socket并启动线程处理 socket_fd[i]=client_fd; pthread_create(&id,NULL,(void *)pthread_function,(int *)client_fd); } }
客户端:
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/un.h>
#include <sys/time.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <pthread.h>char recv_buf[1500],send_buf[1024];
//线程入口函数,负责显示接收到的信息
void pthread_function(int sockfd){
int recvbytes;
while(1){
if((recvbytes = recv(sockfd, recv_buf, 1500, 0))==-1){
perror("recv error");
exit(1);
}else{
recv_buf[recvbytes]='\0';
printf("%s\n", recv_buf);
}
}
}int main(void){
pthread_t id;
int sockfd;
struct sockaddr_in server_addr;
//参数设置
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(12345);
server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
if((sockfd = socket(AF_INET, SOCK_STREAM, 0))==-1){
perror("socket error");
exit(1);
}
//连接
if(connect(sockfd, (struct sockaddr*)&server_addr,sizeof(server_addr))==-1){
perror("connect error");
exit(1);
}
//输入客户端名字
char name[20];
printf("input your name:");
scanf("%s",name);
send(sockfd,name,strlen(name),0);
pthread_create(&id,NULL,(void *)pthread_function,(int *)sockfd);
while(1){
gets(send_buf);
if(send(sockfd,send_buf,strlen(send_buf),0)==-1){
perror("send error");
exit(1);
}
sleep(1);
}
//关闭socket并