UNIX 域协议(本地通信协议)

概述

        Unix 域协议并不是一个实际的协议族,而是在单个主机上执行客户/服务通信的一种方式。是进程间通信(IPC)的一种方式。
        它提供了两类套接字:字节流套接字 SOCK_STREAM(有点像 TCP)和数据报套接字 SOCK_DGRAM(有点像 UDP)
        UNIX 域数据报服务是可靠的,不会丢失消息,也不会传递出错。
        IP 协议标识客户服务器是通过 IP 地址和端口号实现的,UNIX 域协议中用于标识客户机和服务器的协议地址的是普通文件系统中的路径名。
UNIX 域协议的特点:
        1.UNIX 域套接字域 TCP 套接字相比,在同一台主机的传输速度前者是后者的两倍 。UNIX 域套接字仅仅复制数据,并不执行协议处理,不需要添加或删除网络报头,无需计算校验和,不产
生顺序号,也不需要发送确认报文
        2.UNIX 域套接字可以在同一台主机上各进程之间传递文件描述符
        3.UNIX 域套接字域传统套接字的区别是用路径名表示协议族的描述
UNIX 域协议地址
#define UNIX_PATH_MAX 108
struct sockaddr_un
{__kernel_sa_family_t sun_family; /* 通信协议 AF_UNIX / AF_LOCAL */char sun_path[UNIX_PATH_MAX]; /* pathname *///UNIX 域协议的地址,是以'\0'结束的本地文件系统中的绝对路径名//会自动的创建不需要提前创建
};
UNIX 域协议用于本地进程间通信的编程流程和 TCP/UDP 一样,只不过具体的协议地址是使
用文件路径名
        Server:先创建套接字 -> 绑定地址 -> 监听 -> accept 客户端连接 -> 连接成功开始通信 -> 关闭套接字
        Client:先创建套接字 ->绑定地址-> 连接 server -> 开始通信 -> 关闭套接字。

UNIX 域协议之进程间通信方式一:socketpair 函数

【头文件】
#include <sys/types.h>
#include <sys/socket.h>
【函数原型】
int socketpair(int domain, int type, int protocol, int sv[2]);
【函数功能】
        此函数只用于 Unix 域套接口,它的 domain 必须是 AF_LOCAL,protocol 参数必须为 0,type 可以是 SOCK_STREAM 也可以是 SOCK_DGRAM,它与调用 pipe 创建的普通管道类似,差别在 于普通管道只能进行单向读或单向写,而该管道是全双工的,可以单向读写。
【参数含义】
        [domain]: 表示协议族,在 Linux 下只能为 AF_LOCAL 或者 AF_UNIX 。(自从 Linux 2.6.27 后也支持 SOCK_NONBLOCK 和 SOCK_CLOEXEC)
        [type]: 表示协议,可以是 SOCK_STREAM 或者 SOCK_DGRAM。SOCK_STREAM 是基于TCP 的,而 SOCK_DGRAM 是基于 UDP 的
        [protocol]: 表示类型,只能为 0
        [sv]: 套节字柄对,该两个句柄作用相同,均能进行读写双向操作
【返回值】
        0 为创建成功,-1 为创建失败,并且 errno 来表明特定的错误号,具体错误号如下所述:
        EAFNOSUPPORT:本机上不支持指定的 address。 EFAULT: 地址 sv 无法指向有效的进程地址空间内。
        EMFILE: 已经达到了系统限制文件描述符,或者该进程使用过量的描述符。
        EOPNOTSUPP:指定的协议不支持创建套接字对。
        EPROTONOSUPPORT:本机不支持指定的协议。
【备注】
        1、该函数只能用于 UNIX 域( LINUX )下。
        2、只能用于有亲缘关系的进程(或线程)间通信。
        3、所创建的套节字对作用是一样的,均能够可读可写(而管道 PIPE 只能进行单向读写)。
        4、在读的时候,管道内必须有内容,否则将会阻塞;简而言之,该函数是阻塞的。
【示例代码】
/*socketpair1.c*/
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <stdio.h>
int main ()
{int sv[2];int result = socketpair(AF_UNIX, SOCK_STREAM, 0, sv);if (result < 0){exit(1);}printf("sv[0] is : %d \n", sv[0]); //这两个套节字句柄并不相同,但作用是一样的printf("sv[1] is : %d \n", sv[1]);if (fork()){ /* 父进程 */int val = 0;pid_t pid = getpid(); close(sv[1]); //父进程关闭 sv[1]的读写权限while (1){ ++val;printf("%d send message: %d\n", pid, val); write(sv[0], &val, sizeof(val)); //父进程向管道里写数据//也可以去读// read(sv[0], &val, sizeof(val)); //如果字进程不写数据,将会导致此处堵塞//printf("%d receive message: %d\n", pid, val);sleep(1);}}else{ /*子进程*/int val = 0;close(sv[0]); //字进程关闭 sv[0]的读写权限pid_t pid = getpid(); while(1){read(sv[1], &val, sizeof(val)); //字进程从管道中取数据printf("%d receive message: %d\n", pid, val);// printf("%d receive message: %d\n", pid, val); //也可以写// write(sv[1], &val, sizeof(val));}}
}

UNIX 域协议之进程间通信方式二:套接字通信方式

        UNIX 域协议进行本地通信的方式和 TCP、UDP 很类似,下面就以 UNIX 域协议的流式套接字来写一个客户端和服务器间通信的例子,使用流式套接字,则本地间的通信方式就要类似于 TCP,这样的通信方式同时也具有 TCP 通信的特点。
客户端示例程序:
#include<stdio.h>
#include<stdlib.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <sys/un.h>
#include <arpa/inet.h>
#include<netinet/in.h> //为了使用 IPV4 地址结构体
#include <string.h>
#include <arpa/inet.h>
#include <unistd.h>#define UNIX_PATH_PS "/home/china/sockp1" //连接的套接字地址
#define UNIX_PATH_PC "/home/china/sockp2" //绑定的套接字地址int main(int argc,char *argv[])
{system("rm /home/china/sockp2"); //防止文件重名//1.创建一个套接字int sockfd = socket(AF_UNIX, SOCK_STREAM,0); //创建一个本地通信 UNIX 流式套接字if(-1 == sockfd){perror("create socket failed");exit(-1);}//2.绑定一个通信地址(让服务器能看到我),可不绑定struct sockaddr_un local; memset(&local,0,sizeof(struct sockaddr_un)); //清空结构体local.sun_family = AF_UNIX;strcpy(local.sun_path,UNIX_PATH_PC);int ret = bind(sockfd,(struct sockaddr*)&local,sizeof(local));if(ret == -1){perror("bind error");exit(-1);}printf("bind success\n");//连接目标服务器struct sockaddr_un saddr; //保存目标服务器的地址(本地的文件名)memset(&saddr,0,sizeof(struct sockaddr_un)); //清空结构体saddr.sun_family = AF_UNIX;strcpy(saddr.sun_path,UNIX_PATH_PS);ret = connect(sockfd, (struct sockaddr*)&saddr,sizeof(saddr)); //请求连接目标服务器if(ret == -1){perror("connect error");exit(-1);}printf("connect success\n");char buf[1024] = {0};//读取信息ret = recv(sockfd,buf,1024, 0);if(ret == -1){perror("recv error");}printf("recv size:%d,recv data:%s\n",ret,buf);//发送信息ret = sendto(sockfd,"byebye",6,0,NULL,0);if(ret == -1){perror("sendto error");}printf("sendto size:%d\n",ret);//关闭套接字shutdown(sockfd,SHUT_RDWR);close(sockfd);return 0;
}
服务器程序示例:
#include<stdio.h>
#include<stdlib.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <sys/un.h>
#include <arpa/inet.h>
#include<netinet/in.h> //为了使用 IPV4 地址结构体
#include <string.h>
#include <arpa/inet.h>
#include <unistd.h>
#define UNIX_PATH_PS "/home/china/sockp1"int main(int argc,char *argv[])
{system("rm /home/china/sockp1"); //保证该文件没有被使用//1.创建一个套接字int sockfd = socket(AF_UNIX, SOCK_STREAM,0); //创建一个本地通信UNIX 流式套接字if(-1 == sockfd){perror("create socket failed");exit(-1);}//2.绑定一个通信地址(作为服务器本身的地址)struct sockaddr_un local; //保存服务器的地址(文件名)memset(&local,0,sizeof(struct sockaddr_un)); //清空结构体local.sun_family = AF_UNIX;strcpy(local.sun_path,UNIX_PATH_PS);int ret = bind(sockfd,(struct sockaddr*)&local,sizeof(local)); //绑定服务器的地址if(ret == -1){perror("bind error");exit(-1);}printf("bind success\n");//3.开启对一个套接字的监听listen(sockfd,250);//4.等待客户端的连接while(1){struct sockaddr_un caddr; //保存客户端的地址(文件名)socklen_t len = sizeof(caddr);int confd = accept(sockfd,(struct sockaddr*)&caddr,&len); //阻塞等待客户端连接if(confd > 0) //客户端连接成功,返回一个连接套接字专门用来和客户端通信{//一个客户端连接成功.开一个进程/线程去处理这个连接printf("client path:%s\n",caddr.sun_path);ret = sendto(confd,"nishiliangzaima?",20,0,NULL,0); //发送消息if(ret == -1){perror("sendto error");}char buf[1024] = {0};ret = read(confd,buf,1024); //接收消息if(ret <= 0){perror("recv error");}printf("recv size:%d,recv data:%s\n",ret,buf);close(confd);//关闭连接套接字}}//关闭套接字shutdown(sockfd,SHUT_RDWR);close(sockfd);return 0;
}

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

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

相关文章

[论文阅读]Voxel R-CNN——迈向高性能基于体素的3D目标检测

Voxel R-CNN Voxel R-CNN: Towards High Performance Voxel-based 3D Object Detection 迈向高性能基于体素的3D目标检测 论文网址&#xff1a;Voxel R-CNN 论文代码&#xff1a;Voxel R-CNN 简读论文 该论文提出了 Voxel R-CNN&#xff0c;这是一种基于体素的高性能 3D 对象…

Go 语言gin框架的web

节省时间与精力&#xff0c;更高效地打造稳定可靠的Web项目&#xff1a;基于Go语言和Gin框架的完善Web项目骨架。无需从零开始&#xff0c;直接利用这个骨架&#xff0c;快速搭建一个功能齐全、性能优异的Web应用。充分发挥Go语言和Gin框架的优势&#xff0c;轻松处理高并发、大…

Qt实现卡牌对对碰游戏

效果 闲来无事&#xff0c;实现一个对对碰游戏&#xff0c;卡牌样式是火影动漫。 先上效果&#xff1a; 卡牌对对碰_火影主题 玩法 启动游戏&#xff0c;进入第一关卡&#xff0c;所有卡牌都为未翻开状态&#xff0c;即背面朝上&#xff1b;点击卡牌&#xff0c;则将卡牌翻开…

jenkins工具系列 —— 删除Jenkins JOB后清理workspace

文章目录 问题现象分析解决思路脚本实现问题现象分析 Jenkins使用过程中,占用空间最大的两个位置: 1 、workspace: 工作空间,可以随便删除,删除后再次构建时间可能会比较长,因为要重新获取一些资源。 2 、job: 存放的是项目的配置、构建结果、日志等。不建议手动删除,…

双亲委派模式

双亲委派模型 双亲委派的工作过程 一个类加载器收到类加载的请求时&#xff0c;它不会马上加载该类&#xff0c;而是把这个请求委托给父加载器去完成&#xff0c;每一个层次的类加载器都是如此&#xff0c;因此所有的类加载请求都必须先通过启动类加载器尝试加载&#xff0c;只…

【快报】正在把教学视频搬运到B站和油管

hello 大家好&#xff0c;我是老戴。 熟悉我的同学知道&#xff0c;我从14年开始录制GIS相关的教学视频&#xff0c;之前是放到优酷上给大家下载&#xff0c;后期发现很多人把视频弄下来淘宝上卖&#xff0c;然后我就把视频整体放到了我自己的网站上。 随着视频录制的数量越来…

HBuilderX实现安卓真机调试

1. 简介 HBuilderX 简称 HX&#xff0c;HBuilder&#xff0c;H 是 HTML 的缩写&#xff0c;Builder 是建设者。是为前端开发者服务的通用 IDE&#xff0c;或者称为编辑器。与 vscode、sublime、webstorm 类似。 它可以开发普通 web 项目&#xff0c;也可以开发 DCloud 出品的 u…

nodejs+vue+python+php基于微信小程序的在线学习平台设计与实现-计算机毕业设计

困扰管理层的许多问题当中,在线学习也是不敢忽视的一块。但是管理好在线学习又面临很多麻烦需要解决,例如&#xff1a;如何在工作琐碎,记录繁多的情况下将在线学习的当前情况反应给课程问题管理员决策,等等。 流,开发一个在线学习平台小程序一方面的可能会更合乎时宜,另一方面来…

略谈Kamailio调试

Kamailio调试办法有很多&#xff0c;我先来抛转&#xff1a; kamcmd cfg.seti core debug 3 一般debug&#xff08;全局参数&#xff09;设置为2&#xff0c;但通过上述命令可临时设置为3&#xff0c;然后收集日志&#xff0c;跟源码对照&#xff0c;以便找到问题。本人通过这…

Java IDEA设置环境变量 以及代码获取

IDEA 设置环境变量 1.进入如图设置&#xff0c;一般的 java 程序和 spring Boot &#xff0c;还是tomcat 都可以从这里进入 2.可以在如下地方手动添加 3. tomcat 类 4.spring boot 类 代码获取指定值 假定我设置如下 代码则如下获取&#xff08;类均为JDK 自带类&…

代理模式代理模式

目录 1、使用场景 2、静态代理 3、动态代理 JDK动态代理 CGlib 动态代理实现 1、使用场景 使用代理模式主要有两个目的&#xff1a;一是保护目标对象&#xff0c;二是增强目标对象。 2、静态代理 NO.1 抽象接口&#xff1a;定义视频播放器接口Player public interface P…

http1,https,http2,http3总结

1.HTTP 当我们浏览网页时&#xff0c;地址栏中使用最多的多是https://开头的url&#xff0c;它与我们所学的http协议有什么区别&#xff1f; http协议又叫超文本传输协议&#xff0c;它是应用层中使用最多的协议&#xff0c; http与我们常说的socket有什么区别吗&#xff1f; …

怎么在电脑桌面上添加待办事项?

在电脑桌面上选择一款待办事项工具&#xff0c;可以高效率地督促各项任务的按时完成&#xff0c;大大地提高工作的效率&#xff0c;支持在电脑上安装待办事项的工具类型是比较多的&#xff0c;为更好的辅助日常办公&#xff0c;建议大家可以选择高效率辅助办公的电脑便签工具&a…

网络协议--TCP的未来和性能

24.1 引言 TCP已经在从1200 b/s的拨号SLIP链路到以太数据链路上运行了许多年。在80年代和90年代初期&#xff0c;以太网是运行TCP/IP最主要的数据链路方式。虽然TCP在比以太网速率高的环境&#xff08;如T2电话线、FDDI及千兆比网络&#xff09;中也能够正确运行&#xff0c;但…

高并发和存储之间的关系是什么?

文章目录 &#x1f50a;博主介绍&#x1f916;博主的简介&#x1f4e5;博主的目标 &#x1f964;本文内容&#x1f34a; 一、高并发对存储的压力&#x1f34a; 二、存储的性能和可扩展性 &#x1f4e2;总结 &#x1f50a;博主介绍 &#x1f4d5;我是廖志伟&#xff0c;一名Java…

基于CentOS8创建Apache服务镜像

一、构建过程中遇到的问题 容器运行时报错 AH00558: httpd: Could not reliably determine the server’s fully qualified domain name, using 172.17.0.4. Set the ‘ServerName’ directive globally to suppress this message 解决方案&#xff1a;需要在Dockerfile文件中…

[Unity+智谱AI开放平台]调用ChatGLM Tuobo模型驱动AI小姐姐数字人

1.简述 本篇文章主要介绍一下&#xff0c;在Unity端&#xff0c;集成智谱AI开放平台提供的chatglm模型api&#xff0c;实现AI聊天互动相关的功能。从智谱AI官方站点上看到&#xff0c;提供有chatglm turbo的公共模型服务&#xff0c;能够实现32K超长上下文&#xff0c;应用到我…

Spring Security 6.1.x 系列(3)—— 基于过滤器的基础原理(二)

四、SecurityFilterChain 在Serlvet中&#xff0c;一组Security Filter组成SecurityFilterChain&#xff0c;SecurityFilterChain的概念就比较好理解&#xff0c;是Spring Security 提供的过滤器链&#xff0c;用于管理本身所有的过滤器&#xff0c;在上面的流程图中已有说明。…

【嵌入式】Linux C编程——C要注意的东西

1、语法分析中的“贪心法”&#xff1a; 编译器将程序分解成符号的方法是&#xff0c;从左到右一个字符一个字符地读入&#xff0c;如果该字符可能组成一个符号&#xff0c;那么再读入下一个字符&#xff0c;判断已经读入的两个字符组成的字符串是否可能是一个符号的组成部分&…

Lvs+Nginx+NDS

什么是&#xff1f;为什么&#xff1f;需要负载均衡 一个网站在创建初期&#xff0c;一般来说都是只有一台服务器对用户提供服务 ​ 从图里可以看出&#xff0c;用户经过互联网直接连接了后端服务器&#xff0c;如果这台服务器什么时候突然 GG 了&#xff0c;用户将无法访问这…