13.2 Linux_网络编程_UNIX域套接字

devtools/2024/10/22 4:32:39/

概述 

什么是UNIX域套接字:

UNIX域套接字是使用套接字进行本地通信,TCP/UDP是使用套接字进行网络通信。UNIX域套接字也有域流式套接字和域数据报套接字,这两种形式域TCP/UDP的含义类似,使用步骤也完全一致。

bind时绑定的结构体类型:

struct sockaddr_un {sa_family_t sun_family;   //协议类型char sun_path[104];       //套接字文件路径
};

域流式套接字

域流式套接字的创建步骤与TCP创建步骤一致。

server.c代码:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/un.h>
#include <sys/socket.h>
#include <signal.h>
#include <sys/wait.h>#define UNIX_PATH "./UNIX"
void Set_SIGCHLD(void);
void SIGCHLD_Handler(int sig);
int main(int argc ,char** argv){int fd;struct sockaddr_un addr;//判断参数有效性if(argc != 2){printf("param err\n");printf("%s<unix path>\n",argv[0]);return -1;}printf("unix path = %s\n",argv[1]);//1.创建socketif((fd=socket(AF_UNIX,SOCK_STREAM,0))<0){//流式UNIXperror("socket");return -1;}//2.绑定addr.sun_family = AF_UNIX;    				  			//UNIXstrncpy(addr.sun_path,argv[1],sizeof(addr.sun_path)); 	//UNIX文件路径if(bind(fd,(struct sockaddr*)&addr,sizeof(struct sockaddr_un)) == -1){perror("bind");return -1;}//3.监听socketif(listen(fd,5) == -1){ 	//允许最多接入5个客户端perror("listen");return -1;}//多进程并发pid_t pid;int newFd;struct sockaddr_un newAddr;socklen_t newAddrlen;Set_SIGCHLD();//以信号方式回收子进程while(1){//4.接受客户端链接if((newFd = accept(fd,(struct sockaddr*)&newAddr,&newAddrlen)) < 0){perror("accept");return -1;}//父进程处理接收客户端链接的问题//子进程处理与客户端交互的问题if((pid=fork()) == -1){perror("fork");return -1;}else if(pid == 0){char buf[100] = {0};close(fd);//对于子进程,socket返回的fd没有用//5.数据交互while(1){memset(buf,0,sizeof(buf));write(newFd,"server",strlen("server\n"));read(newFd,buf,sizeof(buf)-1);printf("read:%s\n",buf);sleep(1);}exit(0);}else{close(newFd);//对于父进程,accept返回的newFd没有用}}remove(argv[1]);close(fd);return 0;
}
void Set_SIGCHLD(void){struct sigaction act;act.sa_handler = SIGCHLD_Handler;sigemptyset(&act.sa_mask);act.sa_flags = SA_RESTART;//让因为信号而终止的系统调用继续运行if(sigaction(SIGCHLD,&act,NULL) != 0){perror("sigaction");}
}
void SIGCHLD_Handler(int sig){int wstatus;waitpid(-1,&wstatus,WNOHANG);if(WIFEXITED(wstatus)){      //判断子进程是否正常退出printf("子进程的返回值为%d\n",WEXITSTATUS(wstatus));}else{printf("子进程是否被信号结束%d\n",WIFSIGNALED(wstatus));printf("结束子进程的信号类型%d\n",WTERMSIG(wstatus));}
}

client.c代码 :

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/un.h>
#include <sys/socket.h>int main(int argc,char** argv){int fd;struct sockaddr_un addr;//判断参数有效性if(argc != 2){printf("param err\n");printf("%s<unix path>\n",argv[0]);return -1;}printf("unix path = %s\n",argv[1]);//1.创建socketif((fd=socket(AF_UNIX,SOCK_STREAM,0))<0){//流式UNIXperror("socket");return -1;}//2.链接服务器addr.sun_family = AF_UNIX;    				  			//UNIXstrncpy(addr.sun_path,argv[1],sizeof(addr.sun_path)); 	//UNIX文件路径if(connect(fd,(struct sockaddr*)&addr,sizeof(struct sockaddr_un)) == -1){perror("connect");return -1;}//3.数据交互char buf[100] = {0};while(1){if(read(fd,buf,sizeof(buf)-1) > 0){printf("read:%s\n",buf);write(fd,"client:i read it\n",strlen("client:i read it\n"));}}close(fd);return 0;
}

域数据报套接字

server.c

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/un.h>
#include <sys/socket.h>int main(int argc,char** argv){int fd;struct sockaddr_un addr;//判断参数有效性if(argc != 2){printf("param err\n");printf("%s<unix path>\n",argv[0]);return -1;}printf("unix path = %s\n",argv[1]);//1.创建socketif((fd=socket(AF_UNIX,SOCK_DGRAM,0))<0){//数据报UNIXperror("socket");return -1;}//2.绑定IP、端口号addr.sun_family = AF_UNIX;    				  			//UNIXstrncpy(addr.sun_path,argv[1],sizeof(addr.sun_path)); 	//UNIX文件路径if(bind(fd,(struct sockaddr*)&addr,sizeof(struct sockaddr_un)) == -1){perror("bind");return -1;}//3.数据交互char buf[100] = {0};struct sockaddr_un src_addr;socklen_t src_addrlen;while(1){memset(buf,0,sizeof(buf));if(recvfrom(fd,buf,sizeof(buf)-1,0,(struct sockaddr*)&src_addr,&src_addrlen) > 0){printf("read:%s\n",buf);}	}close(fd);return 0;
}

client.c

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/un.h>
#include <sys/socket.h>int main(int argc,char** argv){int fd;struct sockaddr_un addr;//判断参数有效性if(argc != 2){printf("param err\n");printf("%s<unix path>\n",argv[0]);return -1;}printf("unix path = %s\n",argv[1]);//1.创建socketif((fd=socket(AF_UNIX,SOCK_DGRAM,0))<0){//数据报UNIXperror("socket");return -1;}//2.设置要发送到的服务器信息addr.sun_family = AF_UNIX;    				  			//UNIXstrncpy(addr.sun_path,argv[1],sizeof(addr.sun_path)); 	//UNIX文件路径//3.数据交互while(1){sendto(fd,"cilent",strlen("cilent"),0,(struct sockaddr*)&addr,sizeof(addr));sleep(1);}close(fd);return 0;
}


http://www.ppmy.cn/devtools/127735.html

相关文章

pycharm中使用ctrl+鼠标滚轮改变字体大小

文章目录 pycharm使用ctrl鼠标滚轮改变字体大小1.打开pycharm选择file2.选择setting4.选择keymap&#xff0c;然后再右边的输入框中输入increase进行增大字体4.鼠标选择后&#xff0c;点击添加鼠标快捷方式&#xff0c;然后设置鼠标滚轮往上增大字体。5.设置缩小字体&#xff0…

Github 2024-10-14开源项目周报Top14

根据Github Trendings的统计,本周(2024-10-14统计)共有14个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Python项目7C++项目2C项目2Swift项目1Jupyter Notebook项目1Java项目1Rust项目1Python中的算法实现集合 创建周期:2831 天开发语言:Python协议…

【Linux】Linux命令行与环境变量

1.命令行 前⾯写C语⾔时&#xff0c;很少关注过 main 函数的参数&#xff0c;也没有考虑过 main 为什么会有参 数。 实际上在C语⾔中&#xff0c; main 函数⼀共有三个参数&#xff0c;在命令⾏部分先关注前两个参数&#xff1a; 1. argc&#xff1a;表示 main 函数接收到参…

python excel如何转成json,并且如何解决excel转成json时中文汉字乱码的问题

1.解决excel转成json时中文汉字乱码的问题 真的好久没有打开这个博客也好久没有想起来记录一下问题了&#xff0c;今天将表格测试集转成json格式的时候遇到了汉字都变成了乱码的问题&#xff0c;虽然这不是个大问题&#xff0c;但是编码问题挺烦人的&#xff0c;乱码之后像下图…

【Flutter】Dart:环境搭建

Flutter 是一个基于 Dart 的跨平台开发框架&#xff0c;可以帮助我们快速构建移动应用程序。在开始 Flutter 开发之前&#xff0c;我们需要先搭建 Dart 的开发环境&#xff0c;并配置合适的编辑器&#xff0c;比如 VSCode。本教程将引导你一步步完成 Dart 和 Flutter 的环境搭建…

第二十四节 图像直方图

图像直方图的解释 图像直方图是图像像素值的统计学特征&#xff0c;计算代价小&#xff0c;具有图像平移&#xff0c;旋转&#xff0c;缩放不变性的众多优点&#xff0c;广泛的应用于图像处理的各个领域&#xff0c;特别是灰度图像的阈值风格&#xff0c;基于颜色的图像检索以…

Uboot是如何发现Devicetree并将它传递给Linux的

首先我们要知道Uboot对于Devicetree的处理有两种不同的方式&#xff1a; CONFIG_OF_EMBED: 此配置将设备树嵌入到Uboot的镜像中&#xff0c;因此Uboot也能够很轻松的发现设备树并将其加载到内存中。CONFIG_OF_SEPARATE: 此配置意味着设备树单独保存在存储中&#xff0c;例如在…

信息学奥赛复赛复习18-CSP-J2022-01解密-二分答案、二分找边界、二分时间复杂度、二分求最小

PDF文档回复:20241017 1 P8814 [CSP-J 2022] 解密 [题目描述] 给定一个正整数 k&#xff0c;有 k 次询问&#xff0c;每次给定三个正整数 ni,ei,di&#xff0c;求两个正整数 pi,qi&#xff0c;使 nipiqi、eidi(pi−1)(qi−1)1 [输入格式] 第一行一个正整数 k&#xff0c;表…