UNIX 域数据报套接字

简介: UNIX 域数据报套接字

UNIX域数据报套接字的流程可参考 UDP 套接字编程。


服务器端

服务器端流程如下:

微信截图_20221209161053.png


实现示例:( server.c 文件)

#include <stdio.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <strings.h>
#include <sys/un.h>
#define N 128
#define errlog(errmsg)                                                \
    do {                                                              \
        perror(errmsg);                                               \
        printf("--%s--%s--%d--\n", __FILE__, __FUNCTION__, __LINE__); \
        return -1;                                                    \
    } while (0)
int main(int argc, const char *argv[]) {
    int sockfd;
    struct sockaddr_un serveraddr, clientaddr;
    socklen_t addrlen = sizeof(serveraddr);
    char buf[N] = {};
    //Step1. 创建套接字
    if ((sockfd = socket(AF_UNIX, SOCK_DGRAM, 0)) < 0) {
        errlog("fail to socket");
    }
    // Step2. 填充服务器本地信息结构体
    serveraddr.sun_family = AF_UNIX;
    strcpy(serveraddr.sun_path, argv[1]);
    // Step3. 将套接字与服务器本地信息结构体绑定
    if (bind(sockfd, (struct sockaddr *)&serveraddr, addrlen) < 0) {
        errlog("fail to bind");
    }
    ssize_t bytes;
    while (1) {
        if ((bytes = recvfrom(sockfd, buf, N, 0, (struct sockaddr *)&clientaddr, &addrlen)) < 0) {
            errlog("fail to recvfrom");
        } else {
            printf("clientaddr.sun_path = %s\n", clientaddr.sun_path);
            if (strncmp(buf, "quit", 4) == 0) {
                printf("client quit\n");
            } else {
                printf("client: %s\n", buf);
                strcat(buf, " *_*");
                if (sendto(sockfd, buf, N, 0, (struct sockaddr *)&clientaddr, addrlen) < 0) {
                    errlog("fail to sendto");
                }
            }
        }
    }
    close(sockfd);
    return 0;
}点击复制复制失败已复制


客户端

客户端流程如下:

微信截图_20221209161104.png


实现示例:( client.c 文件)

#include <stdio.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/un.h>
#define N 128
#define errlog(errmsg)                                                \
    do {                                                              \
        perror(errmsg);                                               \
        printf("--%s--%s--%d--\n", __FILE__, __FUNCTION__, __LINE__); \
        return -1;                                                    \
    } while (0)
int main(int argc, const char *argv[]) {
    int sockfd;
    struct sockaddr_un serveraddr;
    socklen_t addrlen = sizeof(serveraddr);
    char buf[N] = {};
    // Step1. 创建套接字
    if ((sockfd = socket(AF_UNIX, SOCK_DGRAM, 0)) < 0) {
        errlog("fail to socket");
    }
    // Step2. 填充服务器本地信息结构体
    serveraddr.sun_family = AF_UNIX;
    strcpy(serveraddr.sun_path, argv[1]);
    // 客户端需要绑定自己的信息,否则服务器无法给客户端发送数据
    struct sockaddr_un clientaddr;
    clientaddr.sun_family = AF_UNIX;
    strcpy(clientaddr.sun_path, argv[2]);
    if (bind(sockfd, (struct sockaddr *)&clientaddr, addrlen) < 0) {
        errlog("fail to bind");
    }
    while (1) {
        fgets(buf, N, stdin);
        buf[strlen(buf) - 1] = '\0';
        if (sendto(sockfd, buf, N, 0, (struct sockaddr *)&serveraddr, addrlen) < 0) {
            errlog("fail to sendto");
        }
        if (strncmp(buf, "quit", 4) == 0) {
            printf("client is quited\n");
            break;
        } else {
            if (recvfrom(sockfd, buf, N, 0, (struct sockaddr *)&serveraddr, &addrlen) < 0) {
                errlog("fail to recv");
            }
            printf("server: %s\n", buf);
        }
    }
    close(sockfd);
    return 0;
}点击复制复制失败已复制


上述客户端代码中,需要注意的是,UNIX域数据报套接字与网络数据报套接字( UDP )不同的是,前者的客户端一定要绑定自己的信息结构体。数据报套接字在用于网络通信时,服务器可以为客户端自动分配地址和端口。而在本地通信时,服务器不会为客户端分配套接字文件(本地信息结构体)。如果客户端不绑定自己的信息结构体,服务器则无法知道客户端是谁。


运行交互

先运行服务端,后运行客户端,在客户端输入发送的数据即可。


运行服务端:

$ gcc server.c -o server && ./server dgram点击复制复制失败已复制


运行客户端:

$ gcc client.c -o client && ./client dgram dgram_client点击复制复制失败已复制


效果如下:

演示.gif


提示

运行会生成两个文件 dgramdgram_client ,与UNIX域流式套接字一样,不删除无法启动,但是有一点不同,可以先运行客户端,后运行服务端

目录
相关文章
|
Unix Linux
Linux本地套接字(Unix域套接字)----SOCK_DGRAM方式
这里介绍一下Linux进程间通信的socket方式---Local socket。这篇主要是介绍下SOCK_DGRAM方式的通信,即数据包的方式(与UDP类似),面向无连接。
307 0
|
网络协议 Unix
UNIX 域流式套接字
UNIX 域流式套接字
161 0
UNIX 域流式套接字
|
缓存 网络协议 Unix
UNIX 域流式套接字
UNIX 域流式套接字
125 0
UNIX 域流式套接字
|
网络协议 Unix
UNIX 域套接字概述
UNIX 域套接字概述
162 0
|
Unix 网络协议
UNIX域协议(无名套接字)
关于什么是UNIX域套接字可以参考:http://www.cnblogs.com/xcywt/p/8185597.html这里主要介绍非命名的UNIX域套接字的用法。1.socketpair函数先看man手册: SYNOPSIS       #include           /* See ...
1270 0
|
网络协议 Unix C语言
UNIX域协议(命名套接字)
这里主要介绍命名UNIX域套接字 1.什么是UNIX域套接字Unix域协议并不是一个实际的协议族,而是在单个主机上执行客户/服务通信的一种方式。是进程间通信(IPC)的一种方式。它提供了两类套接字:字节流套接字(有点像TCP)和数据报套接字(有点像UDP)UNIX域数据报服务是可靠的,不会丢失消息,也不会传递出错。
1237 0
|
缓存 Unix 程序员
《现代体系结构上的UNIX系统:内核程序员的对称多处理和缓存技术(修订版)》——2.10 独立的指令高速缓存和数据高速缓存
将指令高速缓存和数据高速缓存分开的做法目前在计算机系统中相当常见。这种做法能够有效地使高速缓存的带宽加倍,因为它能让CPU从指令高速缓存预取指令的同时把数据载入或者保存到数据高速缓存中。图2-17描绘了这样的一种组织结构。
1327 0
|
网络协议 Unix API
《UNIX网络编程 卷1:套接字联网API(第3版)》——导读
本书想达到的目标是向大家提供网络编程指导。这些内容不仅适用于专业人士,也适用于初学者;不仅适用于维护已有代码,也适用于开发新的网络应用程序;此外,还适用于那些只是想了解一下自己系统中网络组件的工作原理的人。
1472 0
|
网络协议 Unix API
《UNIX网络编程 卷1:套接字联网API(第3版)》——8.16 小结
UDP套接字可能产生异步错误,它们是在分组发送完一段时间后才报告的错误。TCP套接字总是给应用进程报告这些错误,但是UDP套接字必须已连接才能接收这些错误。
1493 0