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()
- void do_something(int conn)
- {
- char recvbuf[];
- for(;;)
- {
- memset(recvbuf,,sizeof(recvbuf));
- int ret = read(conn,recvbuf,sizeof(recvbuf));
- if(ret == )
- {
- printf("client closed!\n");
- break;
- }
- else if(ret == -)
- {
- 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,)) < )
- ERR_EXIT("socket");
- int on = ;
- int ret = setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on) );
- struct sockaddr_in servaddr;
- memset(&servaddr,,sizeof(servaddr));
- servaddr.sin_family = AF_INET;
- servaddr.sin_port = htons();
- servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
- if(bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < )
- ERR_EXIT("bind");
- if((listen(listenfd,SOMAXCONN)) < )//主动套接字,变成被动套接字
- 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)) < )// 获得到是主动套接字
- ERR_EXIT("accept");
- printf("ip:%s port:%d\n",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port));
- pid = fork();
- char sendbuf[] = {};
- if(pid == -) ERR_EXIT("pid");
- if(pid == )
- {
- signal(SIGUSR1,handler);
- while(fgets(sendbuf,sizeof(sendbuf),stdin) != NULL)
- {
- write(conn,sendbuf,strlen(sendbuf));
- memset(sendbuf,,sizeof(sendbuf));
- }
- printf("child closed\n");
- exit(EXIT_SUCCESS);
- }
- else
- {
- char recvbuf[];
- while()
- {
- memset(recvbuf,,sizeof(recvbuf));
- int ret = read(conn,recvbuf,sizeof(recvbuf));
- if(ret == -)
- ERR_EXIT("read");
- else if(ret == )
- {
- 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();
- }
- #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()
- 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,)) < )
- ERR_EXIT("socket");
- struct sockaddr_in servaddr;
- memset(&servaddr,,sizeof(servaddr));
- servaddr.sin_family = AF_INET;
- servaddr.sin_port = htons();
- servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
- if(connect(sockfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < )
- ERR_EXIT("connect");
- char sendbuf[] = {};
- char recvbuf[] = {};
- pid_t pid;
- pid = fork();
- if(pid == -)
- ERR_EXIT("fork");
- if(pid == )
- {
- while()
- {
- memset(recvbuf,,sizeof(recvbuf));
- int ret = read(sockfd,recvbuf,sizeof(recvbuf));
- if(ret == -)
- ERR_EXIT("read");
- else if (ret == )
- {
- 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,,sizeof(sendbuf));
- }
- printf("parent close!\n");
- }
- close(sockfd);
- exit();
- }
实现的功能很简单,但是需要注意的几个细节:
1、C和S连接以后,当S关闭后,C仍然没有关闭,我用到了信号的功能。
实现的方法:当父进程关闭的时候,子进程也关闭,当子进程关闭的时候,把父进程也关闭了。
2、一个线程用来监听,一个线程用来等待输入。那么,这是一个多线程的小程序。
创建一个子进程。C端,子进程进程监听,父进程等待输入。S端相反。
3、信号量的问题:
SIGUSR1:用户自定义信号量,函数handler用来杀死进程,实现退出。
程序的测试:
总结:总的来说,这个还是很简单的,也就是几个函数是否灵活应用。注意,read函数如果没有读到,就会进入阻塞,如果收到一个0,代表对方关闭了程序,则退出程序。
代码下载:GitHub
声明:水平有限,如果有什么地方写错了或者理解有误,希望广大网友指正。
Linux下p2p的聊天功能实现的更多相关文章
- 实现Linux下不间断聊天和退出处理
实现Linux下不间断聊天和退出处理
- linux下P2P协议(BitTorrent)-libtorrent库编译,测试
1.libtorrent 简介,下载和编译 libtorrent简介 libtorrent是功能齐全的C ++ bittorrent的p2p协议实现,专注于效率和可伸缩性.它可以在嵌入式设备和台式机上 ...
- linux下文件比对功能
很想对吧两个文本有什么不同,可linux下有没有那么方便的工具,怎么办?其实也很简单:diff命令,一行搞定. 新建a.txt文件
- Linux下启用IP转发功能(主要针对Ubuntu的使用)
说明:以下的操作只要在Linux下都是通用的. Linux发行版默认情况下是不开启IP转发功能的.如果架设一个Linux路由或者VPN服务就需要开启该服务. 1.通过访问sysctl的内核ipv4.i ...
- windows、linux 下启用mysql日志功能
在默认情况下,mysql安装是没有启用日志管理功能的,这为后续的维护带来很多不便的地方. 查看是否启用了日志mysql>show variables like 'log_bin'; 怎样知道当前 ...
- 在Linux下编译带调试功能的Bochs
在Linux下使用Bochs参考: http://wangcong.org/articles/bochs.html http://kinglaw05.blog.163.com/blog/static/ ...
- Linux 下 c 语言 聊天软件
这是我学C语言写的第一个软件,是一个完整的聊天软件,里面包括客户端,和服务器端,可以互现聊天,共享文件,有聊天室等,是一个有TCP和UDP协议的聊天软件,测试过很多次在CENTOS和UBUNTU下都通 ...
- Linux下Apache配置HTTPS功能
Apache配置HTTPS功能 转 https://www.cnblogs.com/liaojiafa/p/6028816.html 一.yum 安装openssl和openssl-devel,ht ...
- Linux下gsoap实现webservice功能
蓝字为关键字,等号=后面为关键字值. 一.介绍 我们用的webservice是根据gsoap编译工具来实现,gSOAP的编译器能够自动的将用户定义的本地化的C或C++数据类型转变为符合XML语法的数据 ...
随机推荐
- PetaPoco 增删改查
1 查询单行 DBInstance.DB.SingleOrDefault<CompanyInfo11>(id); /// <summary> /// 根据id获取公司信息 // ...
- SQL SERVER 查看死锁的存储过程
end
- 灵活运用绑定变量---declare匿名块使用绑定变量
declare type cur01 is ref cursor; v_cur cur01; v_match123 varchar2(2000); v ...
- Objective-C中的分类与协议
分类 在谈分类之前,我们可以先探究下,OC中为什么出现分类这种机制,有什么好处? 假设你接到一个大项目:计算两个整数的和,差.接到任务的你马上动手.编写代码如下: #import <Founda ...
- Chrome调试(转)
原文地址:http://blog.csdn.net/chenmoquan/article/details/44943245#comments 觉得写的很适合web开发的新手 Chrome 的开发者工具 ...
- spring + maven +testng 测试常见依赖包问题
java.lang.ClassNotFoundException: org.apache.commons.dbcp.BasicDataSource解决方法:添加缺少的jar包:commons-coll ...
- php数组(array)输出三种形式
$bbbb=array("11"=>"aaa","22"=>"bbb"); //只能输出值value不能输出 ...
- prototype/constructor/__proto__之prototype
1任何对象都有__proto__属性 属性值Object2并不是所有对象都有prototype属性.只有方法对象(构造函数)以及基本数据类型还有Array,有prototype属性;并且所有方法(对象 ...
- PHP随机生成指定时间段的指定个数时间
/** * 生成某个范围内的随机时间 * @param <type> $begintime 起始时间 格式为 Y-m-d H:i:s * @param <type> $endt ...
- Git error- fatal- Needed a single revision
最近在开发中由于项目结构的重构,有一部分代码被抽出来作为了公共库(git submodule),这样公共库可以独立维护,同时其他库调用它也是非常方便的,避免了到处复制代码的痛苦. 但我在项目重构后第一 ...