要求
利用select函数实现在Linux环境下实现一个聊天室程序,要求:
- 用户默认处于广播模式,一个客户在其客户端发送的消息,其它客户端用户全部可以收到;
- 程序支持下列命令
/help:显示帮助信息(思考:信息是放在客户端还是服务器端);
/quit:用户退出聊天室,同时将退出信息广播给其他用户;
/who:显示在线用户;
/send 用户名 消息:向指定用户发送点到点消息 - 程序退出时清理所有的占用资源,包括内存资源、套接字等
- 支持最大连接数限制;
- 能够管理用户加入和退出。
知识点
- strlen()
计算给定字符串的(unsigned int型)长度,不包括’\0’在内 - sizeof()
返回一个对象或者类型所占的内存字节数
代码
/*chat_server.c*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netdb.h>
#include <sys/time.h>
#include <sys/types.h>
#include <arpa/inet.h>
#define SERVERPORT 1573
/*maximum cnnnect number*/
#define BACKLOG 10
/*maximum data buffer*/
#define BUFSIZE 2048
struct client_info
{
int client_id;
struct sockaddr_in client_address;
char usrname[256];//客户的名字
int first;//用于只是用户是否刚刚登陆
};
int main()
{
struct timeval time;
time.tv_sec=1;
time.tv_usec=0;
/*master file descriptor list*/
fd_set master_fds;
/*temple file descriptor list for select()*/
fd_set read_fds;
/*server address*/
struct sockaddr_in server_addr;
/*client address*/
struct sockaddr_in client_addr;
/*maximum file descriptor number*/
int max_fd;
/*listening socket file descriptor*/
int sockfd;
/*newly accept()ed socket file descreptor*/
int newfd;
/*buffer for saving client data*/
char data_buf[BUFSIZE];
char send_buf[BUFSIZE];
/*number of client data*/
int nbytes;
/*for set socket option*/
int opt;
opt = SO_REUSEADDR;
/*lenth of address*/
int addr_len;
/*for accept() to use*/
int size;
size = sizeof(struct sockaddr);
/*temple varient*/
int tmp_i, tmp_j;
struct client_info clientinfo[BACKLOG];
/*clear the master and temple file descriptor*/
FD_ZERO(&master_fds);
FD_ZERO(&read_fds);
memset(&data_buf, 0, BUFSIZE);
memset(&send_buf, 0, BUFSIZE);
/*create socket*/
if (-1 == (sockfd = socket(AF_INET, SOCK_STREAM, 0)))
{
perror("create socket() error:");
exit(1);
}
/*set the socket*/
if (-1 == setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)))
{
perror("setsockopt() error:");
exit(1);
}
/*bind first config the socket then binding*/
memset(&server_addr, 0, size);
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(SERVERPORT);
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
bzero(&(server_addr.sin_zero), 8);
if (-1 == bind(sockfd, (struct sockaddr*)&server_addr, sizeof(struct sockaddr)))
{
perror("bind() socket error:");
exit(1);
}
/*listen */
if (-1 == listen