Linux: 进程间通信机制

文章目录

  • 1. 前言
  • 2. 进程间通信机制
    • 2.1 管道
      • 2.1.1 匿名管道
      • 2.1.2 popen() 和 pclose()
      • 2.1.3 命名管道 FIFO
    • 2.2 消息队列
    • 2.3 共享内存
    • 2.4 信号量
    • 2.5 网络套接字
    • 2.6 UNIX套接字
    • 2.7 信号
  • 3. 参考资料

1. 前言

限于作者能力水平,本文可能存在谬误,因此而给读者带来的损失,作者不做任何承诺。

2. 进程间通信机制

进程间通信机制,简称 IPC(InterProcess Communication)

2.1 管道

管道特性:

1. 半双工,数据是单向流动。
2. 只能在有共同祖先的进程间使用。
3. 数据一旦被读走,便不在管道中存在,不可反复读取。

在这里插入图片描述

2.1.1 匿名管道

可通过系统调用 pipe() 创建匿名管道
在这里插入图片描述
pipe() 调用返回两个文件句柄:filedes[0] 用于读,filedes[1] 用于写。用于读的句柄在所有数据被读取后,继续读返回0指示达到数据末尾;如果管道读端的 filedes[0] 关闭了,再向管道写端的 filedes[1] 写入,会产生 SIGPIPE 信号。
例子代码:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>

int main(void)
{
	int n, fd[2];
	pid_t pid;
	char buf[4096];
	
	if (pipe(fd)) {
		perror("pipe");
		exit(EXIT_FAILURE);
	}

	pid = fork();
	if (pid < 0) {
		perror("fork");
		exit(EXIT_FAILURE);
	}

	if (pid > 0) { /* 父进程 */
		close(fd[0]); /* 父进程关闭读端 */
		write(fd[1], "hello, child, i'm your father.\n", 31);
	} else if (pid == 0) { /* 子进程 */
		close(fd[1]); /* 子进程关闭写端 */
		n = read(fd[0], buf, sizeof(buf));
		write(STDOUT_FILENO, buf, n);
	}

	return 0;
}

例子中,管道数据流向是 父进程 -> 子进程,我们也完全可以通过关闭父进程的写端和子进程的读端,创建数据流向完全相反的管道。

2.1.2 popen() 和 pclose()

前述匿名管道方式使用起来不是很方便,这时候可以使用 popen()
在这里插入图片描述
函数 popen() 先执行 fork() ,然后调用 exec() 执行 cmdstring 参数指向的程序,最后返回一个标准 I/O 的文件对象指针。
如果 type 参数是 "r" ,则文件对象指针连接到 cmdstring 参数指向程序的标准输出:
在这里插入图片描述
也就是 cmdstring 参数指向程序所在子进程,通过管道将其标准输出数据传递给父进程。
如果 type 参数是 "w" ,则文件对象指针连接到 cmdstring 参数指向程序的标准输入:
在这里插入图片描述
也就是 cmdstring 参数指向程序所在子进程的标准输入,通过管道从父进程读取。
说白了,本质上 popen() 就是对 pipe() 的封装,提高了使用的便利性。
看一个 popen() 使用的例子:

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>

int main(int argc, char *argv[])
{
	FILE *fp;
	char buf[4096] = {0};
	int n;

	fp = popen("/bin/ls -l", "r");
	if (!fp) {
		perror("popen");
		exit(EXIT_FAILURE);
	}

	n = fread(buf, 1, sizeof(buf), fp);
	printf("%d bytes\n%s\n", n, buf);

	pclose(fp);
	
	return 0;
}

2.1.3 命名管道 FIFO

前述的管道方式,只能用于有亲缘关系的进程,不相关的进程之间,可以使用命名管道(FIFO)进行数据传输。通过函数 mkfifo() 创建命名管道(FIFO)
在这里插入图片描述
一旦创建了命名管道(FIFO)文件,就可以通过 open() 打开它,然后进行 read(), write(), close(),unlink() 等操作。
类似于匿名管道,在没有读端进程打开命名管道文件时,如果写端进程对命名管道(FIFO)进行写操作,则会产生 SIGPIPE 信号。
多个进程写同一个命名管道(FIFO),这是很常见的情形,如果不想各进程写入的数据相互穿插,就需要考虑同步问题。
命名管道(FIFO)常见用途如下:

1. shell命令间相互传递数据。
2. 客户/服务端程序,以便客户进程和服务端进程之间传递数据。

看一个例子:

/* FIFO 写端程序:fifo_wend.c */

#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>

#include "fifo_test.h"

int main(void)
{
    int fd;
	char s[] = "hello!\n";

	fd = open(FIFO_FILE, O_WRONLY);
	write(fd, s, sizeof(s));
	close(fd);

	return 0;
}
/* FIFO 读端程序:fifo_rend.c */

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

#include "fifo_test.h"

int main(void)
{
	char buf[4096];
    int fd;

	fd = open(FIFO_FILE, O_RDONLY);
	read(fd, buf, sizeof(buf));
	close(fd);

	printf("%s", buf);

	return 0;
}

公共头文件 fifo_test.h

#define FIFO_FILE "fifo"

编译和运行:

$ make fifo_rend fifo_wend
$ mkfifo fifo  创建 FIFO 命名管道文件 fifo
$ ./fifo_rend 先启动读端程序
$ ./fifo_wend 后启动写端程序
$ rm fifo 测试完成后删除 fifo 文件

读端程序 fifo_rend 会输出 hello! 字串。

2.2 消息队列

管道的方式虽然简单,但是比较低效,不适合进程间频繁的数据交流,并且管道只能传输无格式字节流。对于进程间频繁的数据交流,如果要求高效、带有格式,使能消息队列将是更好的选择。消息队列的数据在进程间的流动如下图(图片取自参考资料中知乎博文):
在这里插入图片描述
创建消息队列函数 msgget()
在这里插入图片描述
更多细节参考 https://man7.org/linux/man-pages/man2/msgget.2.html 。
配置消息队列函数 msgctl()
在这里插入图片描述
更多细节参考 https://man7.org/linux/man-pages/man2/msgctl.2.html 。
发送数据到消息队列函数 msgsnd()
在这里插入图片描述
更多细节参考 https://man7.org/linux/man-pages/man2/msgsnd.2.html 。
从消息队列获取数据函数 :
在这里插入图片描述
更多细节参考 https://man7.org/linux/man-pages/man2/msgrcv.2.html 。
看一个例子,包含消息发送和接收两个程序。程序公共头文件 msg_test.h

#define MSGQ_ID 1234
#define MSG_END "<END>"

#define MAX_TEXT 4096

struct msg_data {
	long int msg_type;
	char text[MAX_TEXT];
};

消息发送程序代码 msg_snd.c

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/msg.h>
#include <errno.h>

#include "msg_test.h"


int main(int argc, char **argv)
{
	struct msg_data data;
	int msgid = -1;
	static const char *msg_text[] = {
		"message 001", 
		"message 002",
		"message 003",
		MSG_END
	};
	int i;

	/* create */
	msgid = msgget((key_t)MSGQ_ID, 0666 | IPC_CREAT);
	if (msgid == -1) {
		perror("msgget");
		exit(EXIT_FAILURE);
	}

	for (i = 0; i < sizeof(msg_text)/sizeof(msg_text[0]); i++) {
		data.msg_type = 1;
		strcpy(data.text, msg_text[i]);
		
		if (msgsnd(msgid, (void *)&data, MAX_TEXT, 0) == -1) { /* send text message */
			perror("msgsnd");
			exit(EXIT_FAILURE);
		}
	}

	return 0;
}

消息接收程序代码 msg_rcv.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/msg.h>
#include <errno.h>

#include "msg_test.h"

int main(int argc, char **argv)
{
	int msgid = -1;
	struct msg_data data;
	long int msgtype = 0;

	msgid = msgget((key_t)MSGQ_ID, 0666 | IPC_CREAT);
	if (msgid == -1) {
		perror("msgget");
		exit(EXIT_FAILURE);
	}

	for (;;) {
		if (msgrcv(msgid, (void *)&data, MAX_TEXT, msgtype, 0) == -1) {
			perror("msgrcv");
			exit(EXIT_FAILURE);
		}

		printf("<MESSAGE>: %s\n", data.text);
		
		if (strcmp(data.text, MSG_END) == 0)
			break;
	}

	/* remove */
	if (msgctl(msgid, IPC_RMID, 0) == -1) {
		perror("msgget");
		exit(EXIT_FAILURE);
	}

	return 0;
}

编译和运行:

$ make msg_snd msg_rcv
$ ./msg_rcv
$ ./msg_snd

当然,消息队列也可用于父子进程间的通信,感兴趣的读者可以自行实践。

2.3 共享内存

为了避免消息队列那样频繁的数据拷贝,以及系统调用开销,引入了共享内存 IPC 机制。
共享内存 允许两个或更多进程共享一块物理内存区域,因为数据不需要在进程间来回拷贝,所以这是本文所有 IPC 机制中最快的一种。共享内存的各进程页表结构如下图(图片取自参考资料中知乎博文):
在这里插入图片描述
上图告诉我们,共享内存的物理区域是相同的,进程使用各自页表项映射到相同的物理内存块。也可以从另一个视角来理解(图片取自参考资料中CSDN博文):
在这里插入图片描述
这里更清晰的看到,共享内存的物理内存区,在进程中对应的虚拟地址可能是不同的。再看下共享内存在进程虚拟地址空间布局中的位置:
在这里插入图片描述
看共享内存相关函数:

#include <sys/shm.h>

int shmget(key_t key, size_t size, int shmflg); /* 创建 或 获取 共享内存 */
int shmctl(int shmid, int cmd, struct shmid_ds *buf); /* 配置共享内存 */
void *shmat(int shmid, const void *shmaddr, int shmflg); /* 建立到共享内存的映射 */
int shmdt(const void *shmaddr); /* 移除到共享内存的映射 */

看一个例子,仍然是一个写程序和一个读程序。两个程序的共同头文件 shm_common.h

#ifndef SHM_COMMON_H
#define SHM_COMMON_H

#define MY_SHM_KEY 1234
#define TEXT_SZ 2048
#define MSG_END "<END>"

struct shared_data {
    int data_avail;
    char text[TEXT_SZ];
};

#endif /* SHM_COMMON_H */

写程序 shm_write.c

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>

#include "shm_common.h"

int main (void)
{
    int running = 1;
	int shmid;
    void *shared_vaddr = NULL;
    struct shared_data *shared;
    char buffer[BUFSIZ];
	static const char *msg_text[] = {
		"message 001", 
		"message 002",
		"message 003",
		MSG_END
	};
	int i;

    shmid = shmget((key_t)MY_SHM_KEY, sizeof(struct shared_data), 0666 | IPC_CREAT);
    if (shmid == -1) {
		perror("shmget");
        exit(EXIT_FAILURE);
    }

    shared_vaddr = shmat(shmid, NULL, 0);
    if (shared_vaddr == (void *)-1) {
        perror("shmat");
        exit(EXIT_FAILURE);
    }

    printf("Memory attached at %p\n", shared_vaddr);
	
    shared = (struct shared_data *)shared_vaddr;
	
    for (i = 0; i < sizeof(msg_text)/sizeof(msg_text[0]); i++) {
        while (shared->data_avail)
			asm("nop");
		
        strcpy(shared->text, msg_text[i]);
        shared->data_avail = 1;
    }

    if (shmdt(shared_vaddr) == -1) {
        perror("shmdt");
        exit(EXIT_FAILURE);
    }

    return 0;
}

读程序 shm_read.c

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <error.h>

#include "shm_common.h"

int main (void)
{
	int shmid;
    void *shared_vaddr = NULL;
    struct shared_data *shared;

	shmid = shmget((key_t)MY_SHM_KEY, sizeof(struct shared_data), 0666 | IPC_CREAT);
    if (shmid == -1) {
		perror("shmget");
        exit(EXIT_FAILURE);
    }

    shared_vaddr = shmat(shmid, NULL, 0);
    if (shared_vaddr == (void *)-1) {
        perror("shmat");
        exit(EXIT_FAILURE);
    }

    printf("Memory attached at %p\n", shared_vaddr);
	
    shared = (struct shared_data *)shared_vaddr;
    shared->data_avail = 0;

    for (;;) {
        if (shared->data_avail) {
            printf("%s\n", shared->text);

			if (strcmp(shared->text, MSG_END) == 0)
				break;

			shared->data_avail = 0;
      	}
    }

    if (shmdt(shared_vaddr) == -1) {
        perror("shmdt");
        exit(EXIT_FAILURE);
    }

    return 0;
}

编译和运行:

$ make shm_read shm_write
$ ./shm_read
$ ./shm_write

同样,共享内存也可用于父子进程间的通信,感兴趣的读者可以自行实践。

2.4 信号量

先看信号量函数集:

#include <sys/sem.h>

int semget(key_t key, int nsems, int semflg); /* 创建 或 获取 一个已有信号量 */
int semctl(int semid, int semnum, int cmd, ...); /* 配置信号量 */
int semop(int semid, struct sembuf *sops, size_t nsops); /* 改变信号量的值 */
int semtimedop(int semid, struct sembuf *sops, size_t nsops,
				const struct timespec *timeout);

看一个例子,代码来自此处:

#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/sem.h>
#include <errno.h>

union semun {
    int val;
    struct semid_ds *buf;
    unsigned short *arry;
};

static int init_semaphore(int sem_id);
static int fini_semaphore(int sem_id);

static int semaphore_p(int sem_id);
static int semaphore_v(int sem_id);


int main(int argc, char *argv[])
{
    char message = 'X';
    int sem_id, i = 0;
 
    sem_id = semget((key_t)1234, 1, 0666 | IPC_CREAT);
	if (sem_id == -1) {
		perror("semget");
		exit(EXIT_FAILURE);
	}
 
    if (argc > 1) { /* first time to init */
        if (!init_semaphore(sem_id))
            exit(EXIT_FAILURE);
 
        message = argv[1][0];
        sleep(2);
    }
 
    for (i = 0; i < 10; ++i) {
        if (!semaphore_p(sem_id))
            exit(EXIT_FAILURE);

        printf("%c", message);
 
        fflush(stdout);
        sleep(rand() % 3);

        printf("%c\n", message);
        fflush(stdout);
 
        if (!semaphore_v(sem_id))
            exit(EXIT_FAILURE);

        sleep(rand() % 2);
    }
 
    sleep(10);
    printf("\n%d - finished\n", getpid());
 
    if (argc > 1) {
        sleep(3);
        fini_semaphore(sem_id);
    }

    exit(EXIT_SUCCESS);
}

static int init_semaphore(int sem_id)
{
    union semun sem_union;
 
    sem_union.val = 1;
    if (semctl(sem_id, 0, SETVAL, sem_union) == -1) {
		perror("semctl");
        return 0;
	}

    return 1;
}
 
static int fini_semaphore(int sem_id)
{
    union semun sem_union;
 
    if (semctl(sem_id, 0, IPC_RMID, sem_union) == -1) {
        perror("semctl");
		return 0;
	}

	return 1;
}
 
static int semaphore_p(int sem_id)
{
    struct sembuf sem_b;

    sem_b.sem_num = 0;
    sem_b.sem_op = -1;
    sem_b.sem_flg = SEM_UNDO;
    if (semop(sem_id, &sem_b, 1) == -1) {
        perror("semop");
        return 0;
    }
 
    return 1;
}
 
static int semaphore_v(int sem_id)
{
    struct sembuf sem_b;

    sem_b.sem_num = 0;
    sem_b.sem_op = 1;
    sem_b.sem_flg = SEM_UNDO;
    if (semop(sem_id, &sem_b, 1) == -1) {
        perror("semop");
        return 0;
    }
 
    return 1;
}

编译和运行:

$ make sem_test
$ ./sem_test 0
$ ./sem_test

2.5 网络套接字

就上张图吧,其它的就不多说了。
在这里插入图片描述
图片来自于 这里 。

2.6 UNIX套接字

有点类似于命名管道(FIFO),也许我们可以把UNIX套接字说成命名socket。更多细节参考博客 https://blog.csdn.net/inthat/article/details/27371939 。

2.7 信号

#include <signal.h>

int kill(pid_t pid, int sig); /* 发送信号到进程 */
int killpg(int pgrp, int sig); /* 发送信号到进程组 */
int tgkill(pid_t tgid, pid_t tid, int sig); /* 发送信号到线程 */
int syscall(SYS_pidfd_send_signal, int pidfd, int sig, siginfo_t *info,
                             unsigned int flags); /* 发送信号到进程 */

3. 参考资料

《Advanced Programming in the UNIX Envinronment》
https://zhuanlan.zhihu.com/p/556004836
https://blog.csdn.net/fukangwei_lite/article/details/120640540
https://blog.csdn.net/m0_52387305/article/details/124149859
https://blog.csdn.net/inthat/article/details/27371939

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Javaee Spring的AOP简介

一.Spring的AOP简介 1.1 什么是AOP AOP 为 Aspect Oriented Programming 的缩写&#xff0c;意思为面向切面编程&#xff0c;是通过预编译方式和运行期动态代 理实现程序功能的统一维护的一种技术。AOP 是 OOP 的延续&#xff0c;是软件开发中的一个热点&#xff0c;也是…

shell脚本控制

shell脚本编程系列 处理信号 Linux利用信号与系统中的进程进行通信&#xff0c;通过对脚本进行编程&#xff0c;使其在收到特定信号时执行某些命令&#xff0c;从而控制shell脚本的操作。 Linux信号 shell脚本编程会遇到的最常见的Linux系统信号如下表所示&#xff1a; 在默…

【获奖案例巡展】信创先锋之星——浙江省某市区视频能力中心

为表彰使用大数据、人工智能等基础软件为企业、行业或世界做出杰出贡献和巨大创新的标杆项目&#xff0c;星环科技自2021年推出了“新科技 星力量” 星环科技科技实践案例评选活动&#xff0c;旨在为各行业提供更多的优秀产品案例&#xff0c;彰显技术改变世界的力量&#xff0…

Cycling 74 Max for Mac:音乐可视化编程软件

Cycling 74 Max是一款音乐、视觉、互动艺术等领域中广泛使用的编程语言和应用软件&#xff0c;它允许用户创作和控制实时音频和视频效果、交互式应用程序和媒体艺术品等。 Max将程序设计和可视化编程相结合&#xff0c;通过简单的拖拽和连接方式&#xff0c;用户可以将各种功能…

基于springboot的大学生租房系统源码论文数据库

3.1系统功能 现在无论是在PC上还是在手机上&#xff0c;相信全国所有地方都在进行大学生租房管理。随着经济的不断发展&#xff0c;系统管理也在不断增多&#xff0c;大学生租房系统就是其中一种&#xff0c;很多人会登录到相关的租房系统查看租房信息&#xff0c;还能查看房屋…

微信小程序开发--利用和风天气API实现天气预报小程序

本来是参照《微信小程序开发实战》做一个天气预报小程序的&#xff0c;实际运行的时候提示错误&#xff0c;code 400&#xff0c;参数错误。说明问题应该出在查询API的语句上&#xff0c;没有返回结果。 查阅后才知道&#xff0c;可能书籍出版时间较早&#xff0c;现在的和风获…

类对象

一、类初识 类&#xff1a;表示一种事物所具有的共同特征和行为 对象&#xff1a;一个类的实例 如下图&#xff0c;通过狗这个类进行详解 这是一个Dog类 对象&#xff1a;斗牛犬、小猎犬、牧羊犬 类中的属性&#xff1a;breed(品种)、size(大小)、color(颜色)、age(年龄)、 …

安全常见基础名词概念

一、域名 1、域名&#xff1a;相当网站的名字&#xff0c;互联网上某一台计算机或计算机组的名称&#xff0c;用于在数据传输时标识计算机的电子方位。 2、网域名系统&#xff08;Domain Name System&#xff09;有时也简称为域名&#xff08;DNS&#xff09;&#xff0c;是互…

探索【Stable-Diffusion WEBUI】的插件:骨骼姿态(OpenPose)

文章目录 &#xff08;零&#xff09;前言&#xff08;一&#xff09;骨骼姿态&#xff08;OpenPose&#xff09;系列插件&#xff08;二&#xff09;插件&#xff1a;PoseX&#xff08;三&#xff09;插件&#xff1a;Depth Lib&#xff08;四&#xff09;插件&#xff1a;3D …

响应式开发(HTML5CSS3)实现媒体查询的功能案例

目录 前言 一、媒体查询知识点 二、实现功能的尺寸 三、代码部分 1.不带嵌套的媒体查询功能 1.1.代码段 1.2.运行结果 2.带嵌套的媒体查询功能 2.1.代码段 2.2.运行结果 2.2.3视频效果 前言 1.本文讲解的响应式开发技术&#xff08;HTML5CSS3Bootstrap&#xff09…

Auto-GPT免费尝鲜之初体验-使用攻略和总结

Auto-GPT免费尝鲜之初体验-使用攻略和总结 写在前面的废话一、部署 Auto-GPT二、试运行 Auto-GPT三、我踩过的坑四、后续探索 写在前面的废话 ChatGPT 的交互模式&#xff0c;是和一个 “人” 对话聊天。 如果你想了解更多ChatGPT和AI绘画的相关知识&#xff0c;请参考&#…

ArcGIS Pro用户界面

目录 1 功能区 1.1 快速访问工具栏 1.2 自定义快速访问工具栏 1.3 自定义功能区选项 1.3.1 添加组和命令 1.3.2 添加新选项卡 2 视图 3 用户界面排列 ​编辑 4 窗格 4.1 内容窗格 4.2 目录窗格 4.3 目录视图&#xff08;类似ArcCatalog&#xff09; 4.4 浏览对话框…

python:面向对象编程(知识点+代码)

文章目录 一、类和对象1、对象属性的默认值设置2、对象属性的添加、修改与删除3 、类属性 二、类的继承 引言&#xff1a;面向对象编程时一门编程语言重要的功能&#xff0c;我们之前所学的 c&#xff0c;java都为面向对象编程语言&#xff0c;这里给大家拓展一下&#xff0c;…

什么是CDN加速?CDN加速有哪些作用?

一、什么是 CDN CDN 的全称是 Content Delivery Network&#xff0c;即内容分发网络。CDN 是在现有 Internet 基础上增加一层新的网络架构&#xff0c;通过部署边缘服务器&#xff0c;采用负载均衡、内容分发、调度等功能&#xff0c;使用户可以就近访问获取所需内容&#xff…

多维时序 | MATLAB实现BO-CNN-LSTM贝叶斯优化卷积神经网络-长短期记忆网络多变量时间序列预测

多维时序 | MATLAB实现BO-CNN-LSTM贝叶斯优化卷积神经网络-长短期记忆网络多变量时间序列预测 目录 多维时序 | MATLAB实现BO-CNN-LSTM贝叶斯优化卷积神经网络-长短期记忆网络多变量时间序列预测效果一览基本介绍模型搭建程序设计参考资料 效果一览 基本介绍 MATLAB实现BO-CNN-…

2023年报考CSM敏捷教练认证好不好?含金量高吗?

CSM&#xff0c;Certified Scrum Master&#xff0c;是Scrum联盟发起的Scrum认证。帮助个人从自身、团队和组织层面&#xff0c;学习技能和工具来扩展实践的层面&#xff0c;帮助团队正确使用Scrum&#xff0c;从而提高项目整体成功的可能性。 认证收益 职业能力提升 推动企业…

图神经网络能做什么?

从概念上讲&#xff0c;我们可以将图神经网络的基本学习任务分为 5 个不同的方向&#xff1a; &#xff08;1&#xff09;图神 经网络方法&#xff1b; &#xff08;2&#xff09;图神经网络的理论理解&#xff1b; &#xff08;3&#xff09;图神经网络的可扩展性&#xff1b…

VLAN基础实验

实验要求&#xff1a; 1、PC1和PC3所在接口为Access接口 PC2/4/5/6处于同一网段:其中PC2可以访问PC4/5/6 PC4可以访问PC5&#xff0c;但不能访问PC6 PC5不能访问PC6 2、PC1/3与PC2/4/5/6不再同一网段 3、所有PC通过DHCP获取IP地址&#xff0c;且PC1/3可以正常访问PC2/4/5/6 实…

jenkins——环境变量、节点配置以及节点环境变量的使用

这里写目录标题 一、Jenkins 环境变量1、环境变量分类a、系统内置环境变量b、用户自定义环境变量 2、环境变量的使用3、控制台输出 二、Jenkins 节点配置1、Jenkins的常用节点2、准备工作3、SSH类节点配置4、Java web节点配置5、Java web节点配置详细步骤 三、Jenkins 节点环境…

操作系统八股文知识点汇总

1. 程序编译过程 gcc HelloWorld.c -E -o HelloWorld.i 预处理&#xff1a;加入头文件&#xff0c;替换宏。gcc HelloWorld.c -S -c -o HelloWorld.s 编译&#xff1a;包含预处理&#xff0c;将 C 程序转换成汇编程序。gcc HelloWorld.c -c -o HelloWorld.o 汇编&#xff1a;包…