【版权声明:尊重原创。转载请保留出处:blog.csdn.net/shallnet 或 .../gentleliu,文章仅供学习交流,请勿用于商业用途】
    在一个较大的project其中。一般都会有多个进程构成,各个功能是一个独立的进程在执行。

既然多个进程构成一个project,那么多个进程之间肯定会存在一些信息交换或共享数据,这就涉及到进程间通信。进程间通道有非常多种。比方有最熟悉网络编程中的socket、还有共享内存、消息队列、信号、管道等非常多方式。每一种方式都有自己的适用情况,在本系列文章中笔者将会对多种进程间通信方式进行具体解释,一是对自己工作多年在这方面的经验做一个积累,二是将其分享给各位民工。也许还能从大家的拍砖其中得到意外收获。
 
socket网络编程可能使用得最多,经经常使用在网络上不同主机之间的通信。

事实上在同一主机内通信也能够使用socket来完毕。socket进程通信与网络通信使用的是统一套接口,不过地址结构与某些參数不同。在使用socket创建套接字时通过指定參数domain是af_inet(ipv4因特网域)或af_inet6(ipv6因特网域)或af_unix(unix域)来实现。
在笔者这一篇文章中曾有具体介绍创建socket通信流程及基础知识。

http://blog.csdn.net/shallnet/article/details/17734919
 
首先来看一下使用af_inet域以及本地环回地址来实现本地主机进程间通信。
服务进程创建监听套接字:

int ser_afinet_listen(int port)
{
int listenfd, on;
struct sockaddr_in seraddr; listenfd = socket(af_inet, sock_stream, 0);
if (listenfd < 0) {
fprintf(stderr, "socket: %s\n", strerror(errno));
return -1;
} on = 1;
setsockopt(listenfd, sol_socket, so_reuseaddr, &on, sizeof(on)); seraddr.sin_family = af_inet;
seraddr.sin_port = port;
seraddr.sin_addr.s_addr = htonl(inaddr_any); if (bind(listenfd, (struct sockaddr *)&seraddr, sizeof(struct sockaddr_in)) < 0) {
fprintf(stderr, "bind: %s\n", strerror(errno));
return -1;
} if (listen(listenfd, sock_ipc_max_conn) < 0) {
fprintf(stderr, "listen: %s\n", strerror(errno));
return -1;
} return listenfd;
}

服务进程处理连接请求例如以下:

int ser_accept(int listenfd)
{
int connfd;
struct sockaddr_un cltaddr;
ssize_t recvlen, sendlen;
char buf[sock_ipc_max_buf];
socklen_t addrlen; addrlen = sizeof(cltaddr);
for (;;) {
connfd = accept(listenfd, (struct sockaddr *)&cltaddr, &addrlen);
if (connfd < 0) {
fprintf(stderr, "accept: %s\n", strerror(errno));
return -1;
} if (recvlen = ipc_recv(connfd, buf, sizeof(buf)) < 0) {
continue;
} printf("recv: %s\n", buf);
snprintf(buf, sizeof(buf), "hello, ipc client!");
if (ipc_send(connfd, buf, strlen(buf)) < 0) {
continue;
} close(connfd);
}
}

客户进程初始化例如以下:
指定要连接的服务器地址为本地换回地址,这样发送的连接就会回到本地服务进程。

int clt_afinet_conn_init(int port)
{
int fd; fd = socket(af_inet, sock_stream, 0);
if (fd < 0) {
fprintf(stderr, "socket: %s\n", strerror(errno));
return -1;
} seraddr.sin_family = af_inet;
seraddr.sin_port = port;
if (inet_pton(af_inet, "127.0.0.1", &seraddr.sin_addr) < 0) {//环回地址
fprintf(stderr, "inet_pton: %s\n", strerror(errno));
return -1;
} return fd;
}

    客户进程向服务进程发送接收请求例如以下:

if (connect(fd, (struct sockaddr *)&seraddr, sizeof(struct sockaddr_in)) < 0) {
fprintf(stderr, "connect: %s\n", strerror(errno));
return -1;
} if ((sendlen = ipc_send(fd, buf, strlen(buf))) < 0) {
return -1;
} if ((recvlen = ipc_recv(fd, buf, sizeof(buf))) < 0) {
return -1;
}

服务进程先执行,客户进程执行:
# ./client
recv: hello, ipc client!
# ./server
recv: hello ipc server!
通信过程完毕。

创建类型为af_unix(或af_local)的socket,表示用于进程通信。

socket进程通信与网络通信使用的是统一套接口:

#include<sys/socket.h>
int socket(int domain, int type, int protocol);

其中,domain 參数指定协议族,对于本地套接字来说,其值被置为 af_unix 枚举值,随便说一下,af_unix和af_local是同一个值,看以下linux/socket.h头文件部分例如以下,两个宏的值都一样为1。

         ……
/* supported address families. */
#define af_unspec 0
#define af_unix 1 /* unix domain sockets */
#define af_local 1 /* posix name for af_unix */
#define af_inet 2 /* internet ip protocol */
#define af_ax25 3 /* amateur radio ax.25 */
……

以af_xx开头和pf_xx开头的域都是一样的。继续看头文件部分内容就一切都明确了:

……
#define pf_unspec af_unspec
#define pf_unix af_unix
#define pf_local af_local
#define pf_inet af_inet
#define pf_ax25 af_ax25
……

所以我们在指定socket的类型时这四个域能够随便用啦,笔者这里统一使用af_unix了。

type 參数可被设置为 sock_stream(流式套接字)或 sock_dgram(数据报式套接字),对于本地套接字来说,流式套接字(sock_stream)是一个有顺序的、可靠的双向字节流,相当于在本地进程之间建立起一条数据通道。数据报式套接字(sock_dgram)相当于单纯的发送消息。在进程通信过程中。理论上可能会有信息丢失、复制或者不按先后次序到达的情况,但因为其在本地通信,不通过外界网络,这些情况出现的概率非常小。
本地套接字的通信两方均须要具有本地地址,地址类型为 struct sockaddr_un结构体(位于linux/un.h):

#ifndef _linux_un_h
#define _linux_un_h #define unix_path_max 108 struct sockaddr_un {
sa_family_t sun_family; /* af_unix */
char sun_path[unix_path_max]; /* pathname */
};
#endif /* _linux_un_h */

创建监听套接字:

int ser_afunix_listen(const char *pathname)
{
int listenfd, on;
struct sockaddr_un seraddr; listenfd = socket(af_unix, sock_stream, 0);
if (listenfd < 0) {
fprintf(stderr, "socket: %s\n", strerror(errno));
return -1;
} unlink(pathname);
seraddr.sun_family = af_unix;
snprintf(seraddr.sun_path, sizeof(seraddr.sun_path), "%s", pathname); if (bind(listenfd, (struct sockaddr *)&seraddr, sizeof(struct sockaddr_un)) < 0) {
fprintf(stderr, "bind: %s\n", strerror(errno));
return -1;
} if (listen(listenfd, sock_ipc_max_conn) < 0) {
fprintf(stderr, "listen: %s\n", strerror(errno));
return -1;
} return listenfd;
}

服务进程处理连接请求相似上面网络通信。
 
客户进程初始化套接字过程为:

int clt_afunix_conn_init(const char *pathname)
{
int fd;
struct sockaddr_un localaddr; fd = socket(af_unix, sock_stream, 0);
if (fd < 0) {
fprintf(stderr, "socket: %s\n", strerror(errno));
return -1;
} localaddr.sun_family = af_unix;
snprintf(localaddr.sun_path, sizeof(localaddr.sun_path), "%s-cltid%d", pathname, getpid()); if (bind(fd, (struct sockaddr *)&localaddr, sizeof(struct sockaddr_un)) < 0) {
fprintf(stderr, "bind: %s\n", strerror(errno));
return -1;
} seraddr.sun_family = af_unix;
snprintf(seraddr.sun_path, sizeof(seraddr.sun_path), "%s", pathname); return fd;
}

客户进程向服务进程发送接收请求相似上面网络通信。
执行结果同网络通信部分。
本节只给出了一个非常easy的通信程序,只简单实现了客户进程和服务进程之间的通信。在下一节笔者将会在本节演示样例的基础上做改动,写一个在实际应用中能够使用的代码。
本节演示样例代码下载链接:
 http://download.csdn.net/detail/gentleliu/8140459

细说linux IPC(一):基于socket的进程间通信(上)的更多相关文章

  1. 细说linux IPC(三):mmap系统调用共享内存

    [版权声明:尊重原创,转载请保留出处:blog.csdn.net/shallnet 或 .../gentleliu,文章仅供学习交流,请勿用于商业用途]         前面讲到socket的进程间通 ...

  2. Linux IPC udp/ip socket 编程

    模型 #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include < ...

  3. Linux IPC tcp/ip socket 编程

    模型 #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include < ...

  4. 细说linux IPC(四):posix 共享内存

    [版权声明:尊重原创.转载请保留出处:blog.csdn.net/shallnet 或 .../gentleliu,文章仅供学习交流,请勿用于商业用途]         上一节讲了由open函数打开一 ...

  5. C语言 linux环境基于socket的简易即时通信程序

    转载请注明出处:http://www.cnblogs.com/kevince/p/3891033.html      ——By Kevince 最近在看linux网络编程相关,现学现卖,就写了一个简易 ...

  6. 基于socket实现上传文件

    基于socket实现文件上传 客户端代码: #!/usr/bin/env python # -*- coding:utf-8 -*- """ 这个是实现上传文件 首先让客 ...

  7. 【Linux 应用编程】进程管理 - 进程间通信IPC之管道 pipe 和 FIFO

    IPC(InterProcess Communication,进程间通信)是进程中的重要概念.Linux 进程之间常用的通信方式有: 文件:简单,低效,需要代码控制同步 管道:使用简单,默认阻塞 匿名 ...

  8. 【Linux 应用编程】进程管理 - 进程间通信IPC之共享内存 mmap

    IPC(InterProcess Communication,进程间通信)是进程中的重要概念.Linux 进程之间常用的通信方式有: 文件:简单,低效,需要代码控制同步 管道:使用简单,默认阻塞 匿名 ...

  9. linux下socket编程-进程间通信

    一.什么是Socket Socket接口是TCP/IP网络通信的API,Socket接口定义了许多函数或例程,可以用它们来开发TCP/IP网络上的应用程序. Socket类型有两种:流式Socket ...

随机推荐

  1. 解决Linux用户模板文件被删除后显示不正常问题

    缺失用户模板文件(用户骨架文件)会导致shell提示符不完整,可以到/etc/skel/目录下复制相关文件来恢复 (1).创建测试环境,删除模板文件 [root@xuexi ~]# useradd t ...

  2. 写的模块和方法 wap 和 pc

    createjs 画了一个曲线功能 rem 的适配方式 $.fn.stop 方法, zepto 没有的, 对于 2d的旋转 变形 还有 移动都可以停下来, 做动画的属性存储, getComputedS ...

  3. Socket读取页面

    http://www.knowsky.com/363189.html http://hi.baidu.com/myyers/item/f90fa3f57d89e1d243c36a34 http://h ...

  4. CXF浅析

    CXF 框架支撑环境    CXF 框架是一种基于 Servlet 技术的 SOA 应用开发框架,要正常运行基于 CXF 应用框架开发的企业应用,除了 CXF 框架本身之外,还需要 JDK 和 Ser ...

  5. 背包系列练习及总结(hud 2602 && hdu 2844 Coins && hdu 2159 && poj 1170 Shopping Offers && hdu 3092 Least common multiple && poj 1015 Jury Compromise)

    作为一个oier,以及大学acm党背包是必不可少的一部分.好久没做背包类动规了.久违地练习下-.- dd__engi的背包九讲:http://love-oriented.com/pack/ 鸣谢htt ...

  6. 【set】【链表】hdu6058 Kanade's sum

    f(l,r,K)表示区间l,r里面的K大值,问你所有连续子区间的f之和. l(i)表示i左侧第一个比它大的数的位置,r(i)表示i右侧第一个比它大的数的位置.可以用set处理出来. 把数从大到小排序, ...

  7. 【判断解是否可行-二分】POJ1064-Cable master

    http://poj.org/problem?id=1064 [题目大意] 给出几条绳子的长度,问如果要切出k条长度相等的绳子,这k条绳子最常多长? [思路] 二分.把下界设为0,上界设为所有绳子长度 ...

  8. 【DFS】POJ3009-Curling 2.0

    [题目大意] 给出一张地图,一旦往一个方向前进就必须一直向前,直到一下情况发生:(1)碰到了block,则停在block前,该block消失:(2)冲出了场地外:(3)到达了终点.改变方向十次以上或者 ...

  9. Springcloud中的region和zone的使用

    一.背景 用户量比较大或者用户地理位置分布范围很广的项目,一般都会有多个机房.这个时候如果上线springCloud服务的话,我们希望一个机房内的服务优先调用同一个机房内的服务 ,当同一个机房的服务不 ...

  10. Problem J: 零起点学算法89——程序设计竞赛

    #include<stdio.h> //选择排序法 int main(){ ]; while(scanf("%d",&n)!=EOF){ ;i<n;i++ ...