Socket TCP【简单聊天实例】

本文介绍了一个简单的TCP客户端与服务器通信示例,通过创建套接字、绑定端口、监听及接收客户端请求等步骤实现双向通信,并使用多线程分别处理客户端连接及消息接收。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Server端:

#include <iostream> 
#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 <pthread.h>  
#include <unistd.h>  
#include <arpa/inet.h>  

#define MYPORT 3490  //定义端口  
#define BACKLOG 10  
#define MAXDATASIZE 1024  
  
int sockfd,new_fd;  
pthread_t accthread,recthread;
void* recmessage(void*){  //接收客户端信息函数  
    while(1){  
        int numbytes;  
        char buf[MAXDATASIZE];  
        if((numbytes = recv(new_fd,buf,MAXDATASIZE,0))==-1){  
            perror("recv");  
            exit(1);  
        }  
        buf[numbytes]='\0';  
        if(strcmp(buf,"exit")==0){  //若收到的是exit字符,则代表退出通信  
            printf("Client is closed\n");  
            close(new_fd);  
            close(sockfd);  
            exit(1);  
        }  
        printf("client:%s\n",buf);  
    }/*while*/  
}  
void* acceptconnect(void*){  //接受客户端连接请求函数  
    struct sockaddr_in their_addr;  
    socklen_t sin_size = sizeof(struct sockaddr_in);  
    // if (( accept_fd = accept(socket_fd, (struct sockaddr *) &remote_addr, &sin_size)) == -1 )
    if((new_fd = accept(sockfd,(struct sockaddr *)&their_addr,&sin_size))==-1){  
        perror("accept");  
        exit(1);  
    }  
    printf("server:got connection from %s\n",inet_ntoa(their_addr.sin_addr));  
    /*创建子线程,用于接收信息*/  
    if((pthread_create(&recthread,NULL,recmessage,NULL)) != 0){  
        printf("create thread error!\r\n");  
        exit(1);  
    }
}
int main(void){  
    struct sockaddr_in my_addr;  
    /*创建套接字*/  
    if((sockfd = socket(AF_INET,SOCK_STREAM,0)) == -1){  
        perror("socket");  
        exit(1);  
    }  
    /*初始化sockaddr_in结构体相关参数*/  
    my_addr.sin_family = AF_INET;  
    my_addr.sin_port = htons(MYPORT);  
    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,BACKLOG)== -1){  
        perror("listen");  
        exit(1);  
    }  
    printf("listening...\n");  
    /*创建子线程,用于接收信息*/  
     if((pthread_create(&accthread,NULL,acceptconnect,NULL))!=0){  
        printf("create thread error!\n");  
        exit(1);  
    }  
    while(1){  
        char msg[MAXDATASIZE];  
        scanf("%s",msg);  
        if(send(new_fd,msg,strlen(msg),0) == -1){  //发送信息,与客户端交流  
            perror("send");  
            exit(1);  
        }  
        if(strcmp(msg,"exit") ==0){  
            printf("byebye\n");  
            close(new_fd);  
            close(sockfd);  
            exit(1);  
        }  
    }/*while*/  
    return 0;  
}/*main*/  

Client端:

#include <iostream>  
#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 <pthread.h>  
#include <netdb.h>  
#include <unistd.h>  
#include <arpa/inet.h>

#define PORT 3490  
#define BACKLOG 10  
#define MAXDATASIZE 1024 
#define SERVERIP "127.0.0.1"
int sockfd;  
pthread_t recthread;  
  
/*接收信息函数*/  
void* recmessage(void*){  
    while(1){  
        int numbytes;  
        char buf[MAXDATASIZE];  
          
        if((numbytes = recv(sockfd,buf,MAXDATASIZE,0))==-1){  
            perror("recv");  
            exit(1);  
        }  
        buf[numbytes]='\0';  
        if(strcmp(buf,"exit")==0){  
            printf("Server is closed\n");  
            close(sockfd);  
            exit(1);  
        }  
        printf("Server:%s\n",buf);  
    }/*while*/  
}  
int main(){  
    struct hostent *he;  /*获取主机IP地址*/  
    struct sockaddr_in their_addr;  

    /*获取主机IP地址*/  
    if((he = gethostbyname(SERVERIP)) == NULL){  
        herror("gethostbyname");  
        exit(1);  
    }  
    /*创建套接字*/  
    if((sockfd = socket(AF_INET,SOCK_STREAM,0)) == -1){  
        perror("socket");  
        exit(1);  
    }  
     /*初始化sockaddr_in结构体*/  
    their_addr.sin_family = AF_INET;  
    their_addr.sin_port = htons(PORT);  
    their_addr.sin_addr = *((struct in_addr *)he->h_addr);  
    bzero(&(their_addr.sin_zero),8);  
    /*向服务器发送连接请求*/  
    if(connect(sockfd,(struct sockaddr *)&their_addr,sizeof(struct sockaddr)) == -1){  
        perror("connect");  
        exit(1);  
    }  
    /*创建子线程,用于接收信息*/  
    if((pthread_create(&recthread,NULL,recmessage,NULL))!= 0){  
        printf("create thread error!\r\n");  
        exit(1);  
    }  
    /*发送信息。接收发送信息用的是同一端口,都 是sockfd*/  
    while(1){  
        char msg[MAXDATASIZE];  
        scanf("%s",msg);  
        if(send(sockfd,msg,strlen(msg),0) == -1){  
            perror("send");  
            exit(1);  
        }  
        if(strcmp(msg,"exit") ==0){  
            printf("byebye\n");  
            close(sockfd);  
            exit(1);  
        }  
    }/*while*/  
    return 0;  
}  



里面包含聊天室的客户端和服务器端的源文件和一份完整的设计报告。 一、 系统概要 本系统能实现基于VC++的网络聊天室系统。有单独的客户端、服务器端。 服务器应用程序能够接受来自客户端的广播,然后向客户端发送本机的IP与服务端口,让客户端接入到服务器进行聊天,检测用户名是否合法(重复),服务器责接收来自客户端的聊天信息,并根据用户的需求发送给指定的人或所有人,能够给出上线下线提示。客户端能够发出连接请求,能编辑发送信息,可以指定发给单人或所有人,能显示聊天人数,上线下线用户等。 二、 通信规范的制定 服务请求规范: 服务器端: (1) 创建一个UDP的套接字,接受来自客户端的广播请求,当请求报文内容为“REQUEST FOR IP ADDRESS AND SERVERPORT”时,接受请求,给客户端发送本服务器TCP聊天室的端口号。 (2) 创建一个主要的TCP协议的套接字负责客户端TCP连接 ,处理它的连接请求事件。 (3)在主要的TCP连接协议的套接字里面再创建TCP套接字保存到动态数组里,在主要的套接字接受请求后 ,就用这些套接字和客户端发送和接受数据。 客户端: (1) 当用户按“连接”按钮时,创建UDP协议套接字,给本地计算机发广播,广播内容为“REQUEST FOR IP ADDRESS AND SERVERPORT”。 (2)当收到服务器端的回应,收到服务器发来的端口号后,关闭UDP连接。根据服务器的IP地址和端口号重新创建TCP连接。 故我思考:客户端一定要知道服务器的一个端口,我假设它知道服务器UDP服务的端口,通过发广播给服务器的UDP服务套接字,然后等待该套接字发回服务器TCP聊天室服务的端口号,IP地址用ReceiveForom也苛刻得到。 通信规范 通信规范的制定主要跟老师给出的差不多,并做了一小点增加: (增加验证用户名是否与聊天室已有用户重复,在服务器给客户端的消息中,增加标志0) ① TCP/IP数据通信 --- “聊天”消息传输格式 客户机 - 服务器 (1)传输“用户名” STX+1+用户名+ETX (2) 悄悄话 STX+2+用户名+”,”+内容+ETX (3) 对所有人说 STX+3+内容+ETX 服务器- 客户机 (0)请求用户名与在线用户名重复 //改进 STX+0+用户名+EXT (1)首次传输在线用户名 STX+1+用户名+ETX (2)传输新到用户名 STX+2+用户名+ETX (3)传输离线用户名 STX+3+用户名+ETX (4)传输聊天数据 STX+4+内容+ETX (注:STX为CHR(2),ETX 为CHR(3)) 三、 主要模块的设计分析 四、 系统运行效果 (要求有屏幕截图) 五、 心得与体会
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值