基于tpc协议的服务器框架——简单的服务器收发

在开始之前,先做几个准备工作

定义答应错误信息的宏函数,包括错误原因以及行号

//打印错误信息的宏函数
#define ERR_MSG(msg) do{\
	fprintf(stderr, "__%d__", __LINE__);\
	perror(msg);\
}while(0)

宏定义网络端口以及IP地址

#define PORT 8888      //自行制定,从1024~49151中选择一个
#define IP "192.168.x.x"          //本机ip,用ifconfig查看,此处隐去

 服务器原理图如下 

 

服务器流式套接字类似于“掮客”(bushi),它可以对接客户端的套接字,然后同时生成一个新的套接字与之对应。我们首先就是要创建套接字

/*************************************************************************/
/******************************以下摘自man手册*****************************/

socket函数
功能:指定传输层网络层的协议,并创建一块缓冲区,获取这块缓冲区对应的文件描述符;
头文件:
       #include <sys/types.h>          /* See NOTES */
       #include <sys/socket.h>

       int socket(int domain, int type, int protocol);
参数:
    int domain:Linux支持的协议族(地址族);
       Name                Purpose                          Man page
       AF_UNIX, AF_LOCAL   Local communication              unix(7)
       AF_INET             IPv4 Internet protocols          ip(7)
       AF_INET6            IPv6 Internet protocols          ipv6(7);

	int type:指定套接字类型
       SOCK_STREAM:字节流式套接字-->TCP : 默认是IPPROTO_TCP协议
       SOCK_DGRAM:数据报式套接字--->UDP 	默认是IPPROTO_UDP协议
	   SOCK_RAW:原始套接字,需要在第三个参数中手动指定协议,

    int protocol:0,默认协议;

返回值:
    成功,返回套接字对应的文件描述符;
	失败,返回-1,更新errno;

 流式套接字创建

//创建流式套接字
int sfd = socket(AF_INET, SOCK_STREAM, 0);
if(sfd < 0)
{
	ERR_MSG("socket");
	return -1;
}
printf("create socket success\n");

填充地址信息结构体,然后将地址信息结构体绑定到套接字上

    //填充地址信息结构体,真实的地址信息结构体与协议族相关
	//AF_INET,所以请翻看man 7 ip
	struct sockaddr_in sin;
	sin.sin_family             =AF_INET;
	sin.sin_port               =htons(PORT); //网络字节序的端口号
	sin.sin_addr.s_addr        =inet_addr(IP);     //网络字节序的ip
    
    //将地址信息结构体绑定到套接字上
	if( bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0 )
	{
		ERR_MSG("bind");
		return -1;
	}
    printf("bind success\n");

其中用到了一些系统自带的函数

/*********************************************************************/
htons  htonl   主机字节序---->网络字节序
原型:
       uint32_t htonl(uint32_t hostlong);
       uint16_t htons(uint16_t hostshort);
参数:
    uint32_t hostlong:32位主机字节序整型;
	uint16_t hostshort:16位主机字节序整型; 

 以及点分十进制与网络字节序的转换

原型:
       in_addr_t inet_addr(const char *cp);
参数:
     char *cp:源IP地址的点分十进制字符串,例如 “192.168.1.10”;
返回值:
    成功,返回转换后的网络字节序IP地址;
					typedef uint32_t in_addr_t;

	失败,返回INADDR_NONE (usually -1);

只能转换IPv4;

 把套接字转化为被动监听状态

//将套接字设置为被动监听状态,让内核取监听是否有客户端链接
	if(listen(sfd, 10) < 0)
	{
		ERR_MSG("listen");
		return -1;
	}
	printf("listen success\n");
    struct  sockaddr_in  cin;
	socklen_t  addrlen = sizeof(cin);
	//从已完成链接的队列头中,取出一个客户端的信息
	//创建生成一个信道套接字文件描述符
	//该文件描述符才是与客户端通信的文件描述符!!!
	int newfd = accept(sfd, (struct sockaddr*)&cin, &addrlen);
	if(newfd < 0)
	{
		perror("accept");
		return -1;
	}
//把网络字节序的IP-->点分十进制  网络字节顺序-->主机字节序
printf("[%s : %d] newfd = %d\n", inet_ntoa( cin.sin_addr ),\
         ntohs( cin.sin_port ), newfd);

要完成服务器的收发,需要多线程或者多进程,不然,会导致卡死在读或者写上

创建线程

//创建线程
	pthread_t tid;
	if(pthread_create(&tid, NULL, mysend, &newfd) != 0)
	{
		perror("pthread_create");
		return -1;
	}

主线程用来接收信息

char buf[128] = "";
ssize_t res = 0;	
while(1)
{
	bzero(buf, sizeof(buf));
	//循环接收
	res = recv(newfd, buf, sizeof(buf), 0);
	if(res < 0)
	{
		ERR_MSG("recv");
		return -1;
	}else if(0 == res)
	{
		printf("\n[%s : %d] newfd = %d客户端退出\n", \
              inet_ntoa( cin.sin_addr ), ntohs( cin.sin_port ), newfd);
		break;
	}
    //输出接收到的信息
	printf("\n[%s : %d] newfd = %d : %s\n",\
           inet_ntoa( cin.sin_addr ), ntohs( cin.sin_port ), newfd, buf);
    //为了代码运行的美观性加一句
	fprintf(stderr,"请输入消息内容>>>");
}

子线程用来处理发送消息

//发送线程
void *mysend(void *arg)
{   //接收newfd
	int *p = (int*)arg; 
	int newfd = *p;
	char buf[128] = "";
	ssize_t res;
	while(1)
	{
		//清零
		bzero(buf, sizeof(buf));
		//循环发送
		fprintf(stderr,"请输入消息内容>>>");
		fgets(buf, sizeof(buf)-1, stdin);   //从终端获取字符
		res = send(newfd, buf, sizeof(buf), 0);
		if(res < 0)
		{
			ERR_MSG("send");
			return NULL;
		}else if(0 == res)
		{
			break;
		}
	}
}

附上完整代码

#include<stdio.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<netinet/in.h>
#include<unistd.h>
#include<pthread.h>

//打印错误信息的宏函数
#define ERR_MSG(msg) do{\
	fprintf(stderr, "__%d__", __LINE__);\
	perror(msg);\
}while(0)

#define PORT 8888      //自行制定,从1024~49151中选择一个
#define IP "192.168.1.106"          //本机ip,用ifconfig查看

//发送线程
void *mysend(void *arg)
{
	int *p = (int*)arg;
	int newfd = *p;
	char buf[128] = "";
	ssize_t res;
	while(1)
	{
		//清零
		bzero(buf, sizeof(buf));
		//循环发送
		fprintf(stderr,"请输入消息内容>>>");
		fgets(buf, sizeof(buf)-1, stdin);
		res = send(newfd, buf, sizeof(buf), 0);
		if(res < 0)
		{
			ERR_MSG("send");
			return NULL;
		}else if(0 == res)
		{
			break;
		}
	}
}

int main(int argc, const char *argv[])
{
	//创建流式套接字
	int sfd = socket(AF_INET, SOCK_STREAM, 0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}
	printf("create socket success\n");
	
	//填充地址信息结构体,真实的地址信息结构体与协议族相关
	//AF_INET,所以请翻看man 7 ip
	struct sockaddr_in sin;
	sin.sin_family             =AF_INET;
	sin.sin_port               =htons(PORT); //网络字节序的端口号
	sin.sin_addr.s_addr        =inet_addr(IP);     //网络字节序的ip

	//将地址信息结构体绑定到套接字上
	if( bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0 )
	{
		ERR_MSG("bind");
		return -1;
	}
    printf("bind success\n");

	//将套接字设置为被动监听状态,让内核取监听是否有客户端链接
	if(listen(sfd, 10) < 0)
	{
		ERR_MSG("listen");
		return -1;
	}
	printf("listen success\n");
	
	struct  sockaddr_in  cin;
	socklen_t  addrlen = sizeof(cin);
	//从已完成链接的队列头中,取出一个客户端的信息
	//创建生成一个信道套接字文件描述符
	//该文件描述符才是与客户端通信的文件描述符!!!
	int newfd = accept(sfd, (struct sockaddr*)&cin, &addrlen);
	if(newfd < 0)
	{
		perror("accept");
		return -1;
	}
	
	//把网络字节序的IP-->点分十进制  网络字节顺序-->主机字节序
	printf("[%s : %d] newfd = %d\n", inet_ntoa( cin.sin_addr ), ntohs( cin.sin_port ), newfd);
	
	//创建线程
	pthread_t tid;
	if(pthread_create(&tid, NULL, mysend, &newfd) != 0)
	{
		perror("pthread_create");
		return -1;
	}

	char buf[128] = "";
	ssize_t res = 0;
	while(1)
	{
		bzero(buf, sizeof(buf));
		//循环接收
		res = recv(newfd, buf, sizeof(buf), 0);
		if(res < 0)
		{
			ERR_MSG("recv");
			return -1;
		}else if(0 == res)
		{
			printf("\n[%s : %d] newfd = %d客户端退出\n", inet_ntoa( cin.sin_addr ), ntohs( cin.sin_port ), newfd);
			break;
		}
		printf("\n[%s : %d] newfd = %d : %s\n", inet_ntoa( cin.sin_addr ), ntohs( cin.sin_port ), newfd, buf);
		fprintf(stderr,"请输入消息内容>>>");
	}
	
    
	close(sfd);
	close(newfd);

	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

老K殿下

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值