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的聊天功能实现的更多相关文章

  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. oracle中drop、delete和truncate的区别

    oracle中drop.delete和truncate的区别 oracle中可以使用drop.delete和truncate三个命令来删除数据库中的表,网上有许多文章和教程专门讲解了它们之间的异同,我 ...

  2. 开源的Android开发框架-------PowerFramework使用心得(四)数据库管理DBFarmer

    DBFarmer是PowerFramework数据库管理工具的集合. 可以进行对象的存储,添加了setter和getter的参数会被收录到数据库中,每个参数作为一个项,int类型的id或_id会被作为 ...

  3. eclipse下:selenium+python自动化之Chrome driver

    1.下载chromedriver.exe文件: 2.下载的chromedriver.exe文件放置在chrome的安装目录下XXX\Chrome\Application\ ; 3.设置path环境变量 ...

  4. oracle服务器端-登陆

    由于的的操作系统是windows server版本,所以想装服务器端的server版本,一般的oracle都有'scott'用户,但是貌似服务器端的没有该用户,我用以下方式登陆: sqlplus / ...

  5. 类和ID选择器的区别

    学习了类选择器和ID选择器,我们会发现他们之间有很多的相似处,是不是两者可以通用呢?我们不要着急先来总结一下他们的相同点和不同点: 相同点:可以应用于任何元素不同点: 1.ID选择器只能在文档中使用一 ...

  6. 安卓开发之viewpager学习(头条显示)

    activity_luancher.xml代码如下: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res ...

  7. 关键字throw(something)限制

    C++函数后加关键字throw(something)限制,是对这个函数的异常安全性作出限制.void f() throw() 表示f不允许抛出任何异常,即f是异常安全的.void f() throw( ...

  8. 【BZOJ2741】【块状链表+可持久化trie】FOTILE模拟赛L

    Description FOTILE得到了一个长为N的序列A,为了拯救地球,他希望知道某些区间内的最大的连续XOR和. 即对于一个询问,你需要求出max(Ai xor Ai+1 xor Ai+2 .. ...

  9. ExtJs中动态加载机制研究(转)

    觉得写的太好了,怕弄丢了,转一下:http://extjs.org.cn/node/659 昨天我们team对于extjs的动态加载机制做了些深入研究,这里先share下controller加载的结果 ...

  10. Oracle数据库之动态SQL

    Oracle数据库之动态SQL 1. 静态SQLSQL与动态SQL Oracle编译PL/SQL程序块分为两个种:一种为前期联编(early binding),即SQL语句在程序编译期间就已经确定,大 ...