linux 的网络编程(一)

来源:互联网 发布:刚开的淘宝店没生意怎么办 编辑:程序博客网 时间:2024/04/29 11:10

                决定在把linux网络编程好好的扎扎实实看一遍,环境,putty跑服务端,虚拟机上redhat7.0的shell的netstat查看,windows的7个cmd跑客户端。经过测试之后对基本的这些函数有了更深的了解。还涉及到很多的细节,socket(),bind(),listen(),accept(),connect(),inet_aton()函数等等就不说了,没意思,网上很多文章都写这个的用法的。自己还认为的网络中头文件不太熟悉,也就不投机取巧直接封装在一个文件里面,手动一个一个敲上去。下面这个两个代码极为相似,主要差别就是inet_pton()和inet_aton(),一个也就是另外一个的改进版仅此而已。该程序验证listen(int fd,int backlog)中backlog的对listen系统调用的影响。


#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <sys/socket.h>#include <arpa/inet.h>#include <signal.h>#include <netinet/in.h>#include <assert.h>#include <string.h>#define false (0)#define true  (1)static int stop  = false;static void handle_term(int sig){     stop = true;}int main(int argc,char *argv[]){    signal(SIGTERM,handle_term);    if(argc <= 3){        fprintf(stderr,"Usage:%s ip_address  port_number backlog\n",argv[0]);        exit(1);    }    const char *ip = argv[1];    int port = atoi(argv[2]);    int backlog = atoi(argv[3]);    int sock = socket(PF_INET,SOCK_STREAM,0);    assert(sock >= 0);    struct sockaddr_in address;    bzero(&address,sizeof(address));    address.sin_family = AF_INET;    inet_pton(AF_INET,ip,&address.sin_addr);    address.sin_port = htons(port);    int ret = bind(sock,(struct sockaddr*)(&address),sizeof(address));    assert(ret != -1);    ret = listen(sock,backlog);    assert(ret != -1);    while( !stop){        sleep(1);    }    close(sock);    return 0;}



两个代码都放上去吧,运行结果也都一样

#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <sys/socket.h>#include <arpa/inet.h>#include <signal.h>#include <netinet/in.h>#include <assert.h>#include <string.h>#define false (0)#define true  (1)static int stop  = false;static void handle_term(int sig){     stop = true;}int main(int argc,char *argv[]){    signal(SIGTERM,handle_term);    if(argc <= 3){        fprintf(stderr,"Usage:%s ip_address  port_number backlog\n",argv[0]);        exit(1);    }    const char *ip = argv[1];    int port = atoi(argv[2]);    int backlog = atoi(argv[3]);    int sock = socket(AF_INET,SOCK_STREAM,0);    assert(sock >= 0);    struct sockaddr_in address;    bzero(&address,sizeof(address));    address.sin_family = AF_INET;    inet_aton(ip,&address.sin_addr);    address.sin_port = htons(port);    int ret = bind(sock,(struct sockaddr*)(&address),sizeof(address));    assert(ret != -1);    ret = listen(sock,backlog);    assert(ret != -1);    while( !stop){        sleep(1);    }    close(sock);    return 0;}


运行结果:






最多连接6个客户端也就是backlog+1处于ETABLISHED的状态,后面来连接的都将处于SYN_RECV,当客户端断开连接的时候,将会出现CLOSE_WAIT状态。因为一方断开连接了。



  在接收连接之后也就是accept从listen监听队列中拿出一个连接,这个程序测试一个异常连接情况下服务端状态图的变化,我们在不同的时间断开客户端telnet,这个需要ctrl+],关闭连接一定要quit,我在这里调试很长时间才知道quit真正的断开和服务端的连接,最后两种情况下ESTABLISHED和CLOASE_WAIT终于出来了。

#include <sys/socket.h>#include <netinet/in.h>#include <arpa/inet.h>#include <assert.h>#include <stdio.h>#include <unistd.h>#include <stdlib.h>#include <errno.h>#include <string.h>int main(int argc,char *argv[]){    if(argc <= 2){         printf("Usage:%s ip_tables port_number\n",argv[0]);         exit(0);    }    const char *ip = argv[1];    int port  = atoi(argv[2]);    struct sockaddr_in address;    bzero(&address,sizeof(address));    address.sin_family = AF_INET;    inet_pton(AF_INET,ip,&address.sin_addr);    address.sin_port = htons(port);    int sock = socket(PF_INET,SOCK_STREAM,0);    assert(sock>=0);    int ret = bind(sock,(struct sockaddr*)(&address),sizeof(address));    assert(ret != -1);    ret = listen(sock,5);    assert(ret != -1);    sleep(20);    struct sockaddr_in client;    socklen_t client_addrlength = sizeof(client);    int connfd = accept(sock,(struct sockaddr *)(&client),&client_addrlength);    if(connfd < 0){         printf("errno is :%d\n",errno);    }else{        char remote[INET_ADDRSTRLEN];        printf("connected with ip:%s and port :%d\n",inet_ntop(AF_INET,&client.sin_addr,remote,INET_ADDRSTRLEN),ntohs(client.sin_port));        sleep(10);        close(connfd);    }    close(sock);    return 0;}

客户端继续用windows下的telnet 192.168.111.111 54321


这个是服务端的截图


..


 上面有吐核的错误是因为刚刚关闭服务端端运行完程序,立即用开始执行程序。因为虽然程序执行完,并没有全部关闭端口和缓存,处于TIME_WAIT状态,这段时间持续大概两分钟。因此在bind的哪里出现断言错误。直接assert出来了,因此两分钟之后再次执行就不会出现错误了。也可以用setsocketopt,一般来说,一个端口释放后会等待两分钟之后才能再被使用,SO_REUSEADDR是让端口释放后立即就可以被再次使用。SO_REUSEADDR用于对TCP套接字处于TIME_WAIT状态下的socket,才可以重复绑定使用。server程序总是应该在调用bind()之前设置SO_REUSEADDR套接字选项。TCP,先调用close()的一方会进入TIME_WAIT状态。

int  ret,sock_reuse=1; 
ret=setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,(char *)&sock_reuse,sizeof(sock_reuse));

0 0