学习目标:
-
1、编写TCP多线程服务器
-
2、编写TCP多进程并发服务器
网络编程基础:
TCP:(Transfer Control protocol,传输控制协议)提供面向连接的,一对一的可靠传输的协议
数据无误,数据不丢失,数据无失序
适用场景:
适合对传输质量要求较高,以及传输大量数据的通信
在需要可靠数据传输的场合,通常适用TCP协议
MSN/QQ等即时通讯软件的用户登录账号管理相关的功能通常采用TCP协议
UDP:(user Datagram Protocol,用户数据报协议):提供不可靠,无连接的传输协议
适用场景发送小尺寸数据(如对DNS服务器进行IP地址查询时) 在接收数据,给出应答较困难的网络中适用UDP(如无线网络) 适合于广播/组播式通信中 MSN/QQ/Skype等即时通讯软件的点对点文本通讯以及音视频通常采用UDP协议 流媒体,VOD,VoIP,IPTV等网络多媒体服务中通常采用UDP方式进行实时数据传输
字节转换函数:
主机字节序到网络字节序:
u_long htonl(u_long hostlong);
u_short htons(u_short short);
网络字节序到主机字节序:
u_long ntohl(u_long hostlong);
u_short ntohs(u_short short);
IP地址的转换:
inet_aton()
将strptr所指的字符串转换成32位的网络字节序二进制值
inet_addr()
功能同上,返回转换后的地址
仅适用于IPV4,出错时返回-1。
局限性:不能用于255.255.255.255的转换
inet_ntoa()将32位网络字节序二进制地址转换成点分十进制的字符串
inet_pton()
int inet_pton(int af, const char* src ,void* dst)
将IPV4/IPV6 的地址转换成binary格式
使用于IPV4/IPV6
能正确处理255.255.255.255的转换问题
参数:
1.地址协议族(AF_INET或AF_INET6)
2.src:是一个指针(填写分点形式的IP地址(主要指IPV4))
3.dst:转换的结果给到dst
inet_ntop(int af,const void *src,char *dst,socklen_t size)
把ipv4和ipv6的网络字节序变成本地的字符串形式的IP地址
参数:
1.af:地址协议族(AF_INET或AF_INET6)
2.src:是一个指针(32)
3.dst:输出结果为32位点分形式的IP地址
4.size:长度
熟悉TCP编程API
1.socket函数
int socket(int domain,int type,int protocol);
参数
domain:
AF_INET
AF_INET6
AF_UNIX,AF_LOCAL
AF_NETLINK
AF_PACKET
type:
SOCK_STREAM: 流式套接字,唯一对应于TCP
SOCK_DGRAM:数据报套接字,唯一对应着UDPSOCK_RAW:原始套接字
protocol:
一般填0,原始套接字编程时需填充
返回值:
成功返回文件描述符
出错返回-1
如果是IPV6编程,要使用struct sockddr_in6结构体(man 7 IPV6),通常使用struct sockaddr_storage来编程。
2.bind函数
int bind(int sockfd,const struct sockaddr *addr,socklen_t addrlen)
参数:
sockfd:通过socket()函数拿到的fd
addr:采用struct sockaddr的结构体地址,通用结构体
struct sockaddr{
sa_family_t sa_family;
char sa_data[4];
}
// 基于Internel通信结构体
struct sockaddr_in{
as_family_t sin_family;
in_port_t sin_port;
struct in_addr sin_addr;
sin_zero , //填充字节,需清零
}
struct in_addr{
uint32_t s_addr;
}
addrlen:地址长度
3.listen()函数
int listen(int sockfd,int backlog);
参数:
sockfd: 通过socket()函数拿到的fd;
backLog:同时允许几路客户端和服务器进行正在连接的过程(正在三次握手),一般填5。
内核中服务器的套接字fd会维护2个链表
1.正在三次握手的客户端链表(数量=2*backlog+1)
2.已经建立好连接的客户端链表(已经完成三次握手分配好了的newfd)
返回值:
成功返回0; 出错返回-1
listen(fd,5);//表示系统允许11(2*5+1)个客户端同时进行三次握手
4.accept()函数
阻塞等待客户端连接请求
int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
参数:
sockfd:经过前面socket()创建并通过bind(),listen()设置过的fd
addr:指向存放地址信息的结构体的首地址
获取客户端IP地址和端口号
addrlen:存放地址信息的结构体的大小
返回值:
成功,返回返回已经建立连接的新的newfd
出错,返回-1
5.客户端连接函数connect()
int connect (int sockfd, struct sockaddr * serv_addr, int addrlen)
参数:
sockfd:通过socket()函数拿到的fd
addr:struct sockaddr的结构体变量地址
addrlen:地址长度
返回值:
成功,返回0
失败,返回-1
TCP编程基本流程:
TCP实现并发服务器的基本流程:
TCP实现多进程并发服务器
服务端:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<strings.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<pthread.h>
#include<signal.h>
#include<sys/wait.h>
#define BUFSIZE 1024
#define QUIT_STR "QUIT"
#define SERV_IP 5001
#define SERV_IP_ADDR "192.168.224.132"
#define BACKLOG 5
void *client_data_handle( void *arg)
{
//强转为int型的指针,对它再取值
int newfd=*(int *) arg;
int ret = -1;
char buf[BUFSIZE];
printf(" child handle progess : newfd =%d\n",newfd);
while(1)
{
do
{
bzero(buf,BUFSIZE);
ret=read(newfd,buf,BUFSIZE-1);
}while(ret < 1);
if ( ! ret)
{
break;
}
printf(" recevice data :%s\n",buf);
//两个字符串做比较,不一样返回0,一样返回1,则执行下面的内容
if( ! strncasecmp(buf,QUIT_STR,strlen(QUIT_STR))) {
printf(" Client is exiting !\n");
break;
}
}
close(newfd);
return NULL;
}
void child_data_handle(int signum)
{
if( SIGCHLD == signum)
{
waitpid(-1,NULL,WNOHANG);
}
}
int main()
{
printf(" this is service!\n");
int fd = -1;
signal(SIGCHLD,child_data_handle);
struct sockaddr_in sin;
//1.socket
fd=socket(AF_INET,SOCK_STREAM,0);
if( fd <0)
{
perror("socket");
exit(1);
}
//把结构体清零
bzero(&sin,sizeof(sin));
sin.sin_family=AF_INET;
sin.sin_port=htons(SERV_IP);//两个字节
/*
inet_addr(),返回转换后的地址,仅适用于IPV4,出错时返回-1
将IPV4/IPV6 的地址转换成binary格式
inet_pton(AF_INET,SERV_IP_ADDR ,(void*) sin.sin_addr.s_addr);
两种方法都可以
*/
sin.sin_addr.s_addr=inet_addr(SERV_IP_ADDR);
//优化,可以在任何的服务器中运行,自动获取当前的ip地址
//sin.sin_addr.s_addr=INADDR_ANY;
//2.bind
if(bind(fd,(struct sockaddr *)&sin,sizeof(sin))<0)
{
perror("bind");
exit(1);
}
//3.listen
if(listen(fd,BACKLOG)<0)
{
perror("listen");
exit(1);
}
//4.acccept
//进程
int newfd = -1;
pid_t pid;
struct sockaddr_in cin;
socklen_t addrlen =sizeof(cin);
while(1)
{
newfd=accept(fd,NULL,NULL);
if( newfd <0)
{
perror("acccept");
exit(1);
}
pid =fork();
if( pid <0)
{
perror("fork");
break;
}
if ( pid == 0)
{
char ipv4_addr[16];
if(! inet_ntop(AF_INET,(void *)&cin.sin_addr,ipv4_addr,sizeof(cin)))
{
perror("inet_ntop");
exit(1);
}
//获取端口号和ip地址
printf("client : (%s , %d) is connect !\n", ipv4_addr,ntohs(cin.sin_port));
client_data_handle(&newfd);
close(fd);
}
if( pid > 0)
{
close(newfd);
}
}
close(fd);
return 0;
}
客户端:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<strings.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#define BUFSIZE 1024
#define QUIT_STR "QUIT"
#define SERV_PORT 5001
#define SERV_IP_ADDR "192.168.224.132"
int main(int argc , char **argv)
{
int fd = -1;
if( argc != 3)
{
exit(1);
}
int port = -1;
port=atoi(argv[2]);
struct sockaddr_in sin;
//1.socket
fd=socket(AF_INET,SOCK_STREAM,0);
if( fd <0)
{
perror("socket");
exit(1);
}
//把结构体清零
bzero(&sin,sizeof(sin));
sin.sin_family=AF_INET;
sin.sin_port=htons(port);
sin.sin_addr.s_addr=inet_addr(argv[1]);
if(connect(fd,(struct sockaddr *)&sin,sizeof(sin)) < 0)
{
perror("connect");
exit(1);
}
char buf[BUFSIZE];
while(1)
{
bzero(buf,BUFSIZE);
if(fgets(buf ,BUFSIZE-1,stdin) == NULL)
{
continue;
}
write(fd,buf,strlen(buf));
if( !strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
{
break;
}
}
return 0;
}
实现效果:
TCP实现多线程并发服务器:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<strings.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<pthread.h>
#include<signal.h>
#include<sys/wait.h>
#define BUFSIZE 1024
#define QUIT_STR "QUIT"
#define SERV_IP 5001
#define SERV_IP_ADDR "192.168.224.132"
#define BACKLOG 5
void *client_data_handle( void *arg)
{
//强转为int型的指针,对它再取值
int newfd=*(int *) arg;
int ret = -1;
char buf[BUFSIZE];
printf(" pthread progess : newfd =%d\n",newfd);
while(1)
{
do
{
bzero(buf,BUFSIZE);
ret=read(newfd,buf,BUFSIZE-1);
}while(ret < 1);
if(ret < 0)
{
exit(1);
}
if ( ! ret)
{
break;
}
printf(" recevice data :%s\n",buf);
//两个字符串做比较
if( ! strncasecmp(buf,QUIT_STR,strlen(QUIT_STR))) {
printf(" Client is exiting !\n");
break;
}
}
close(newfd);
return NULL;
}
void child_data_handle(int signum)
{
if( SIGCHLD == signum)
{
waitpid(-1,NULL,WNOHANG);
}
}
int main()
{
printf(" this is service!\n");
int fd = -1;
signal(SIGCHLD,child_data_handle);
struct sockaddr_in sin;
//1.socket
fd=socket(AF_INET,SOCK_STREAM,0);
if( fd <0)
{
perror("socket");
exit(1);
}
//把结构体清零
bzero(&sin,sizeof(sin));
sin.sin_family=AF_INET;
sin.sin_port=htons(SERV_IP);//两个字节
sin.sin_addr.s_addr=inet_addr(SERV_IP_ADDR);
//优化,可以在任何的服务器中运行,自动获取当前的ip地址
//sin.sin_addr.s_addr=INADDR_ANY;
//2.bind
if(bind(fd,(struct sockaddr *)&sin,sizeof(sin))<0)
{
perror("bind");
exit(1);
}
//3.listen
if(listen(fd,BACKLOG)<0)
{
perror("listen");
exit(1);
}
//4.acccept
/*
int newfd = -1;
newfd=accept(fd,NULL,NULL);
if( newfd <0)
{
perror("acccept");
exit(1);
}
*/
//优化,服务端知道哪个客户端连接了,并可以获取端口号和ip地址
/*
每接收一个就打印它的客户信息,并创建一个线程,并跳转到线程这个函数里面去
*/
int newfd = -1;
pthread_t tid;
struct sockaddr_in cin;
socklen_t addrlen =sizeof(cin);
while(1)
{
newfd=accept(fd,(struct sockaddr *)&cin,&addrlen);
if( newfd <0)
{
perror("acccept");
exit(1);
}
char ipv4_addr[16];
if(! inet_ntop(AF_INET,(void *)&cin.sin_addr,ipv4_addr,sizeof(cin)))
{
perror("inet_ntop");
exit(1);
}
//获取端口号和ip地址
printf("client : (%s , %d) is connect !\n", ipv4_addr,ntohs(cin.sin_port));
pthread_create(&tid,NULL,client_data_handle,(void *)&newfd);
}
close(fd);
return 0;
}
服务端和TCP实现多进程并发服务器的服务端代码一致