当前位置: 首页 > news >正文

Linux进程间共享内存通信时如何同步?(附源码)

今天我们来讲讲进程间使用共享内存通信时为了确保数据的正确,如何进行同步?

在Linux中,进程间的共享内存通信需要通过同步机制来保证数据的正确性和一致性,常用的同步机制包括信号量互斥锁条件变量等。

其中,使用信号量来同步进程间的共享内存访问是一种常见的方法。每个共享内存区域可以关联一个或多个信号量,以保护共享内存区域的读写操作。在访问共享内存之前,进程需要获取信号量的使用权,当完成读写操作后,再释放信号量的使用权,以便其他进程可以访问共享内存区域。

1、信号量同步

下面是一个简单的示例程序,展示了如何使用信号量来同步共享内存区域的读写操作:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <string.h>#define SHM_SIZE 1024
#define SEM_KEY 0x123456// 定义联合体,用于信号量操作
union semun {int val;struct semid_ds *buf;unsigned short *array;
};int main() {int shmid, semid;char *shmaddr;struct sembuf semops[2];union semun semarg;// 创建共享内存区域shmid = shmget(IPC_PRIVATE, SHM_SIZE, IPC_CREAT | 0666);if (shmid == -1) {perror("shmget");exit(1);}// 将共享内存区域附加到进程地址空间中shmaddr = shmat(shmid, NULL, 0);if (shmaddr == (char *) -1) {perror("shmat");exit(1);}// 创建信号量semid = semget(SEM_KEY, 1, IPC_CREAT | 0666);if (semid == -1) {perror("semget");exit(1);}// 初始化信号量值为1semarg.val = 1;if (semctl(semid, 0, SETVAL, semarg) == -1) {perror("semctl");exit(1);}// 等待信号量semops[0].sem_num = 0;semops[0].sem_op = 0;semops[0].sem_flg = 0;if (semop(semid, semops, 1) == -1) {perror("semop");exit(1);}// 在共享内存中写入数据strncpy(shmaddr, "Hello, world!", SHM_SIZE);// 释放信号量semops[0].sem_num = 0;semops[0].sem_op = 1;semops[0].sem_flg = 0;if (semop(semid, semops, 1) == -1) {perror("semop");exit(1);}// 等待信号量semops[0].sem_num = 0;semops[0].sem_op = 0;semops[0].sem_flg = 0;if (semop(semid, semops, 1) == -1) {perror("semop");exit(1);}// 从共享内存中读取数据printf("Received message: %s\n", shmaddr);// 释放共享内存区域if (shmdt(shmaddr) == -1) {perror("shmdt");exit(1);}// 删除共享内存区域if (shmctl(shmid, IPC_RMID, NULL) == -1) {perror("shmctl");exit(1);}// 删除信号量if (semctl(semid, 0, IPC_RMID, semarg) == -1) {perror("semctl");exit(1);}return 0;

在这个示例程序中,使用了System V信号量来同步共享内存的读写操作。程序首先创建一个共享内存区域,并将其附加到进程地址空间中。然后,使用semget()函数创建一个信号量,并将其初始化为1。在写入共享内存数据之前,程序使用semop()函数等待信号量。一旦获取了信号量的使用权,程序就可以在共享内存区域中写入数据。写入数据完成后,程序再次使用semop()函数释放信号量的使用权。在读取共享内存数据时,程序同样需要等待信号量的使用权,读取数据完成后,再次释放信号量的使用权。

需要注意的是,使用信号量来同步共享内存访问时,需要确保每个进程都按照一定的顺序进行读写操作。否则,就可能出现死锁等问题。因此,在设计进程间共享内存通信时,需要仔细考虑数据的读写顺序,并采取合适的同步机制来确保数据的正确性和一致性。

2、互斥锁同步

互斥量也是一种常用的同步机制,可以用来实现多个进程之间的共享内存访问。在Linux中,可以使用pthread库中的互斥量来实现进程间共享内存的同步。

下面是一个使用互斥量实现共享内存同步的示例程序:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>#define SHM_SIZE 1024// 共享内存结构体
typedef struct {pthread_mutex_t mutex;char data[SHM_SIZE];
} shm_data_t;int main() {int fd;shm_data_t *shm_data;pthread_mutexattr_t mutex_attr;pthread_mutex_t *mutex;// 打开共享内存文件if ((fd = shm_open("/my_shm", O_CREAT | O_RDWR, 0666)) == -1) {perror("shm_open");exit(1);}// 调整共享内存文件大小if (ftruncate(fd, sizeof(shm_data_t)) == -1) {perror("ftruncate");exit(1);}// 将共享内存映射到进程地址空间中if ((shm_data = mmap(NULL, sizeof(shm_data_t), PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0)) == MAP_FAILED) {perror("mmap");exit(1);}// 初始化互斥量属性pthread_mutexattr_init(&mutex_attr);pthread_mutexattr_setpshared(&mutex_attr, PTHREAD_PROCESS_SHARED);// 创建互斥量mutex = &(shm_data->mutex);pthread_mutex_init(mutex, &mutex_attr);// 在共享内存中写入数据pthread_mutex_lock(mutex);sprintf(shm_data->data, "Hello, world!");pthread_mutex_unlock(mutex);// 在共享内存中读取数据pthread_mutex_lock(mutex);printf("Received message: %s\n", shm_data->data);pthread_mutex_unlock(mutex);// 解除共享内存映射if (munmap(shm_data, sizeof(shm_data_t)) == -1) {perror("munmap");exit(1);}// 删除共享内存文件if (shm_unlink("/my_shm") == -1) {perror("shm_unlink");exit(1);}return 0;
}

在这个示例程序中,使用了pthread库中的互斥量来同步共享内存的读写操作。程序首先创建一个共享内存文件,并将其映射到进程地址空间中。然后,使用pthread_mutex_init()函数创建一个互斥量,并将其初始化为共享内存中的一部分。在写入共享内存数据之前,程序使用pthread_mutex_lock()函数等待互斥量。一旦获取了互斥量的使用权,程序就可以在共享内存区域中写入数据。写入数据完成后,程序再次使用pthread_mutex_unlock()函数释放互斥量的使用权。在读取共享内存数据之前,程序再次使用pthread_mutex_lock()函数等待互斥量。一旦获取了互斥量的使用权,程序就可以在共享内存区域中读取数据。读取数据完成后,程序再次使用pthread_mutex_unlock()函数释放互斥量的使用权。最后,程序解除共享内存映射,并删除共享内存文件。

使用互斥量来同步共享内存访问有以下几点注意事项:

1、互斥量需要初始化。在创建互斥量之前,需要使用pthread_mutexattr_init()函数初始化互斥量属性,并使用pthread_mutexattr_setpshared()函数将互斥量属性设置为PTHREAD_PROCESS_SHARED,以便多个进程可以共享互斥量。

2、在访问共享内存之前,需要使用pthread_mutex_lock()函数获取互斥量的使用权。一旦获取了互斥量的使用权,程序才能访问共享内存。在完成共享内存的访问之后,需要使用pthread_mutex_unlock()函数释放互斥量的使用权,以便其他进程可以访问共享内存。

3、互斥量必须存储在共享内存区域中。在创建互斥量时,需要将其初始化为共享内存区域中的一部分,以便多个进程可以访问同一个互斥量。

4、程序必须保证互斥量的一致性。多个进程共享同一个互斥量时,必须保证互斥量的一致性。否则,可能会导致多个进程同时访问共享内存区域,导致数据错误或者系统崩溃。

总之,使用互斥量来同步共享内存访问可以有效地避免多个进程同时访问共享内存区域的问题,从而保证数据的一致性和程序的稳定性。在实际编程中,需要根据具体的需求选择不同的同步机制,以保证程序的正确性和效率。

 资料直通车:Linux内核源码技术学习路线+视频教程内核源码

学习直通车:Linux内核源码内存调优文件系统进程管理设备驱动/网络协议栈

3、条件变量同步

在Linux下,可以使用条件变量(Condition Variable)来实现多进程之间的同步。条件变量通常与互斥量(Mutex)结合使用,以便在共享内存区域中对数据进行同步访问。

条件变量是一种线程同步机制,用于等待或者通知某个事件的发生。当某个进程需要等待某个事件发生时,它可以通过调用pthread_cond_wait()函数来阻塞自己,并将互斥量释放。一旦事件发生,其他进程就可以通过调用pthread_cond_signal()或pthread_cond_broadcast()函数来通知等待线程。等待线程接收到通知后,会重新获取互斥量,并继续执行。

在共享内存通信中,可以使用条件变量来实现进程之间的同步。具体操作步骤如下:

初始化互斥量和条件变量。在创建共享内存之前,需要使用pthread_mutexattr_init()和pthread_condattr_init()函数分别初始化互斥量属性和条件变量属性。然后,需要使用pthread_mutexattr_setpshared()和pthread_condattr_setpshared()函数将互斥量属性和条件变量属性设置为PTHREAD_PROCESS_SHARED,以便多个进程可以共享它们。

等待条件变量。在读取共享内存之前,程序可以使用pthread_cond_wait()函数等待条件变量。调用pthread_cond_wait()函数会自动释放互斥量,并阻塞当前进程。一旦其他进程发送信号通知条件变量发生变化,等待线程就会重新获得互斥量,并继续执行。

发送信号通知条件变量变化。在向共享内存中写入数据之后,程序可以使用pthread_cond_signal()或pthread_cond_broadcast()函数发送信号通知条件变量发生变化。调用pthread_cond_signal()函数会发送一个信号通知等待线程条件变量发生变化,而调用pthread_cond_broadcast()函数会向所有等待线程发送信号通知条件变量发生变化。

使用条件变量来同步共享内存访问有以下几点注意事项:

1、程序必须使用互斥量来保护共享内存。在使用条件变量之前,程序必须先获取互斥量的使用权,以便保护共享内存区域中的数据不被多个进程同时访问。

2、程序必须保证条件变量的一致性。多个进程共享同一个条件变量时,必须保证条件变量的一致性。否则,可能会导致多个进程同时访问共享内存区域,导致数据错误或者系统崩溃。

3、程序必须正确使用条件变量。在使用条件变量时,需要正确地使用pthread_cond_wait()、pthread_cond_signal()和pthread_cond_broadcast()函数,否则可能会导致死锁或者其他问题。

4、程序必须正确处理信号。当调用pthread_cond_wait()函数时,程序可能会因为接收到信号而提前返回,此时程序需要正确地处理信号。

下面是一个使用条件变量实现进程间共享内存同步的示例代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <pthread.h>#define SHM_SIZE 4096
#define SHM_NAME "/myshm"
#define SEM_NAME "/mysem"typedef struct {pthread_mutex_t mutex;pthread_cond_t cond;char buffer[SHM_SIZE];
} shm_t;int main(int argc, char *argv[]) {int fd, pid;shm_t *shm;pthread_mutexattr_t mutex_attr;pthread_condattr_t cond_attr;// 创建共享内存区域fd = shm_open(SHM_NAME, O_CREAT | O_RDWR, S_IRUSR | S_IWUSR);if (fd < 0) {perror("shm_open");exit(1);}// 设置共享内存大小if (ftruncate(fd, sizeof(shm_t)) < 0) {perror("ftruncate");exit(1);}// 将共享内存映射到进程地址空间shm = mmap(NULL, sizeof(shm_t), PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);if (shm == MAP_FAILED) {perror("mmap");exit(1);}// 初始化互斥量属性和条件变量属性pthread_mutexattr_init(&mutex_attr);pthread_condattr_init(&cond_attr);pthread_mutexattr_setpshared(&mutex_attr, PTHREAD_PROCESS_SHARED);pthread_condattr_setpshared(&cond_attr, PTHREAD_PROCESS_SHARED);// 初始化互斥量和条件变量pthread_mutex_init(&shm->mutex, &mutex_attr);pthread_cond_init(&shm->cond, &cond_attr);// 创建子进程pid = fork();if (pid < 0) {perror("fork");exit(1);}if (pid == 0) {// 子进程写入共享内存sleep(1);pthread_mutex_lock(&shm->mutex);sprintf(shm->buffer, "Hello, world!");pthread_cond_signal(&shm->cond);pthread_mutex_unlock(&shm->mutex);exit(0);} else {// 父进程读取共享内存pthread_mutex_lock(&shm->mutex);pthread_cond_wait(&shm->cond, &shm->mutex);printf("Received message: %s\n", shm->buffer);pthread_mutex_unlock(&shm->mutex);}// 删除共享内存if (shm_unlink(SHM_NAME) < 0) {perror("shm_unlink");exit(1);}return 0;
}

在这个示例中,程序创建了一个名为"/myshm"的共享内存区域,并将其映射到进程地址空间中。然后,程序使用互斥量和条件变量来同步进程之间的访问共享内存区域。具体来说,父进程首先锁定互斥量,然后等待条件变量的信号。子进程等待一秒钟后,锁定互斥量,将"Hello, world!"字符串写入共享内存区域,然后发送条件变量信号,并释放互斥量。此时,父进程将收到条件变量信号并锁定互斥量,读取共享内存区域中的内容,并释放互斥量。

需要注意的是,在使用条件变量时,我们需要遵循一些规则来保证程序的正确性,如在等待条件变量时必须锁定互斥量,并使用while循环来检查条件变量的值是否满足要求,等待条件变量信号的线程必须在等待之前锁定互斥量,在等待之后解锁互斥量,等待条件变量信号的线程可能会因为接收到信号而提前返回等等。

总之,使用互斥量和条件变量来实现进程间共享内存通信的同步,需要我们仔细考虑程序中所有可能出现的情况,并正确地使用互斥量和条件变量函数来同步进程之间的访问。

小结

好了,这次我们通过Linux下进程间共享内存通信方式讲解了常用的同步机制:信号量、互斥锁、条件变量。希望对小伙伴们在日常的编程当中有所帮助。

原文作者:Linux兵工厂

 

相关文章:

Linux进程间共享内存通信时如何同步?(附源码)

今天我们来讲讲进程间使用共享内存通信时为了确保数据的正确&#xff0c;如何进行同步? 在Linux中&#xff0c;进程间的共享内存通信需要通过同步机制来保证数据的正确性和一致性&#xff0c;常用的同步机制包括信号量、互斥锁、条件变量等。 其中&#xff0c;使用信号量来同…...

spring注解驱动开发(二)

17、Bean的生命周期 bean的生命周期&#xff1a;bean的创建—初始化—销毁的过程 容器负责管理bean的生命周期 我们可以自定义初始化和销毁方法&#xff0c;容器在bean进行到当前生命周期的时候来调用我们自定义的初始化和销毁方法 构造&#xff08;对象创建&#xff09; 单…...

【C++】——类和对象

目录 面向过程和面向对象的初步认识类的引入类的定义类的访问限定符及封装类的作用域类的实例化this指针类的6个默认成员函数构造函数析构函数 面向过程和面向对象的初步认识 C语言是面向过程的&#xff0c;关注的是过程&#xff0c;分析求解问题的步骤&#xff0c;通过函数调用…...

【Docker】使用docker-maven-plugin插件构建发布推镜像到私有仓库

文章目录 1. 用docker-maven-plugin插件推送项目到私服docker1.1. 构建镜像 v1.01.2. 构建镜像 v2.01.3. 推送到镜像仓库 2. 拉取私服docker镜像运行3. 参考资料 本文描述了在Spring Boot项目中通过docker-maven-plugin插件把项目推送到私有docker仓库中&#xff0c;随后拉取仓…...

区块链学习笔记

区块链技术与应用 数组 列表 二叉树 哈希函数 BTC中的密码学原理 cryptographic hash function collsion resistance(碰撞抵抗) 碰撞指的是找到两个不同的输入值&#xff0c;使得它们的哈希值相同。也就是说&#xff0c;如果存在任意两个输入x和y&#xff0c;满足x ≠ y…...

实用上位机--QT

实用上位机–QT 通信协议如下 上位机设计界面 #------------------------------------------------- # # Project created by QtCreator 2023-07-29T21:22:32 # #-------------------------------------------------QT += core gui serialportgreaterThan(QT_MAJOR_V…...

os.signal golang中的信号处理

在程序进行重启等操作时&#xff0c;我们需要让程序完成一些重要的任务之后&#xff0c;优雅地退出&#xff0c;Golang为我们提供了signal包&#xff0c;实现信号处理机制&#xff0c;允许Go 程序与传入的信号进行交互。 Go语言标准库中signal包的核心功能主要包含以下几个方面…...

Python源码:Tkinter组件布局管理的3种方式

Tkinter组件布局管理可以使用pack()方法、grid()方法和place()方法。pack()方法将组件放置在窗口中&#xff0c;grid()方法将组件放置在网格布局中&#xff0c;place()方法将组件放置在指定位置。 01使用pack()方法布局&#xff1a; 在Tkinter中&#xff0c;pack方法用于将控…...

网络防御之VPN

配置IKE 第一阶段 [r1]ike proposal 1 [r1-ike-proposal-1]encryption-algorithm aes-cbc-128 [r1-ike-proposal-1]authentication-algorithm sha1 [r1-ike-proposal-1]dh group2 [r1-ike-proposal-1]authentication-method pre-share[r1]ike peer aaa v1 [r1-ike-peer-aaa…...

VUE使用docxtemplater导出word(带图片) 踩坑 表格循环空格 ,canvas.toDataURL图片失真模糊问题

参考&#xff1a;https://www.codetd.com/article/15219743 安装 // 安装 docxtemplater npm install docxtemplater pizzip --save // 安装 jszip-utils npm install jszip-utils --save // 安装 jszip npm install jszip --save // 安装 FileSaver npm install file-save…...

ubuntu 安装 Pycharm社区版

在Ubuntu中安装pycharm社区版_上玄下纁的博客-CSDN博客 里面可以创建快捷方式&#xff0c;蛮好用的...

IP 监控软件

IP 监控软件可帮助管理员主动监控网络资源。随着各种设备连接到网络&#xff0c;监控设备和接口可能很复杂&#xff0c;为管理员提供这些设备的IP监控&#xff0c;了解其各种性能指标和问题。 使用有效的 IP 监控软件的优势 使用有效的 IP 监控系统和一套全面的 IP 监控工具&…...

C#实现读写CSV文件的方法详解

目录 CSV文件标准 文件示例RFC 4180简化标准读写CSV文件 使用CsvHelper使用自定义方法总结 项目中经常遇到CSV文件的读写需求&#xff0c;其中的难点主要是CSV文件的解析。本文会介绍CsvHelper、TextFieldParser、正则表达式三种解析CSV文件的方法&#xff0c;顺带也会介绍一…...

04 http连接处理(上)

基础知识&#xff1a;epoll、http报文格式、状态码和有限状态机 代码&#xff1a;对服务端处理http请求的全部流程进行简要介绍&#xff0c;然后结合代码对http类及请求接收进行详细分析。 epoll epoll_create函数 #include <sys/epoll.h> int epoll_create(int size)…...

c++(强生成关键字+可变参数模板+emplace)[26]

强制生成 不生成 在C中&#xff0c;可以通过一些方式来控制编译器是否生成某些特殊成员函数&#xff08;如默认构造函数、拷贝构造函数、拷贝赋值运算符、析构函数等&#xff09;。 默认生成&#xff1a;如果你没有显式地定义这些特殊成员函数&#xff0c;编译器会自动生成它们…...

Mysql 数据库开发及企业级应用

文章目录 1、Mysql 数据库开发及企业级应用1.1、为什么要使用数据库1.1.1、数据库概念&#xff08;Database&#xff09;1.1.2、为什么需要数据库 1.2、程序员为什么要学习数据库1.3、数据库的选择1.3.1、主流数据库简介1.3.2、使用 MySQL 的优势1.3.3、版本选择 1.4、Windows …...

【数据结构】_6.队列

目录 1.概念 2.队列的使用 3.队列模拟实现 4.循环队列 5.双端队列 6.OJ题 6.1 用队列实现栈 6.2 用栈实现队列 1.概念 &#xff08;1&#xff09;队列是只允许在一端进行插入数据操作&#xff0c;在另一端进行删除数据操作的特殊线性表&#xff1b; &#xff08;2&am…...

7 网络通信(上)

文章目录 网络通信概述ip地址ip的作用ip地址的分类私有ip 掩码和广播地址 linux 命令&#xff08;ping ifconfig&#xff09;查看或配置网卡信息&#xff1a;ifconfig(widows 用ipconfig)测试远程主机连通性&#xff1a;ping路由查看 端口端口是怎样分配的知名端口动态端口 查看…...

MFC图表控件high-speed-charting的使用

high-speed-charting是MFC上的开源图表库,Teechart的替代品。 high-speed-charting的下载地址 https://www.codeproject.com/Articles/14075/High-speed-Charting-Control 特性 High-speed drawing (when axis is fixed) which allows fast plotting of dataUnlimited number …...

Unity中常用方法

1.基础 //初始化引入 [RequireComponent(typeof(BoxCollider2D))] [RequireComponent(typeof(Rigidbody2D))]//游戏帧率设置 60帧Application.targetFrameRate 60;//获取物体对象 //获取到当前物体(根据名称&#xff0c;也可以根据路径)GameObject go GameObject.Find("…...

JS设计模式(4):观察者模式

JS设计模式(4):观察者模式 一、引入 在开发中&#xff0c;我们经常会遇到这样的场景&#xff1a;一个对象的状态变化需要自动通知其他对象&#xff0c;比如&#xff1a; 电商平台中&#xff0c;商品库存变化时需要通知所有订阅该商品的用户&#xff1b;新闻网站中&#xff0…...

Webpack性能优化:构建速度与体积优化策略

一、构建速度优化 1、​​升级Webpack和Node.js​​ ​​优化效果​​&#xff1a;Webpack 4比Webpack 3构建时间降低60%-98%。​​原因​​&#xff1a; V8引擎优化&#xff08;for of替代forEach、Map/Set替代Object&#xff09;。默认使用更快的md4哈希算法。AST直接从Loa…...

wpf在image控件上快速显示内存图像

wpf在image控件上快速显示内存图像https://www.cnblogs.com/haodafeng/p/10431387.html 如果你在寻找能够快速在image控件刷新大图像&#xff08;比如分辨率3000*3000的图像&#xff09;的办法&#xff0c;尤其是想把内存中的裸数据&#xff08;只有图像的数据&#xff0c;不包…...

水泥厂自动化升级利器:Devicenet转Modbus rtu协议转换网关

在水泥厂的生产流程中&#xff0c;工业自动化网关起着至关重要的作用&#xff0c;尤其是JH-DVN-RTU疆鸿智能Devicenet转Modbus rtu协议转换网关&#xff0c;为水泥厂实现高效生产与精准控制提供了有力支持。 水泥厂设备众多&#xff0c;其中不少设备采用Devicenet协议。Devicen…...

【UE5 C++】通过文件对话框获取选择文件的路径

目录 效果 步骤 源码 效果 步骤 1. 在“xxx.Build.cs”中添加需要使用的模块 &#xff0c;这里主要使用“DesktopPlatform”模块 2. 添加后闭UE编辑器&#xff0c;右键点击 .uproject 文件&#xff0c;选择 "Generate Visual Studio project files"&#xff0c;重…...

高效的后台管理系统——可进行二次开发

随着互联网技术的迅猛发展&#xff0c;企业的数字化管理变得愈加重要。后台管理系统作为数据存储与业务管理的核心&#xff0c;成为了现代企业不可或缺的一部分。今天我们要介绍的是一款名为 若依后台管理框架 的系统&#xff0c;它不仅支持跨平台应用&#xff0c;还能提供丰富…...

EasyRTC音视频实时通话功能在WebRTC与智能硬件整合中的应用与优势

一、WebRTC与智能硬件整合趋势​ 随着物联网和实时通信需求的爆发式增长&#xff0c;WebRTC作为开源实时通信技术&#xff0c;为浏览器与移动应用提供免插件的音视频通信能力&#xff0c;在智能硬件领域的融合应用已成必然趋势。智能硬件不再局限于单一功能&#xff0c;对实时…...

更新 Docker 容器中的某一个文件

&#x1f504; 如何更新 Docker 容器中的某一个文件 以下是几种在 Docker 中更新单个文件的常用方法&#xff0c;适用于不同场景。 ✅ 方法一&#xff1a;使用 docker cp 拷贝文件到容器中&#xff08;最简单&#xff09; &#x1f9f0; 命令格式&#xff1a; docker cp <…...

DriveGPT4: Interpretable End-to-end Autonomous Driving via Large Language Model

一、研究背景与创新点 (一)现有方法的局限性 当前智驾系统面临两大核心挑战:一是长尾问题,即系统在遇到新场景时可能失效,例如突发交通状况或非常规道路环境;二是可解释性问题,传统方法无法解释智驾系统的决策过程,用户难以理解车辆行为的依据。传统语言模型(如 BERT…...

OCC笔记:TDF_Label中有多个相同类型属性

注&#xff1a;OCCT版本&#xff1a;7.9.1 TDF_Label中有多个相同类型的属性的方案 OCAF imposes the restriction that only one attribute type may be allocated to one label. It is necessary to take into account the design of the application data tree. For exampl…...