一、循环服务器
伪代码:
sfd
=
socket
();
bind
();
listen
();
while
(
1
)
{
newfd
=
accept
();
while
(
1
)
{
recv
();
send
();
}
close
(
newfd
);
}
close
(
sfd
);
代码实现:
#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<string.h>
#include<netinet/in.h>
//打印错误信息
#define ERR_MSG(msg) do{\
fprintf(stderr,"__%d__",__LINE__); \
perror(msg);}while(0)
#define PORT 8888 //1024——49151
#define IP "192.168.31.73" //本机IP
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");
//允许端口快速重用
int reuse = 1;
if(setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&reuse,sizeof(reuse)) < 0)
{
ERR_MSG("setsockopt");
return -1;
}
//填充地址信息结构体,真实的地址信息结构体与协议族相关
//AF_INET
struct sockaddr_in sin;
sin.sin_family = AF_INET;
sin.sin_port = htons(PORT); //网络字节序的端口号
sin.sin_addr.s_addr = inet_addr(IP); //网络字节序的IP地址
//将地址信息结构体绑定到套接字上
bind(sfd,(struct sockaddr *)&sin,sizeof(sin));
if(bind < 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);
while(1)
{
//从已经完成链接的队列头中,取出一个客户端的信息,创建生成一套新的套接字文件描述符
//该文件描述符才是与客户端通信的文件描述符
int newfd = accept(sfd,(struct sockaddr*)&cin,&addrlen);
if(newfd < 0)
{
ERR_MSG("accept");
return -1;
}
//网络字节序的IP--->点分十进制,网络字节序的poet---->本机字节序
printf("[%s : %d] newfd = %d\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd);
char buf[128] = "";
ssize_t res = 0;
while(1)
{
bzero(buf,sizeof(buf)