1. 概述:
该demo主要实现了linux下通过select(tcp)方式的socket并发通讯,相关接口介绍可以参考<<UNIX环境高级编程>>
2. 场景(服务端(一)<----(多)客户端):
1) 服务端:
输入:服务端监听的port;
接收到客户端发送的消息,打印到终端,并且回复客户端"ok"
2) 客户端:
输入:客户端的port,以及服务端ip和port;
输入消息内容,将消息发送给服务端,阻塞直到接收到服务端的返回信息,并打印到终端(超时未接收到时,报错);
当输入消息内容为"exit"时,选择退出客户端进程
3. 测试:
SELECT(TCP) 服务端
客户端1
客户端2
/*
demo_select_client.c
网络编程demo(select客户端)
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "lpm.h"
void signal_func() {
printf(RED BLINK"Server Close\n\n"NONE);
}
int main(int argc, char **argv){
int ret;
int client_fd = -1;
char client_port[8];
char server_ip[32];
char server_port[8];
char buf[MAX_BUF];
printf(RED BLINK"SELECT 客户端\n"NONE);
printf("请输入客户端 port\t");
scanf("%s", client_port);
printf("请输入服务端 ip \t");
scanf("%s", server_ip);
printf("请输入服务端 port\t");
scanf("%s", server_port);
printf("\n");
/*
SIGPIPE : 在reader中止之后写Pipe的时候发送
SIG_DFL : 默认信号处理程序
SIG_IGN : 忽略信号的处理程序
signal(SIGPIPE,SIG_IGN);
对一个对端已经关闭的socket调用两次读/写, 第二次将会生成SIGPIPE信号, 该信号默认结束进程
*/
if(SIG_ERR == signal(SIGPIPE,signal_func)){
perror("signal:");
return -1;
}
/*
协议域 : AF_INET(IPV4) AF_INET6(IPV6) AF_LOCAL AF_ROUTE 等等
socket类型 : SOCK_STREAM(TCP) SOCK_DGRAM(UDP) 等等
0 : 表示协议域和socket类型选择默认的协议, 通常是0
*/
client_fd = socket(AF_INET, SOCK_STREAM, 0);
if(client_fd < 0){
/*printf("socket:%s\n", strerror(errno));*/
perror("socket:");
return -1;
}
/*
应用协议 : SOL_SOCKET(套接字) IPPROTO_TCP IPPROTO_IP...
设置项 : SO_REUSEADDR(是否可以重用bind的地址)...
opt : 一些设置项指的是开关...
*/
int opt = 1;
ret = setsockopt(client_fd, SOL_SOCKET, SO_REUSEADDR, (void *)&opt, sizeof(opt));
if(ret < 0){
perror("setsockopt");
close(client_fd);
return -1;
}
/*
设置项 : SO_RCVTIMEO(为recv设置超时时间)...
*/
struct timeval tv;
tv.tv_sec = RECV_TIME_OUT;
tv.tv_usec = 0;
ret = setsockopt(client_fd, SOL_SOCKET, SO_RCVTIMEO, (void *)&tv, sizeof(tv));
if(ret < 0){
perror("setsockopt");
close(client_fd);
return -1;
}
struct sockaddr_in client_addr;
memset(&client_addr, 0x0, sizeof(struct sockaddr_in));
client_addr.sin_family = AF_INET;
client_addr.sin_port = htons(atoi(client_port));
client_addr.sin_addr.s_addr = htonl(INADDR_ANY);
ret = bind(client_fd, (struct sockaddr*)&client_addr, sizeof(client_addr));
if(ret < 0){
perror("bind:");
close(client_fd);
return -1;
}
struct sockaddr_in server_addr;
memset(&server_addr, 0x0, sizeof(struct sockaddr_in));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(atoi(server_port));
inet_pton(AF_INET, server_ip, (void *)&(server_addr.