代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <poll.h>
#define ERR_MSG(msg) do{\
fprintf(stderr, "line:__%d__ ", __LINE__);\
perror(msg);\
}while(0)
#define SER_PORT 8089 //服务器的端口
#define SER_IP "192.168.8.168" //服务器绑定的IP
int main(int argc, const char *argv[])
{
//创建流式套接字
int cfd = socket(AF_INET, SOCK_STREAM, 0);
if(cfd < 0)
{
ERR_MSG("socket");
return -1;
}
//填充要连接的服务器的IP和端口
struct sockaddr_in sin;
sin.sin_family = AF_INET; //必须填AF_INET;
sin.sin_port = htons(SER_PORT); //服务器绑定的端口号
sin.sin_addr.s_addr = inet_addr(SER_IP); //服务器绑定的IP地址
//连接服务器
if(connect(cfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)
{
ERR_MSG("connect");
return -1;
}
printf("connect server success\n");
//定义要监测的文件描述符集合
struct pollfd fds[2] = {0};
fds[0].fd = 0; //制定监测0号文件描述符 用于终端输入 并将消息发送给服务器
fds[0].events = POLLIN; //监测0号的读事件
fds[1].fd = cfd; //制定监测cfd 用于接收服务器消息
fds[1].events = POLLIN; //监测文件描述符的读事件
char buf[128];
ssize_t res = 0;
int p_res = 0;
while(1)
{
p_res = poll(fds, 2, -1);
if(p_res < 0)
{
ERR_MSG("poll");
return -1;
}
else if(0 == p_res)
{
printf("time out ,,,,\n");
break;
}
//能运行到当前位置,则代表集合中有文件描述符触发事件
//需要判断集合中的文件描述符的revents中有没有POLLIN事件
//实际产生的revents中可能有多个事件,POLLIN只是其中一个事件
//所以需要通过按位与的方式,提取出POLLIN事件,判断有无POLLIN事件
if(fds[0].revents & POLLIN)
{
//触发键盘输入事件
bzero(buf, sizeof(buf)); //绑定客户端自身的地址信息结构体--->非必须绑定,可以选择不绑
//如果选择不绑定,则操作系统会默认绑定一个端口号给这个客户端
fgets(buf, sizeof(buf), stdin);
buf[strlen(buf)-1] = 0;
if(send(cfd, buf, sizeof(buf), 0) < 0)
{
ERR_MSG("send");
return -1;
}
printf("send success\n");
}
if(fds[1].revents & POLLIN)
{
bzero(buf, sizeof(buf));
res = recv(cfd, buf, sizeof(buf), 0);
if(res < 0)
{
ERR_MSG("recv");
return -1;
}
else if(0 == res)
{
printf("服务器端关闭\n");
break;
}
printf(":%s\n", buf);
}
}
//关闭文件描述符
close(cfd);
return 0;
}