Linux下p2p的聊天功能实现

Linux下p2p的聊天功能实现细节

Do one thing at a time, and do well.

今天闲着没事,写一个P2P的点对点的聊天功能的小程序,我觉得对网络编程初学者的学习很有用的。二话不说,我先贴代码吧。有几个地方需要考虑清楚。我会在代码的后面写出来。代码的下载文章的末尾。

server.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include <signal.h>
#define ERR_EXIT(m)     do     {          perror(m);         exit(EXIT_FAILURE);     } while(0)

void do_something(int conn)
{
     char recvbuf[1024];
     for(;;)
     {
     memset(recvbuf,0,sizeof(recvbuf));
     int ret = read(conn,recvbuf,sizeof(recvbuf));
     if(ret == 0)
     {
         printf("client closed!\n");
         break;
     }
     else if(ret == -1)
     {
         ERR_EXIT("read");
     }
     fputs(recvbuf,stdout);
     write(conn,recvbuf,ret);
     }
}
void handler(int sig)
{
    printf("recv a sig = %d\n",sig);
    exit(EXIT_SUCCESS);
}

int main()
{
    int listenfd;
    if((listenfd = socket(AF_INET,SOCK_STREAM,0)) < 0)
      ERR_EXIT("socket");

    int on = 1;
    int ret = setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on) );

    struct sockaddr_in servaddr;
    memset(&servaddr,0,sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(10001);
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);

    if(bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < 0)
      ERR_EXIT("bind");

    if((listen(listenfd,SOMAXCONN)) < 0 )//主动套接字,变成被动套接字
      ERR_EXIT("listen");

    struct sockaddr_in peeraddr;
    socklen_t socklen = sizeof(peeraddr);
    int conn;
    pid_t  pid;
    if((conn = accept(listenfd,(struct sockaddr*)&peeraddr,&socklen)) < 0)// 获得到是主动套接字
        ERR_EXIT("accept");
     printf("ip:%s   port:%d\n",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port));       

        pid = fork();
        char sendbuf[1024] = {0};
        if(pid == -1) ERR_EXIT("pid");
        if(pid == 0)
        {
            signal(SIGUSR1,handler);
            while(fgets(sendbuf,sizeof(sendbuf),stdin) != NULL)
            {
                write(conn,sendbuf,strlen(sendbuf));
                memset(sendbuf,0,sizeof(sendbuf));
            }
            printf("child closed\n");
            exit(EXIT_SUCCESS);
        }
        else
        {

            char recvbuf[1024];
            while(1)
            {
                memset(recvbuf,0,sizeof(recvbuf));
                int ret = read(conn,recvbuf,sizeof(recvbuf));
                if(ret  == -1)
                  ERR_EXIT("read");
                else if(ret == 0)
                {
                  printf("peer close\n");
                  break;
                }
                fputs(recvbuf,stdout);
            }
            printf("kill parent!\n");
            kill(pid,SIGUSR1);
            exit(EXIT_SUCCESS);
            //do_something(conn);

        }
        close(conn);
        close(listenfd);
        exit(0);
}
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <arpa/inet.h>
#include <signal.h>

#define ERR_EXIT(m)      do      {             perror(m);            exit(EXIT_FAILURE);      } while(0)

void handler(int sig)
{
    printf("recv a sig = %d\n",sig);
    exit(EXIT_SUCCESS);
}

int main(int argc,char *argv[])
{
    int sockfd;
    if((sockfd = socket(AF_INET,SOCK_STREAM,0)) < 0)
      ERR_EXIT("socket");
    struct sockaddr_in servaddr;
    memset(&servaddr,0,sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(10001);
    servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");

    if(connect(sockfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < 0)
      ERR_EXIT("connect");

    char sendbuf[1024] = {0};
    char recvbuf[1024] = {0};
    pid_t pid;
    pid = fork();
    if(pid == -1)
        ERR_EXIT("fork");
    if(pid == 0)
    {
        while(1)
        {
            memset(recvbuf,0,sizeof(recvbuf));
            int ret = read(sockfd,recvbuf,sizeof(recvbuf));
            if(ret == -1)
                ERR_EXIT("read");
            else if (ret == 0)
            {
                printf("peer closed\n");
                break;
            }
            fputs(recvbuf,stdout);
        }

        printf("child close\n");
        kill(pid,SIGUSR1);
        exit(EXIT_SUCCESS);
    }
    else
    {
        signal(SIGUSR1,handler);
        while(fgets(sendbuf,sizeof(sendbuf),stdin) != NULL)
        {
            write (sockfd,sendbuf,strlen(sendbuf));
            memset(sendbuf,0,sizeof(sendbuf));
        }
        printf("parent close!\n");
    }
    close(sockfd);
    exit(0);
}

实现的功能很简单,但是需要注意的几个细节:

1、C和S连接以后,当S关闭后,C仍然没有关闭,我用到了信号的功能。

实现的方法:当父进程关闭的时候,子进程也关闭,当子进程关闭的时候,把父进程也关闭了。

2、一个线程用来监听,一个线程用来等待输入。那么,这是一个多线程的小程序。

  创建一个子进程。C端,子进程进程监听,父进程等待输入。S端相反。

3、信号量的问题:

  SIGUSR1:用户自定义信号量,函数handler用来杀死进程,实现退出。

程序的测试:

总结:总的来说,这个还是很简单的,也就是几个函数是否灵活应用。注意,read函数如果没有读到,就会进入阻塞,如果收到一个0,代表对方关闭了程序,则退出程序。

代码下载:GitHub

声明:水平有限,如果有什么地方写错了或者理解有误,希望广大网友指正。

时间: 2024-10-25 02:40:24

Linux下p2p的聊天功能实现的相关文章

开启Ubuntu Linux下VirtualBox访问USB功能

解决方法如下: 1.增加用户组usbfs sudo groupadd usbfs 2.查看usbfs用户组的gid cat /etc/group | grep usbfs usbfs:x:1002: 3.把当前用户增加到usbfs组 sudo gedit /etc/group 把 usbfs:x:1002: 修改为 usbfs:x:1002:joson(joson改为自己的用户名) 开启Ubuntu Linux下VirtualBox访问USB功能

linux下文件比对功能

很想对吧两个文本有什么不同,可linux下有没有那么方便的工具,怎么办?其实也很简单:diff命令,一行搞定. 新建a.txt文件 新建b.txt文件 执行命令:diff -C0 a.txt b.txt,对比效果如图 *** 1 ***表示第一个文件的第一行 --- 1 --- 表示第二个文件的第一行 以此类推. 当然,该命令还有很多其他功能等待大家去探索.

Linux 下 c 语言 聊天软件

这是我学C语言写的第一个软件,是一个完整的聊天软件,里面包括客户端,和服务器端,可以互现聊天,共享文件,有聊天室等,是一个有TCP和UDP协议的聊天软件,测试过很多次在CENTOS和UBUNTU下都通过,写的简单,但五脏俱全,全部贴出来不保留.运行不了,发信给我,还有个使用指南,我是法语写的,有时间我再写个中文的贴上去.废话不说了,代码来了. #ifndef CHATHEAD_H_ #define CHATHEAD_H_ #include        <arpa/inet.h> #inclu

linux下启用ip转发功能

linux发行版默认情况下是不开启ip转发功能的.这是一个好的做法,因为大多数人是用不到ip转发的,但是如果我们架设一个linux路由或者vpn服务我们就需要开启该服务了.下面我会通过几种方式开通它.检查ip转发是否开启我们需要通过访问sysctl的内核ipv4.ip_forward来判断转发是否开启.使用 sysctl: sysctl net.ipv4.ip_forward net.ipv4.ip_forward = 0或者检查/proc下的文件: cat /proc/sys/net/ipv4

Linux 下常见目录及其功能

1.linux常见的一级目录都是固定的.2.[/bin].[/sbin].[/usr/bin]和[/usr/sbin]四个目录都是用来存放系统命令的.区别是:[/bin]和[/usr/bin]中的命令普通用户和超级用户都能用,而[/sbin]和[/usr/sbin]中的命令只能超级用户使用.3.[boot]目录保存用户的启动数据,不能随便操作,一般都单独分区,否则如果写满了,系统会无法启动.4.[dev]里面都是硬件文件,不要随便乱动.5.[etc]保存系统默认配置文件的目录.6.[home]普

Linux下提权常用小命令

有些新手朋友在拿到一个webshell后如果看到服务器是Linux或Unix操作系统的就直接放弃提权,认为Linux或Unix下的提权很难,不是大家能做的,其实Linux下的提权并没有很多人想象的那么难,你真去尝试做了,也许你就会发现Linux下的提权并不难,尤其是一些简单的提权方法是很容易学会的.Linux下的提权我知道的比较简单的方法都是在命令行下完成的,很多新手叉子可能根本没接触过Linux下的一些常用命令,今天危险漫步就给大家介绍一些Linux下提权过程中常用到的Linux命令,由于我也

windows/Linux下的程序员文档浏览工具

Dash + Alfred https://www.jianshu.com/p/77d2bf8df81f 对于程序员来说,查看api文档是非常频繁,经常窗口之间切换非常麻烦,mac下就有一个查文档的神器:Dash. 详细使用可以参考如上连接的说明. 本文重点整理下windows和linux下使用类似的功能如何操作. IntelliJ插件 https://github.com/gdelmas/IntelliJDashPlugin/releases ShortcutYou can change th

Linux下C语言多线程,网络通信简单聊天程序

原文:Linux下C语言多线程,网络通信简单聊天程序 功能描述:程序应用多线程技术,可是实现1对N进行网络通信聊天.但至今没想出合适的退出机制,除了用Ctr+C.出于演示目的,这里采用UNIX域协议(文件系统套接字),程序分为客户端和服务端.应用select函数来实现异步的读写操作. 先说一下服务端:首先先创建套接字,然后绑定,接下进入一个无限循环,用accept函数,接受“连接”请求,然后调用创建线程函数,创造新的线程,进入下一个循环.这样每当有一个新的“连接”被接受都会创建一个新的线程,实现

rlwrap: command not found和解决linux下sqlplus 提供浏览历史命令行的功能

rlwrap工具可以解决linux下sqlplus 提供浏览历史命令行的功能,和删除先前输入错误的字母等问题 1.安装 需要readline包 这个安装光盘就有 [[email protected] RedHat]# cd RPMS/[[email protected] RPMS]# rpm -Uvh readline*warning: readline-4.3-13.i386.rpm: V3 DSA signature: NOKEY, key ID db42a60eerror: Failed