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

Do one thing at a time, and do well.

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

server.c

  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <sys/socket.h>
  4. #include <stdlib.h>
  5. #include <netinet/in.h>
  6. #include <arpa/inet.h>
  7. #include <unistd.h>
  8. #include <string.h>
  9. #include <signal.h>

  10. #define ERR_EXIT(m) \
  11. do \
  12. { \
  13. perror(m); \
  14. exit(EXIT_FAILURE); \
  15. } while()
  16.  
  17. void do_something(int conn)
  18. {
  19. char recvbuf[];
  20. for(;;)
  21. {
  22. memset(recvbuf,,sizeof(recvbuf));
  23. int ret = read(conn,recvbuf,sizeof(recvbuf));
  24. if(ret == )
  25. {
  26. printf("client closed!\n");
  27. break;
  28. }
  29. else if(ret == -)
  30. {
  31. ERR_EXIT("read");
  32. }
  33. fputs(recvbuf,stdout);
  34. write(conn,recvbuf,ret);
  35. }
  36. }
  37. void handler(int sig)
  38. {
  39. printf("recv a sig = %d\n",sig);
  40. exit(EXIT_SUCCESS);
  41. }
  42.  
  43. int main()
  44. {
  45. int listenfd;
  46. if((listenfd = socket(AF_INET,SOCK_STREAM,)) < )
  47. ERR_EXIT("socket");
  48.  
  49. int on = ;
  50. int ret = setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on) );
  51.  
  52. struct sockaddr_in servaddr;
  53. memset(&servaddr,,sizeof(servaddr));
  54. servaddr.sin_family = AF_INET;
  55. servaddr.sin_port = htons();
  56. servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  57.  
  58. if(bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < )
  59. ERR_EXIT("bind");
  60.  
  61. if((listen(listenfd,SOMAXCONN)) < )//主动套接字,变成被动套接字
  62. ERR_EXIT("listen");
  63.  
  64. struct sockaddr_in peeraddr;
  65. socklen_t socklen = sizeof(peeraddr);
  66. int conn;
  67. pid_t pid;
  68. if((conn = accept(listenfd,(struct sockaddr*)&peeraddr,&socklen)) < )// 获得到是主动套接字
  69. ERR_EXIT("accept");
  70. printf("ip:%s port:%d\n",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port));
  71.  
  72. pid = fork();
  73. char sendbuf[] = {};
  74. if(pid == -) ERR_EXIT("pid");
  75. if(pid == )
  76. {
  77. signal(SIGUSR1,handler);
  78. while(fgets(sendbuf,sizeof(sendbuf),stdin) != NULL)
  79. {
  80. write(conn,sendbuf,strlen(sendbuf));
  81. memset(sendbuf,,sizeof(sendbuf));
  82. }
  83. printf("child closed\n");
  84. exit(EXIT_SUCCESS);
  85. }
  86. else
  87. {
  88.  
  89. char recvbuf[];
  90. while()
  91. {
  92. memset(recvbuf,,sizeof(recvbuf));
  93. int ret = read(conn,recvbuf,sizeof(recvbuf));
  94. if(ret == -)
  95. ERR_EXIT("read");
  96. else if(ret == )
  97. {
  98. printf("peer close\n");
  99. break;
  100. }
  101. fputs(recvbuf,stdout);
  102. }
  103. printf("kill parent!\n");
  104. kill(pid,SIGUSR1);
  105. exit(EXIT_SUCCESS);
  106. //do_something(conn);
  107.  
  108. }
  109. close(conn);
  110. close(listenfd);
  111. exit();
  112. }
  1. #include <stdio.h>
  2. #include <sys/socket.h>
  3. #include <sys/types.h>
  4. #include <netinet/in.h>
  5. #include <string.h>
  6. #include <unistd.h>
  7. #include <stdlib.h>
  8. #include <arpa/inet.h>
  9. #include <signal.h>
  10.  
  11. #define ERR_EXIT(m) \
  12. do \
  13. { \
  14. perror(m); \
  15. exit(EXIT_FAILURE); \
  16. } while()
  17.  
  18. void handler(int sig)
  19. {
  20. printf("recv a sig = %d\n",sig);
  21. exit(EXIT_SUCCESS);
  22. }
  23.  
  24. int main(int argc,char *argv[])
  25. {
  26. int sockfd;
  27. if((sockfd = socket(AF_INET,SOCK_STREAM,)) < )
  28. ERR_EXIT("socket");
  29. struct sockaddr_in servaddr;
  30. memset(&servaddr,,sizeof(servaddr));
  31. servaddr.sin_family = AF_INET;
  32. servaddr.sin_port = htons();
  33. servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
  34.  
  35. if(connect(sockfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < )
  36. ERR_EXIT("connect");
  37.  
  38. char sendbuf[] = {};
  39. char recvbuf[] = {};
  40. pid_t pid;
  41. pid = fork();
  42. if(pid == -)
  43. ERR_EXIT("fork");
  44. if(pid == )
  45. {
  46. while()
  47. {
  48. memset(recvbuf,,sizeof(recvbuf));
  49. int ret = read(sockfd,recvbuf,sizeof(recvbuf));
  50. if(ret == -)
  51. ERR_EXIT("read");
  52. else if (ret == )
  53. {
  54. printf("peer closed\n");
  55. break;
  56. }
  57. fputs(recvbuf,stdout);
  58. }
  59.  
  60. printf("child close\n");
  61. kill(pid,SIGUSR1);
  62. exit(EXIT_SUCCESS);
  63. }
  64. else
  65. {
  66. signal(SIGUSR1,handler);
  67. while(fgets(sendbuf,sizeof(sendbuf),stdin) != NULL)
  68. {
  69. write (sockfd,sendbuf,strlen(sendbuf));
  70. memset(sendbuf,,sizeof(sendbuf));
  71. }
  72. printf("parent close!\n");
  73. }
  74. close(sockfd);
  75. exit();
  76. }

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

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

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

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

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

3、信号量的问题:

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

程序的测试:

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

代码下载:GitHub

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

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

  1. 实现Linux下不间断聊天和退出处理

    实现Linux下不间断聊天和退出处理

  2. linux下P2P协议(BitTorrent)-libtorrent库编译,测试

    1.libtorrent 简介,下载和编译 libtorrent简介 libtorrent是功能齐全的C ++ bittorrent的p2p协议实现,专注于效率和可伸缩性.它可以在嵌入式设备和台式机上 ...

  3. linux下文件比对功能

    很想对吧两个文本有什么不同,可linux下有没有那么方便的工具,怎么办?其实也很简单:diff命令,一行搞定. 新建a.txt文件

  4. Linux下启用IP转发功能(主要针对Ubuntu的使用)

    说明:以下的操作只要在Linux下都是通用的. Linux发行版默认情况下是不开启IP转发功能的.如果架设一个Linux路由或者VPN服务就需要开启该服务. 1.通过访问sysctl的内核ipv4.i ...

  5. windows、linux 下启用mysql日志功能

    在默认情况下,mysql安装是没有启用日志管理功能的,这为后续的维护带来很多不便的地方. 查看是否启用了日志mysql>show variables like 'log_bin'; 怎样知道当前 ...

  6. 在Linux下编译带调试功能的Bochs

    在Linux下使用Bochs参考: http://wangcong.org/articles/bochs.html http://kinglaw05.blog.163.com/blog/static/ ...

  7. Linux 下 c 语言 聊天软件

    这是我学C语言写的第一个软件,是一个完整的聊天软件,里面包括客户端,和服务器端,可以互现聊天,共享文件,有聊天室等,是一个有TCP和UDP协议的聊天软件,测试过很多次在CENTOS和UBUNTU下都通 ...

  8. Linux下Apache配置HTTPS功能

    Apache配置HTTPS功能  转 https://www.cnblogs.com/liaojiafa/p/6028816.html 一.yum 安装openssl和openssl-devel,ht ...

  9. Linux下gsoap实现webservice功能

    蓝字为关键字,等号=后面为关键字值. 一.介绍 我们用的webservice是根据gsoap编译工具来实现,gSOAP的编译器能够自动的将用户定义的本地化的C或C++数据类型转变为符合XML语法的数据 ...

随机推荐

  1. PetaPoco 增删改查

    1 查询单行 DBInstance.DB.SingleOrDefault<CompanyInfo11>(id); /// <summary> /// 根据id获取公司信息 // ...

  2. SQL SERVER 查看死锁的存储过程

    end

  3. 灵活运用绑定变量---declare匿名块使用绑定变量

    declare        type cur01 is ref cursor;     v_cur cur01;        v_match123 varchar2(2000);        v ...

  4. Objective-C中的分类与协议

    分类 在谈分类之前,我们可以先探究下,OC中为什么出现分类这种机制,有什么好处? 假设你接到一个大项目:计算两个整数的和,差.接到任务的你马上动手.编写代码如下: #import <Founda ...

  5. Chrome调试(转)

    原文地址:http://blog.csdn.net/chenmoquan/article/details/44943245#comments 觉得写的很适合web开发的新手 Chrome 的开发者工具 ...

  6. spring + maven +testng 测试常见依赖包问题

    java.lang.ClassNotFoundException: org.apache.commons.dbcp.BasicDataSource解决方法:添加缺少的jar包:commons-coll ...

  7. php数组(array)输出三种形式

    $bbbb=array("11"=>"aaa","22"=>"bbb"); //只能输出值value不能输出 ...

  8. prototype/constructor/__proto__之prototype

    1任何对象都有__proto__属性 属性值Object2并不是所有对象都有prototype属性.只有方法对象(构造函数)以及基本数据类型还有Array,有prototype属性;并且所有方法(对象 ...

  9. PHP随机生成指定时间段的指定个数时间

    /** * 生成某个范围内的随机时间 * @param <type> $begintime 起始时间 格式为 Y-m-d H:i:s * @param <type> $endt ...

  10. Git error- fatal- Needed a single revision

    最近在开发中由于项目结构的重构,有一部分代码被抽出来作为了公共库(git submodule),这样公共库可以独立维护,同时其他库调用它也是非常方便的,避免了到处复制代码的痛苦. 但我在项目重构后第一 ...