socket 实现的简单的p2p服务程序


p2pclient.c


#include <unistd.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <signal.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

/*
 int socket(int domain, int type, int protocol);
 */

/*
 int bind(int sockfd, const struct sockaddr *my_addr, socklen_t addrlen);

 //man 2 bind
 struct sockaddr {
 sa_family_t sa_family;
 char        sa_data[14];
 }

 //man 7 ip
 struct sockaddr_in {
 sa_family_t    sin_family; // address family: AF_INET
 u_int16_t      sin_port;   // port in network byte order
 struct in_addr sin_addr;   // internet address
 };

 //Internet address.
 struct in_addr {
 u_int32_t      s_addr;     //address in network byte order
 };

 */

//int listen(int sockfd, int backlog);
//accept 接受一个新的连接 ,这个新的连接是一个主动套接字
/*   int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
 On success, accept() returns a non-negative integer that is a descriptor for the accepted socket.   On  error,
 -1 is returned, and errno is set appropriately.
 int conn = 0;
 */

int main()
{
    int sockfd = 0;
    sockfd = socket(PF_INET, SOCK_STREAM, 0);
    if (sockfd == -1)
    {
        perror("fun socket\n");
        exit(0);
    }
    //设置客服端链接的tcpip结构体信息
    struct sockaddr_in srvaddr;
    //协议族
    srvaddr.sin_family = AF_INET;
    //设置端口
    srvaddr.sin_port = htons(8001);
    //设置ip
    srvaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //127.0.0.1
    //srvaddr.sin_addr.s_addr = inet_addr(INADDR_ANY); //绑定本机的任意一个地址
    //客服端链接
    if (connect(sockfd, (struct sockaddr*) (&srvaddr), sizeof(srvaddr)) < 0)
    {
        perror("fun socket\n");
        exit(0);
    }

    pid_t pid = fork();
    if(pid == -1)
    {
        perror("fork");
        return -1;
    }
    else if(pid == 0)
    {
        //child process,read process
        char revbuf[1024] = { 0 };
        while(1)
        {
            int ret = read(sockfd, revbuf, sizeof(revbuf));
            if (ret == 0)
            {
                //如果在读的过程中,对方已经关闭,tcpip协议会返回一个0数据包
                printf("对方已关闭\n");
                break;
            }
            else if (ret < 0)
            {
                perror("读数据失败\n");
                break;
            }
            fputs(revbuf, stdout); //服务器端收到数据,打印屏幕
            memset(revbuf,0x00,sizeof(revbuf));
        }

    }
    else
    {
        char sendbuf[1024] = { 0 };
        while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)
        {
            write(sockfd, sendbuf, strlen(sendbuf)); //服务器端回发信息
            memset(sendbuf, 0, sizeof(sendbuf));
        }
    }
    close(sockfd);
    return 0;
}

p2pserver.c


#include <unistd.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <signal.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

/*
 int socket(int domain, int type, int protocol);
 */

/*
 int bind(int sockfd, const struct sockaddr *my_addr, socklen_t addrlen);

 //man 2 bind
 struct sockaddr {
 sa_family_t sa_family;
 char        sa_data[14];
 }

 //man 7 ip
 struct sockaddr_in {
 sa_family_t    sin_family; // address family: AF_INET
 u_int16_t      sin_port;   // port in network byte order
 struct in_addr sin_addr;   // internet address
 };

 //Internet address.
 struct in_addr {
 u_int32_t      s_addr;     //address in network byte order
 };

 */

//int listen(int sockfd, int backlog);
/*   int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
 On success, accept() returns a non-negative integer that is a descriptor for the accepted socket.   On  error,
 -1 is returned, and errno is set appropriately.
 int conn = 0;
 */
int main()
{
    int sockfd = 0;
    //创建socket
    sockfd = socket(PF_INET, SOCK_STREAM, 0);
    if (sockfd == -1)
    {
        perror("fun socket\n");
        exit(0);
    }
    //定义socket结构体
    struct sockaddr_in srvaddr;
    //设置协议族
    srvaddr.sin_family = AF_INET;
    //设置端口
    srvaddr.sin_port = htons(8001);
    //增加ip地址
    srvaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //127.0.0.1
    //srvaddr.sin_addr.s_addr = inet_addr(INADDR_ANY); //绑定本机的任意一个地址
    //绑定IP
    if (bind(sockfd, (struct sockaddr *) &srvaddr, sizeof(srvaddr)) < 0)
    {
        perror("fun bind\n");
        exit(0);
    }

    //一但调用listen函数,这个套接字sockfd将变成被动套接字;只能接受连接,不能主动的发送连接
    //listen 做了两个队列。。。。。。
    // 队列由内核管理,一部分是完成三次握手的,一部分是没有完成三次握手的。
    if (listen(sockfd, SOMAXCONN) < 0)
    {
        perror("fun listen\n");
        exit(0);
    }

    //struct sockaddr peeraddr;  //通用ip
    //socklen_t perrlen ;
    struct sockaddr_in peeraddr; //tcpip地址结构
    socklen_t peerlen = sizeof(peeraddr);

    unsigned int conn = 0;
    //accept接受已经完成三次握手的链接,没有链接会阻塞直到有链接
    conn = accept(sockfd, (struct sockaddr *) &peeraddr,
            (socklen_t *) &peerlen);
    if (conn == -1)
    {
        perror("fun listen\n");
        exit(0);
    }

    pid_t pid = fork();
    if(pid == -1)
    {
        perror("fork");
        return -1;
    }
    else if(pid == 0)
    {
        //this is child process
        //child read
        char revbuf[1024] = { 0 };
        while(1)
        {
            int ret = read(conn, revbuf, sizeof(revbuf));
            if (ret == 0)
            {
                //如果在读的过程中,对方已经关闭,tcpip协议会返回一个0数据包
                printf("对方已关闭\n");
                break;
            }
            else if (ret < 0)
            {

                perror("读数据失败\n");
                break;
            }
            fputs(revbuf, stdout); //服务器端收到数据,打印屏幕
            memset(revbuf,0x00,sizeof(revbuf));
        }

    }
    else
    {
        //this is parent process
        //parent process write
        char sendbuf[1024] = { 0 };
        while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)
        {
            //向服务写数据
            //ssize_t write(int fd, const void *buf, size_t count);
            // 从buf中读取count大小的数据存入文件描述符为fd的文件中。
            write(conn, sendbuf, strlen(sendbuf)); //服务器端回发信息
            memset(sendbuf, 0, sizeof(sendbuf));
        }
    }

    close(conn);
    close(sockfd);

    return 0;
}


  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值