socketpair理解

原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 、作者信息和本声明。否则将追究法律责任。http://liulixiaoyao.blog.51cto.com/1361095/533469

 今天跟人谈到socketpair的问题,晚上回来写了个程序验证下自己的猜测!

     先说说我的理解:socketpair创建了一对无名的套接字描述符(只能在AF_UNIX域中使用),描述符存储于一个二元数组,eg. s[2] .这对套接字可以进行双工通信,每一个描述符既可以读也可以写。这个在同一个进程中也可以进行通信,向s[0]中写入,就可以从s[1]中读取(只能从s[1]中读取),也可以在s[1]中写入,然后从s[0]中读取;但是,若没有在0端写入,而从1端读取,则1端的读取操作会阻塞,即使在1端写入,也不能从1读取,仍然阻塞;反之亦然......

      验证所用代码:

  1. #include <stdio.h> 
  2. #include <string.h> 
  3. #include <unistd.h> 
  4. #include <sys/types.h> 
  5. #include <error.h> 
  6. #include <errno.h> 
  7. #include <sys/socket.h> 
  8. #include <stdlib.h> 
  9.  
  10. #define BUF_SIZE 30 
  11.  
  12. int main(){ 
  13.         int s[2]; 
  14.         int w,r; 
  15.         char * string = "This is a test string"
  16.         char * buf = (char*)calloc(1 , BUF_SIZE); 
  17.  
  18.         if( socketpair(AF_UNIX,SOCK_STREAM,0,s) == -1 ){ 
  19.                 printf("create unnamed socket pair failed:%s\n",strerror(errno) ); 
  20.                 exit(-1); 
  21.         } 
  22.  
  23.         /*******test in a single process ********/ 
  24.         if( ( w = write(s[0] , string , strlen(string) ) ) == -1 ){ 
  25.                 printf("Write socket error:%s\n",strerror(errno)); 
  26.                 exit(-1); 
  27.         } 
  28.         /*****read*******/ 
  29.         if( (r = read(s[1], buf , BUF_SIZE )) == -1){ 
  30.                 printf("Read from socket error:%s\n",strerror(errno) ); 
  31.                 exit(-1); 
  32.         } 
  33.         printf("Read string in same process : %s \n",buf); 
  34.           if( (r = read(s[0], buf , BUF_SIZE )) == -1){ 
  35.                           printf("Read from socket s0 error:%s\n",strerror(errno) ); 
  36.                                           exit(-1); 
  37.                                                   } 
  38.                                                   printf("Read from s0 :%s\n",buf); 
  39.  
  40.         printf("Test successed\n"); 
  41.         exit(0); 

 

     若fork子进程,然后在服进程关闭一个描述符eg. s[1] ,在子进程中再关闭另一个 eg. s[0]    ,则可以实现父子进程之间的双工通信,两端都可读可写;当然,仍然遵守和在同一个进程之间工作的原则,一端写,在另一端读取;

     这和pipe有一定的区别,pipe是单工通信,一端要么是读端要么是写端,而socketpair实现了双工套接字,也就没有所谓的读端和写端的区分

验证代码:

  1. #include <stdio.h> 
  2. #include <string.h> 
  3. #include <unistd.h> 
  4. #include <sys/types.h> 
  5. #include <error.h> 
  6. #include <errno.h> 
  7. #include <sys/socket.h> 
  8. #include <stdlib.h> 
  9.  
  10. #define BUF_SIZE 30 
  11.  
  12. int main(){ 
  13.         int s[2]; 
  14.         int w,r; 
  15.         char * string = "This is a test string"
  16.         char * buf = (char*)calloc(1 , BUF_SIZE); 
  17.         pid_t pid; 
  18.  
  19.         if( socketpair(AF_UNIX,SOCK_STREAM,0,s) == -1 ){ 
  20.                 printf("create unnamed socket pair failed:%s\n",strerror(errno) ); 
  21.                 exit(-1); 
  22.         } 
  23.  
  24.         /***********Test : fork but don't close any fd in neither parent nor child process***********/ 
  25.         if( ( pid = fork() ) > 0 ){ 
  26.                 printf("Parent process's pid is %d\n",getpid()); 
  27.              close(s[1]); 
  28.                 if( ( w = write(s[0] , string , strlen(string) ) ) == -1 ){ 
  29.                         printf("Write socket error:%s\n",strerror(errno)); 
  30.                         exit(-1); 
  31.                 } 
  32.         }else if(pid == 0){ 
  33.                 printf("Fork child process successed\n"); 
  34.                 printf("Child process's pid is :%d\n",getpid()); 
  35.                 close(s[0]); 
  36.         }else
  37.                 printf("Fork failed:%s\n",strerror(errno)); 
  38.                 exit(-1); 
  39.         } 
  40.  
  41.         /*****read***In parent and child****/ 
  42.         if( (r = read(s[1], buf , BUF_SIZE )) == -1){ 
  43.                 printf("Pid %d read from socket error:%s\n",getpid() , strerror(errno) ); 
  44.                 exit(-1); 
  45.         } 
  46.         printf("Pid %d read string in same process : %s \n",getpid(),buf); 
  47.         printf("Test successed , %d\n",getpid()); 
  48.         exit(0); 

以上代码中在父子进程之间各关闭了一个描述符,则在父进程写可从子进程读取,反之若子进程写,父进程同样可以读取;大家可以验证下

另外,我也测试了在父子进程中都不close(s[1]),也就是保持两个读端,则父进程能够读到string串,但子进程读取空串,或者子进程先读了数据,父进程阻塞于read操作!

 

之所以子进程能读取父进程的string,是因为fork时,子进程继承了父进程的文件描述符的,同时也就得到了一个和父进程指向相同文件表项的指针;若父子进程均不关闭读端,因为指向相同的文件表项,这两个进程就有了竞争关系,争相读取这个字符串.父进程read后将数据转到其应用缓冲区,而子进程就得不到了,只有一份数据拷贝(若将父进程阻塞一段时间,则收到数据的就是子进程了,已经得到验证,让父进程sleep(3),子进程获得string,而父进程获取不到而是阻塞)

有网友"笨笨"回复:

“若将父进程阻塞一段时间,则收到数据的就是子进程了,已经得到验证,让父进程sleep(3),子进程获得string,而父进程获取不到”

我验证的情况是,父进程一直阻塞在read上。我想不明白,为什么这时候父进程不能读取数据呢。
而上一种情况,父进程先读取数据,子进程仍然可以读取数据(数据为空),但子进程不会阻塞在read上。

关于这个问题,解释如下:

1.该网友说的情况的确存在,如果先让子进程sleep,此时父进程获得数据,子进程被唤醒之后读到EOF返回;若是让父进程sleep先,子进程先获取数据,之后父进程被唤醒却是一直阻塞不能返回.按理来说这两种情况应该没差别,这个区别下文描述.

2.对于网友提到问题的这个测试,我最初的目的是想说明如果通过产生子进程的方式,对一个写端同时有多个读端,这这些读端之间相互竞争.我们可以用个更有说服力的测试方法来看出这个问题.原来的测试是让一个进程sleep然后另一个进程读完所有字符,可以看到之后醒来的进程就读不到任何字符了.更好的方法是先有一个进程读取一部分的字符,然后第二个进程被唤醒,会发现这第二个进程还能读到一些字符,而这些字符是第一个进程读完剩下的.

3.第一条中的遗留问题,为什么这两种情况有不同的表现.

  原因是:如果子进程先sleep,父进程读取完数据之后,父进程退出,此时写端s[0]的引用计数变为0(之前子进程已主动close了一次),被系统释放,根据read的语义,当子进程被唤醒后会读取到EOF;但是当我们先让父进程sleep的时候,子进程读取完后退出,由于写端在父进程,没有被释放,所以父进程此时阻塞在读操作上.

  用另外一个测试来证明,我们在子进程中不主动执行close[0],也就是有两个写端,然后其他不变,子进程先sleep,父进程先读取到数据然后退出,但此时更刚刚有个区别,父进程退出的时候s[0]这个写端的描述符并不会减到0,因为子进程中还持有一个引用,所以写端健在,子进程被唤醒之后不会读到EOF返回,而是阻塞在读操作上

 

最后,有关socketpair在内核中实现的一点点描述:

socketpair会创建两个描述符,但改描述符不属于任何的实际文件系统,而是网络文件系统,虚拟的.同时内核会将这两个描述符彼此设为自己的peer即对端(这里即解决了如何标识读写端,可以想象,两个描述符互为读写缓冲区,即解决了这个问题).然后应用相应socket家族里的read/write函数执行读写操作.

有了这个基础,即可明白为什么试用fork产生的两个子进程都不关闭读端的时候会竞争,如上所述,他们共享相同的文件表项,有相同的inode和偏移量,两个进程的操作当然是相互影响的.

本文出自 “流离and逍遥” 博客,请务必保留此出处http://liulixiaoyao.blog.51cto.com/1361095/533469

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/434673.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

小数点化分数的过程_分数和小数的互化

《分数和小数的互化》由会员分享&#xff0c;可在线阅读&#xff0c;更多相关《分数和小数的互化(3页珍藏版)》请在人人文库网上搜索。1、_五__年级__数学____备课组教案教师备课时间课时课题分数和小数的互化课型新授课前准备&#xff1a;教材剖析(考点、易错点、关联考点)。1…

pycharm不同py文件共享参数_PyCharm安装笔记

1. 介绍1.1 介绍今天福哥带着大家学习如何安装非常好用的Python编辑器&#xff0c;也就是jetbrains全家桶的PyCharm编辑器。PyCharm是jetbrans开发的一款专门用来编写Python程序的编辑器&#xff0c;它的自动补全、代码联想、框架支持、插件支持以及高效的反应速度成为了编写Py…

WPF - 自定义标记扩展

在使用WPF进行编程的过程中&#xff0c;我们常常需要使用XAML的标记扩展&#xff1a;{Binding}&#xff0c;{x:Null}等等。那么为什么WPF提供了XAML标记扩展这一功能&#xff0c;我们又如何创建自定义的标记扩展呢。这就是本文将要讨论的内容。 一&#xff0e;从标记扩展的分析…

mysql c api 函数 linux下 mysql_query_Linux C 调用MYSQL API 函数mysql_escape_string()转义插入数据...

标签&#xff1a;Title:Linux C 调用MYSQL API 函数mysql_escape_string()转义插入数据 --2013-10-11 11:57#include #include #include #include "mysql.h"int main(int argc, char *argv[]){MYSQL my_connection;int res;mysql_init(&my_connection);char UNAM…

使用LVM

一些比较常用的命令&#xff1a; vgdisplay 展示创建的volume group lvdisplay 展示创建的logic volume 在我们的项目中&#xff0c;vg把磁盘阵列的所有空间分给它&#xff0c;pv为每个盘&#xff0c;lv分配为每个虚拟机。 这样需要修改的地方&#xff1a;转载于:https://www.c…

mysql客户端保存数据乱码_mysql客户端数据乱码问题

往mysql插入中文数据时出现乱码&#xff0c;如下&#xff1a;mysql> select * from test;-----------------| id | name |-----------------| 1 | 鐜嬪煿鍧?|| 2 | 鐜嬫檽宄?|| 3 | 寮犲皬闆?|| 4 | 鐜嬮洩宄?|| 5 | 闄嗕簯 || 6 | 妗冭姳濂…

基于半同步/半反应堆线程池实现的HTTP解析服务端程序

简介&#xff1a; 半同步/半反应堆线程池是通过一个线程往工作队列添加任务T&#xff0c;然后工作线程竞争工作队列获得任务T。HTTP请求解析服务端程序&#xff1a;逐行解析客户端发送来的HTTP请求然后作出HTTP回答。采用线程池就是&#xff1a;服务端创建一个线程池&#xff0…

GridView中实现CheckBox的全选

用服务器端的方法&#xff1a; 在页面上放一个gridview控件&#xff0c;配置好数据源&#xff0c;编辑列&#xff0c;添加一个模版列&#xff0c;再编辑模版&#xff0c;放入一个checkbox控件。代码如下&#xff1a; <asp:GridView ID"GridView1" runat"serv…

mysql replace first_Java replaceFirst()方法

Java replaceFirst()方法replaceFirst() 方法使用给定的参数 replacement 替换字符串第一个匹配给定的正则表达式的子字符串。语法publicStringreplaceFirst(Stringregex,Stringreplacement)参数 regex -- 匹配此字符串的正则表达式。replacement -- 用来替换第一个匹配项的字符…

Windows与linux双系统安装

[源] [http://xiaomaimai.blog.51cto.com/1182965/294256] Windows与linux双系统 Linux的安装方式有硬盘安装、网络安装、光驱安装。 双系统的安装最后先安装windows&#xff0c;再安装linux&#xff0c;因为windows每次安装时都会重新修改系统引导文件&#xff0c;如果安装…

第一步:Axure 使用svn多人协作产品开发(提交文件)

注册登陆http://www.svnxiezuo.com站点 注册登陆http://www.svnxiezuo.com站点 从注册http://www.svnxiezuo.com站点获取svn项目地址 编辑axure文件 开始创建axure共享文件 创建axure共享文件 axure提交文件到svn版本库当中 登陆http://www.svnxiezuo.com站点svn版本库 axure创…

python文件函数_python文件和文件夹复制函数

本文实例为大家分享了python文件和文件夹复制函数&#xff0c;供大家参考&#xff0c;具体内容如下文件复制函数代码和注释如下&#xff1a;# 文件复制函数def copy_file(file1,file2):将文件一复制到文件二:param file1::param file2::return:# 首先打开源文件file1和目标文件…

\r,\n,\r\n

\r 是回车&#xff0c;return(光标到达最左侧) \n 是换行&#xff0c;newline(移到下一行) 如果只换行&#xff0c;打印的东西会在下一行的先一个位置继续&#xff0c;比如 #### 如果只回车&#xff0c;打印的东西会覆盖原有的东西 test\r123-->123t atest\r…

sql server2005索引

create index myindex on Student(StudentID) ---创建一个索引&#xff0c;括号内为要为那一列添加索引drop index Student.myindex ---删除索引&#xff0c;需指定“表名.索引列” 详细参考于&#xff1a;http://www.2cto.com/database/201201/117131.html转载于:https://www.…

python socketio async client_electron/socket.io client-python socketio/aiohttp server//连接失败

此websocket连接失败。有趣的是这几天前还在起作用。我把电子从6降到了5.0.6,但这没用。服务器from aiohttp import webimport socketioapp web.Application()sio socketio.AsyncServer()# or sio socketio.AsyncServer(cors_allowed_origins*)sio.attach(app)sio.on(connec…

2012.2.18-silverlight设计器崩溃

先把所有与siverlight有关的silverlight,silverlightsdk,silverlight_too ,silverlight toolkit,全部卸掉&#xff0c;然后重装&#xff0c;然后启动2010&#xff0c;打开原来的silverlight的项目OK&#xff0c;新疆项目也OK 总结&#xff1a;通过以上解决方法&#xff0c;可以…

领导者/追随者(Leader/Followers)模型和半同步/半异步(half-sync/half-async)模型

领导者/追随者&#xff08;Leader/Followers&#xff09;模型和半同步/半异步&#xff08;half-sync/half-async&#xff09;模型都是常用的客户-服务器编程模型。这几天翻了些文章&#xff0c;发现对领导者/追随者模型说的比较少&#xff0c;下面就这个模型打个比方&#xff1…

php如何操作mysql数据库代码_php如何操作mysql数据库的类(附代码)

这篇文章主要介绍了php简单操作mysql数据库的类,涉及php操作mysql的连接、查询、插入、删除等基本操作方法,非常具有实用价值,需要的朋友可以参考下。<?php /*** Database class** version: 2.2* revised: 27 may 2007***/class Database {var $host;var $name;var $user;v…

插件之基类窗体

基类窗体要实现的需求包括&#xff1a;控件窗体的字体&#xff0c;权限控制&#xff0c;控制子窗体的显示范围&#xff0c;以及其它一些通用方法的声明。。。。。。 通用方法可声明为VIRTUAL方法&#xff0c;以允许从它继承的子窗体 可以OVERRIDE这些方法&#xff0c;达到多态的…

网络编程模型综述 之 成熟的IO框架介绍

ACE “重量级的C I/O框架&#xff0c;用面向对象实现了一些I/O策略和其它有用的东西&#xff0c;特别是它的Reactor是用OO方式处理非阻塞I/O&#xff0c;而Proactor是用OO方式处理异步I/O的( In particular, his Reactor is an OO way of doing nonblocking I/O, and Proactor …