inet_ntop():把IPV4/IPV6的网络字节序的地址变成本地的字符串形式的IP地址
1.socket()函数
文件(内核的缓冲区)操作
socket tcp server
创建套接字
int lfd = socket
2.bind()函数
绑定本地IP和端口
struct sockaddr_in serv;
serv.port = htons(port)
serv.IP = htonl(INADDR_ANY);
bind();
3.listen函数
4.accept(): 阻塞等待客户端的连接,成功时返回已经建立好连接的新的newfd
5.connect() : 客户端的连接函数
connect()函数和服务器的bind()函数写法类似:
内核中的服务器的套接字fd会维护2个链表:
1.正在三次握手的客户端链表(数量 = 2*backlog+1)
2.已经建立好连接的客户端链表(已经完成3次握手分配newfd)
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netint/ip.h>
#include <errno.h>
#define SERV_PORT 5001
#define SERV_IP_ADDR "192.168.7.246" //服务器地址
#define BACKLOG 5 //backlog同时允许几路客户端和服务器进行正在连接的过程(正在三次握手)一般填5,测试得知,ARM最大为8
#define QUIT_STR "quit"
int main()
{
int fd =-1;
struct socke_in sin;
/*1.创建socket fd*/
if((fd = socket(AF_INET,SOCK_STREAM,0))<0)
{
perror("socket");
exit(1);
}
/*2绑定*/
/*2.1填充struct sockaddr_in结构体变量*/
bzero(&sin,sizeof(sin));
sin.sin_family = AF_INET; //IPV4
sin.sin_port = htons(SERV_PORT); //网络字节序的端口号
/*优化1:让服务器程序能绑定在任意的IP上*/
#if 1
sin.sin_addr.s_addr = hton1(INADDY_ANY); //点分形式的IP地址,结果为#32整数
#else
if(inet_pton(AF_INET,SERV_IP_ADDR,(void *)&sin.sin_addr.s_addr)!=1)
{
perror("inet_pton");
exit(1);
}
#endif
/*2.2绑定*/
if(bind(fd,(struct sockaddr *)&sin,sizeof(sin)) < 0 )
{
perror("bind");
exit(1);
}
//如果是IPV6编程呢个,要使用struct sockaddr_in6结构体(详细情况请参考man 7 ipv6 ),通常更通用的方法可以通过struct sockaddr_storage来编程
/*3.调用listen()把主动套接字变成被动套接字*/
if(listen(fd,BACKLOG)<0)
{
perror("listen");
exit(1);
}
/*4.阻塞等待客户端连接请求*/
#if 0
newfd = accept(fd,NULL,NULL);
if(newfd < 0)
{
perrror("accept");
exit(1);
}
#else
/*优化2:通过程序获取刚建立连接的socket的客户端的IP地址和端口号*/
struct sockaddr_in cin;
socklen_t addrlen = sizeof(cin);
if((newfd = accept(fd,(struct sockaddr *)&cin,&addrlen))< 0)
{
perror("accept");
exit(1);
}
char ipv4_addr[16];
if(!inet_ntop(AF_INET,(void * )&cin.sin_addr,ipv4_addr,sizeof(cin))){
perror("inet_ntop");
exit(1);
}
printf("client(%s:%d) is connected!",ipv4_addr,ntons(cin.sin_port));
#endif
/*5.读写*/
//和newfd进行数据读写
char buf[BUFSIZ];
int ret =-1;
bzero(buf,BUFSIZ);
while(1){
do{
ret = read(newfd,buf,BUFSIZ-1);
}while(ret<0 &&EINTER == errno);
if(ret<0)
{
perror("read");
exit(1);
}
if(!ret){ //对方已经关闭了
break;
}
printf("receive data: %s\n",buf);
if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR),strlen(QUIT_STR)))//用户输入了quit字符
{
printf("client is exiting !\n");
break;
}
}
close(newfd);
close(fd);
return 0;
}
client代码
int main(void)
{
int fd = -1;
struct sockaddr_in sin;
/*1.创建socket fd*/
if((fd = socket(AF_INET,SOCK_STREAM,0))<0)
{
perror("socket");
}
/*2.连接*/
/*2.1填充struct sockaddr_in结构体变量*/
bzero(&sin,sizeof(sin));
sin.sin_family = AF_INET; //IPV4
sin.sin_port = htons(SERV_PORT); //网络字节序的端口号
#if 0
sin.sin_addr.s_addr = inet_addr(SERV_IP_ADDR); //点分形式的IP地址,结果为#32整数
#else
if(inet_pton(AF_INET,SERV_IP_ADDR,(void *)&sin.sin_addr.s_addr)!=1)
{
perror("inet_pton");
exit(1);
}
#endif
if(connect(fd,(struct sockaddr *)&sin,sizeof(sin)<0)
{
perror("connect");
exit(1);
}
/*3.读写数据*/
char buf[BUFSIZ];
while(1){
bzero(buf,BUFSIZ);
if(fgets(buf,BUFSIZ-1,stdin) == NULL)
{
continue;
}
write(fd,buf,strlen(buf));
if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
{
printf("client is exiting!\n");
break;
}
}
}
///client_tcp.c///
#include <sys/types.h>
#include <sys/stat.h>
#include <string.h>
#include <arpa/inet.h>
int main(int argc,const char* argv[])
{
if(argc < 2)
{
printf("eg:/a.out port\n");
exit(1);
}
int port = atoi(argv[1]);
//1.创建套接字
int fd = socket(AF_INF,SOCK_STREAM,0);
//连接服务器
struct sockaddr_in serv;
memset(&serv,0,sizeof(serv));
serv.sin_family = AF_INET;
serv.sin_port = htons(port);
inet_pton(AF_INET,"127.0.0.1",&serv.sin_addr.s_addr);
connect(fd,(struct sockaddr*)&serv,sizeof(serv));
//通信
while(1)
{
char buf[1024];
printf("请输入要发送的字符串:\n");
fgets(buf,sizeof(buf),stdin);
write(fd,buf,strlen(buf));
//等待接受数据
int len = read(fd,buf,sizeof(buf));
if(len == -1)
{
perror("read error");
exit(1);
}
else if(len == 0)
{
printf("服务器端关闭了连接\n");
break;
}
else
{
printf("recv buf: \n");
}
}
}