IO进程线程(十)进程间通信 消息队列 共享内存 信号灯集

文章目录

  • 一、IPC(Inter-Process Communication)进程间通信相关命令 :
    • (一)ipcs --- 查看IPC对象
    • (二)获取IPC键值
    • (三)删除IPC对象的命令
    • (四)获取IPC键值的函数
      • 1. 函数定义
      • 2. 使用示例
  • 二、消息队列
    • (一) 特点
    • (二) 相关API
      • 1. 创建或获取一个消息队列
      • 2. 向消息队列中写消息
      • 3. 在消息队列中读取一条消息
      • 4. 控制消息队列
    • (三) 不关注消息类型
    • (四)关注消息类型
    • (五)消息队列属性结构体
  • 三、共享内存 shared memory(shm)
    • (一)特点
    • (二) 相关API
      • 1. 创建共享内存
      • 2. 映射共享内存到当前的进程空间
      • 3. 取消地址映射
      • 4. 共享内存控制
    • (三)使用示例
    • (四) 属性
  • 四、信号灯集---控制进程间同步
    • (一)特点
    • (二) 相关API
    • (三) 原生函数使用示例
    • (四)封装函数

一、IPC(Inter-Process Communication)进程间通信相关命令 :

(一)ipcs — 查看IPC对象

在这里插入图片描述

(二)获取IPC键值

ipcs -q 查看消息队列
在这里插入图片描述

ipcs -m 查看共享内存
在这里插入图片描述

ipcs -s 查看信号灯集
在这里插入图片描述

(三)删除IPC对象的命令

ipcrm -q id 删除消息队列
ipcrm -m id 删除共享内存
ipcrm -s id 删除信号灯集

(四)获取IPC键值的函数

1. 函数定义

#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok(const char *pathname, int proj_id);
功能:使用给定的文件和字符来生成一个IPC通信的key值
参数:pathname:路径和文件名(必须是已存在的可访问的)proj_id:[0-255]的值  一般我们传一个字符即可 如 'A'  'm'
返回值:成功  key值失败  -1  重置错误码
  • 注:
  • typedef int key_t key_t 即 int 类型
  • pathname 要求必须是一个已存在的文件,因为key值的是由proj_id的后八位,设备号的后8位以及inode号的后16位组成。因此这种机制并不保证key值一定不重复
  • proj_id 只是用了int的一个字节,即取值范围是[0-255],一般使用时传一个字符,字符ASCII码是0-127。

2. 使用示例

生成并打印出key,分析key值的由来

验证代码

#include <my_head.h>int main(int argc, char const *argv[])
{key_t key=0;key = ftok("/home/linux/05work",'A');printf("my_key = %#x\n",key);struct stat file_stat= {0};stat("/home/linux/05work",&file_stat);printf("proj_id=%#x  dev=%#lx inode=%#lx\n",'A', file_stat.st_dev, file_stat.st_ino);return 0;
}

输出结果
在这里插入图片描述

二、消息队列

(一) 特点

  1. 消息队列也是基于内核实现的,存放在内存上(而非硬盘上)。
  2. 消息队列的大小,默认是 16K。
  3. 消息队列中的消息有类型和正文。
    A进程将消息写入消息队列;
    B进程可以根据消息的类型从消息队列中将对应类型的消息取走。
  4. 半双工通信

(二) 相关API

1. 创建或获取一个消息队列

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>int msgget(key_t key, int msgflg);
功能:创建或者获取一个消息队列
参数:key:键值key 通过ftok获取的IPC_PRIVATE 表示只有亲缘进程间能用msgflg:消息队列的标志位IPC_CREAT|0666 消息队列不存在则创建,权限0666 或者  IPC_CREAT|IPC_EXCL|0666 消息不存在则创建,存在则但会-1,置错误码为EEXIST
返回值:成功 消息队列的id失败 -1 重置错误码

2. 向消息队列中写消息

int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
功能:向消息队列中写入一条消息
参数:msqid:消息队列的idmsgp: 要写入的数据的首地址struct msgbuf {long mtype;      /* 消息的类型 必须大于 0 */char mtext[1];   /* 消息正文 可以自定义 */};msgsz:消息正文的大小msgflg:标志位 0 阻塞发送  IPC_NOWAIT 非阻塞发送
返回值:成功 0失败 -1  重置错误码
  • 注:
  • 关于void *msgp参数,第一个long类型大小的空间必须用来存放消息的类型,消息的正文可以自定义
  • size_t msgsz参数,只包含正文数据的大小(sizeof(struct msgbuf)-sizeof(mtype))
  • 阻塞发送的情况下,如果消息队列满了,A进程还想向消息队列中写入消息,此时A进程将会阻塞。

3. 在消息队列中读取一条消息

ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
功能:在消息队列中读取一条消息
参数:msqid:消息队列的idmsgp: 用来保存接收的数据的缓冲区的首地址struct msgbuf {long mtype;     /* 消息的类型 必须大于 0 */char mtext[1];  /* 消息正文 可以自定义 */};msgsz:消息正文的大小msgtyp:要接受的消息的类型0 :接收消息队列中第一条消息>0 : 接收指定类型的第一条消息<0 :一般不使用,了解即可,表示接收消息队列中第一条类型最小的小于msgtyp的绝对值的消息3-2-5-500-200-8读取时,类型传 -200读取的顺序  2-3-5 msgflg:标志位 0 阻塞接收  IPC_NOWAIT 非阻塞接收
返回值:成功 实际读到的正文的字节数失败 -1  重置错误码
  • 注:读消息队列和写消息队列中的void *msgp结构体的内部成员要尽量对应。

4. 控制消息队列

int msgctl(int msqid, int cmd, struct msqid_ds *buf);
功能:控制消息队列
参数:msqid:消息队列idcmd:指令IPC_STAT:获取消息队列的属性IPC_SET:设置消息队列的属性IPC_RMID:立即删除消息队列只有消息队列的创建者和所有者以及root用户可以删除消息队列msgctl函数的第三个参数被忽略buff:属性结构体的地址
返回值:成功 0失败 -1  重置错误码

(三) 不关注消息类型

此时进程间通信不关心消息类型,按顺序接收第一条消息。
注意,当其中一个进程关闭消息队列后,另一个进程再试图关闭,就会报错,错误码EINVAL

read.c

#include <my_head.h>typedef struct msgbuf{long type;char name[20];
}msg_t;int main(int argc, char const *argv[])
{//获取key值key_t key=ftok("/home/linux/05work",'A');//创建消息队列int msgid = msgget(key,IPC_CREAT|0666);if(-1 == msgid)ERR_LOG("msgget error");//定义消息结构体msg_t msg={0};int type=0;while(1){if(-1 == msgrcv(msgid,&msg,sizeof(msg_t)-sizeof(msg.type),type,0)){ERR_LOG("msgrcv error");}if(!strcmp(msg.name,"quit")){break;}printf("%ld:%s\n",msg.type,msg.name);}if(-1 == msgctl(msgid,IPC_RMID,NULL)){if(EINVAL == errno){return 0;}ERR_LOG("msgctl error");}return 0;
}

write.c

#include <my_head.h>typedef struct msgbuf{long type;char name[20];
}msg_t;int main(int argc, char const *argv[])
{//获取key值key_t key=ftok("/home/linux/05work",'A');//创建消息队列int msgid = msgget(key,IPC_CREAT|0666);if(-1 == msgid)ERR_LOG("msgget error");//定义消息结构体msg_t msg={0};while(1){printf("请输入消息:<类型> <正文>:");scanf("%ld %s",&msg.type,msg.name);msgsnd(msgid,&msg,sizeof(msg_t)-sizeof(msg.type),0);if(!strcmp(msg.name,"quit")){break;}}//销毁消息队列if(-1 == msgctl(msgid,IPC_RMID,NULL)){if(EINVAL == errno){return 0;}ERR_LOG("msgctl error");}return 0;
}

(四)关注消息类型

此时进程间通信关心消息类型,按顺序接收第一条符合类型的消息,如果消息队列中没有同类型消息,会阻塞等待。
在这里插入图片描述
read.c

#include <my_head.h>typedef struct msgbuf{long type;char name[20];
}msg_t;int main(int argc, char const *argv[])
{//获取key值key_t key=ftok("/home/linux/05work",'A');//创建消息队列int msgid = msgget(key,IPC_CREAT|0666);if(-1 == msgid)ERR_LOG("msgget error");//定义消息结构体msg_t msg={0};int type=0;while(1){printf("请输入想要接收的消息类型:");scanf("%d",&type);if(-1 == msgrcv(msgid,&msg,sizeof(msg_t)-sizeof(msg.type),type,0)){ERR_LOG("msgrcv error");}if(!strcmp(msg.name,"quit")){break;}printf("%ld:%s\n",msg.type,msg.name);}if(-1 == msgctl(msgid,IPC_RMID,NULL)){if(EINVAL == errno){return 0;}ERR_LOG("msgctl error");}return 0;
}

write.c

#include <my_head.h>typedef struct msgbuf{long type;char name[20];
}msg_t;int main(int argc, char const *argv[])
{//获取key值key_t key=ftok("/home/linux/05work",'A');//创建消息队列int msgid = msgget(key,IPC_CREAT|0666);if(-1 == msgid)ERR_LOG("msgget error");//定义消息结构体msg_t msg={0};while(1){printf("请输入消息:<类型> <正文>:");scanf("%ld %s",&msg.type,msg.name);msgsnd(msgid,&msg,sizeof(msg_t)-sizeof(msg.type),0);if(!strcmp(msg.name,"quit")){break;}}//销毁消息队列if(-1 == msgctl(msgid,IPC_RMID,NULL)){if(EINVAL == errno){return 0;}ERR_LOG("msgctl error");}return 0;
}

(五)消息队列属性结构体

struct msqid_ds {struct ipc_perm msg_perm;     /* IPC权限结构体 */time_t          msg_stime;    /* 最后一次执行msgsnd的时间 */time_t          msg_rtime;    /* 最后一次执行msgrcv的时间 */time_t          msg_ctime;    /* 最后一次被修改的时间 */unsigned long   __msg_cbytes; /* 当前消息队列中的字节数 */msgqnum_t       msg_qnum;     /* 当前消息队列中的消息数 */msglen_t        msg_qbytes;   /* 允许的最大字节数 */pid_t           msg_lspid;    /* 最后一次执行msgsnd的进程的PID */pid_t           msg_lrpid;    /* 最后一次执行msgrcv的进程的PID */
};struct ipc_perm {key_t          __key;       /* 键值 */uid_t          uid;         /* 所属用户的id */gid_t          gid;         /* 所属用户的组id */uid_t          cuid;        /* 创建者的id */gid_t          cgid;        /* 创建者的组id */unsigned short mode;        /* 权限 */
};
  • 注:
    qbytes可以改小,改大的话需要sudo权限

三、共享内存 shared memory(shm)

(一)特点

在内核中创建共享内存,让进程A和进程B都能够访问到,通过这段内存进行数据的传递。
共享内存是所有进程间通信方式中效率最高的(不需要来回进行数据的拷贝)
在这里插入图片描述

(二) 相关API

1. 创建共享内存

#include <sys/ipc.h>
#include <sys/shm.h>int shmget(key_t key, size_t size, int shmflg);
功能:创建共享内存
参数:key:键值key 通过ftok获取IPC_PRIVATE:只能用于亲缘进程间的通信size:共享内存的大小  PAGE_SIZE(4k)的整数倍shmflg:共享的标志位IPC_CREAT|0666 或 IPC_CREAT|IPC_EXCL|0666
返回值:成功 共享内存编号失败 -1 重置错误码
  • 注:
  • 共享内存大小必须要4k的整数倍,因为一页是4k。如果申请时不要求4
    的整数倍,分配时也是分配4k的整数倍。
  • 内核空间越界会直接段错误
  • 同一个key值可以同时用于消息队列,共享内存,信号灯集

2. 映射共享内存到当前的进程空间

#include <sys/ipc.h>
#include <sys/shm.h>
void *shmat(int shmid, const void *shmaddr, int shmflg);
功能:映射共享内存到当前的进程空间
参数:shmid:共享内存编号shmaddr:指定共享内存出现在进程内存地址的什么位置,直接指定为NULL,让系统自动分配shmflg:共享内存操作方式0    读写SHM_RDONLY    只读
返回值:成功 指向共享内存的地址失败 (void *)-1 重置错误码

3. 取消地址映射

#include <sys/ipc.h>
#include <sys/shm.h>
int shmdt(const void *shmaddr);
功能:取消地址映射
参数:shmaddr:指向共享内存的指针
返回值:成功 0失败 -1 重置错误码

4. 共享内存控制

int shmctl(int shmid, int cmd, struct shmid_ds *buf);
功能:共享内存控制的函数
参数:shmid:共享内存编号cmd:操作的命令码IPC_STAT:获取IPC_SET:设置IPC_RMID:删除共享内存标记要销毁的段。实际上,只有在最后一个进程将其分离之后 (也就是说,关联结构shmid_ds的shm_nattch成员为零时), 段才会被销毁。调用者必须是段的所有者或创建者,或具有特权。buf参数被忽略。buf:共享内存属性结构体指针
返回值:成功 0失败 -1 重置错误码

(三)使用示例

read.c

#include <my_head.h>#define SHM_PAGE 1024*4
int main(int argc, char const *argv[])
{//获取键值key_t key = ftok("/home/linux/05work",'A');if(-1 == key)ERR_LOG("ftok error");//创建共享内存int shmid = shmget(key, SHM_PAGE,IPC_CREAT|0666);if(-1 == shmid)ERR_LOG("shmget error");//映射共享内存char *addr = (char *)shmat(shmid,NULL,SHM_RDONLY);while(1){getchar();printf("%s",addr);}return 0;
}

write.c

#include <my_head.h>#define SHM_PAGE 1024*4
int main(int argc, char const *argv[])
{//获取键值key_t key = ftok("/home/linux/05work",'A');if(-1 == key)ERR_LOG("ftok error");//创建共享内存int shmid = shmget(key, SHM_PAGE,IPC_CREAT|0666);if(-1 == shmid)ERR_LOG("shmget error");//映射共享内存char *addr = (char *)shmat(shmid,NULL,0);while(1){printf("请输入要发送的内容:");scanf("%s",addr);}return 0;
}

(四) 属性

struct shmid_ds{struct ipc_perm shm_perm;    //权限结构体size_t shm_segsz;             //共享内存大小,单位是字节 __time_t shm_atime;         //最后一次映射的时间 __pid_t shm_cpid;             //创建共享内存进程的pid __pid_t shm_lpid;             //最后一次操作共享内存进程的pid shmatt_t shm_nattch;         //共享内存映射的次数
};
struct ipc_perm{__key_t __key;                //ftok获取的key__uid_t uid;                 //用户的ID__gid_t gid;                 //组ID__uid_t cuid;                //创建共享内存的用户的ID__gid_t cgid;                //创建共享内存的组的IDunsigned short int mode;     //消息队列的权限
};

四、信号灯集—控制进程间同步

(一)特点

进程间同步的机制

(二) 相关API

#include <sys/sem.h>
int semget(key_t key, int nsems, int semflg);
功能:创建一个信号灯集
参数:key:键值IPC_PRIVATE  keynsems:信号灯集合中信号灯的个数semflag:创建的标志位IPC_CREAT|0666 或 IPC_CREAT|IPC_EXCL|0666
返回值:成功 semid失败 -1  重置错误码int semctl(int semid, int semnum, int cmd, ...);
功能:信号灯集的控制函数
参数:semid信号灯集的IDsenum:信号灯的编号 从0开始cmd:命令码SETVAL:设置信号灯的值 --->第四个参数val选项GETVAL:获取信号灯的值 --->不需要第四个参数IPC_STAT:获取信号灯集的属性--->第二参数被忽略,第四个参数buf选项IPC_SET :设置信号灯集的属性--->第二参数被忽略,第四个参数buf选项IPC_RMID:删除信号灯集 第二参数被忽略,第4个参数不用填写 @...:可变参union semun{int val; /* Value for SETVAL */struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */};
返回值:成功:GETVAL:成功返回信号灯的值其余的命令码成功返回0失败 -1  重置错误码int semop(int semid, struct sembuf *sops, size_t nsops);
功能:信号灯集中信号灯的操作函数
参数:semid:信号灯集的编号sops:操作方式struct sembuf{unsigned short sem_num; //信号灯的编号short sem_op; //操作方式(PV)-1:P操作,申请资源 1:V操作,释放资源short sem_flg; //操作的标志位 0:阻塞 IPC_NOWAIT:非阻塞方式操作}nsops:本次操作信号灯的个数
返回值:成功 0失败 -1  重置错误码
  • 注:
  • 初始化操作必须只能一次,如果多次可能会将其他进程的值初始化
  • 同时对多个信号灯进行操作时,可以定义一个结构体数组

(三) 原生函数使用示例

(四)封装函数

原生函数直接使用会比较繁琐,因此会做二次封装。

初始化(解决可能重复初始化的问题):


P操作


V操作


销毁


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

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

相关文章

django ORM model update常规用法

Django ORM&#xff08;对象关系映射&#xff09;提供了一种强大而直观的方式&#xff0c;通过Python类和方法与数据库交互。在Django模型中更新记录是一个常见的任务&#xff0c;可以通过多种方式完成。以下是一些常见的更新记录的方法&#xff1a; 1. 更新单条记录 使用 sa…

Linux 内核之 mmap 内存映射触发的缺页异常 Page Fault

文章目录 前言一、简介1. MMU 内存管理2. 缺页中断3. 页表4. 小节 二、mmap 提前分配物理内存1. mm_populate 函数2. __mm_populate 函数3. populate_vma_page_range 函数4. __get_user_pages 函数5. find_extend_vma 函数6. find_vma 函数7. follow_page_mask 函数8. follow_p…

从零开始:腾讯云轻量应用服务器上部署MaxKB项目(基于LLM大语言模型的知识库问答系统)

使用腾讯云轻量应用服务器部署和使用MaxKB项目 前言 一&#xff0c; MaxKB介绍 MaxKB是基于LLM大语言模型的知识库问答系统&#xff0c;旨在成为企业的最强大脑。它支持开箱即用&#xff0c;无缝嵌入到第三方业务系统&#xff0c;并提供多模型支持&#xff0c;包括主流大模型…

数据结构之ArrayList与顺序表(上)

找往期文章包括但不限于本期文章中不懂的知识点&#xff1a; 个人主页&#xff1a;我要学编程(ಥ_ಥ)-CSDN博客 所属专栏&#xff1a;数据结构&#xff08;Java版&#xff09; 顺序表的学习&#xff0c;点我 上面这篇博文是关于顺序表的基础知识&#xff0c;以及顺序表的实现。…

Java和Web前端哪个有发展前景?

Java和Web前端都是当今技术行业里的热门岗位&#xff0c;岗位招聘需求量大&#xff0c;人才竞争度高&#xff0c;同学们掌握这两个岗位里其中任何一个的相关主流技术&#xff0c;都可以找到一份不错的职位。下面请允许笔者做一个简要的分析阐述&#xff1a; 一、Web前端 Web前…

生成ssh密钥,使用ssh连接linux系统

这里写目录标题 ssh密钥大概介绍1、密钥在哪里生成&#xff08;客户端/服务器&#xff09;&#xff1f;2、密钥生成是什么样子的&#xff1f; ssh &#xff08;生成密钥、密钥传输、配置连接、连接服务&#xff09;过程1、生成密钥提示一&#xff1a;输入保存密钥的文件&#x…

JVM 虚拟机

JVM 是 Java Virtual Machine 的简称&#xff0c;意为 Java 虚拟机&#xff0c;虚拟机是指通过软件模拟的具有完整硬件功能的、运行在一个完全隔离的环境中的完整计算机系统。 常见的虚拟机有&#xff1a;JVM、VMwave、Virtual Box等。JVM 是一台被定制过的现实当中不存在的计算…

在keil5中打开keil4工程的方法

文章目录 1. 打开文件 2. 安装旧版本包 3. 在keil4中打开keil5工程 1. 打开文件 在keil5 MDK的环境下&#xff0c;打开keil4的工程文件&#xff0c;会弹出下图所示的窗口&#xff1a; 参考官网的解释这两个方法分别为&#xff1a; 1. 使用MDK 版本 4 Legacy Pack时&#x…

在推荐四款软件卸载工具,让流氓软件无处遁形

Revo Uninstaller Revo Uninstaller是一款电脑软件、浏览器插件卸载软件&#xff0c;目前已经有了17年的历史了。可以扫描所有window用户卸载软件后的残留物&#xff0c;并及时清理&#xff0c;避免占用电脑空间。 Revo Uninstaller可以通过命令行卸载软件&#xff0c;可以快速…

【C++ | 拷贝构造函数】一文了解C++的 拷贝(复制)构造函数

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; ⏰发布时间⏰&#xff1a;2024-06-07 2…

链表题目练习----重排链表

这道题会联系到前面写的一篇文章----快慢指针相关经典问题。 重排链表 指针法 这道题乍一看&#xff0c;好像有点难处理&#xff0c;但如果仔细观察就会发现&#xff0c;这道题是查找中间节点反转链表链表的合并问题&#xff0c;具体细节有些不同&#xff0c;这个在反装中间链…

如何有效提问?

有效提问&#xff1a;正确地向别人提问是一种艺术&#xff0c;可以帮助你获得清晰、有用的答案。 明确表达问题&#xff1a;确保你的问题清晰明了&#xff0c;避免含糊不清或模棱两可的语言。这可以帮助对方更好地理解你的问题&#xff0c;并给出准确的答复。 尊重对方&#x…

毕业论文word常见问题

0、前言&#xff1a; 这里的问题都是以office办公软件当中的word为例&#xff0c;和WPS没有关系。 1、页眉横线删不掉&#xff1a; 解决方案&#xff1a;进入页眉编辑状态&#xff0c;在开始选项栏中选择页眉字体样式&#xff0c;清除格式。 修改方式如下&#xff1a; 2、…

社区服务支持

社区服务支持 原创 小王搬运工 时序课堂 2024-06-07 19:29 四川 &#x1f31f; 邀请函 | 加入我们的时序数据挖掘社区 &#x1f680; 尊敬的数据爱好者们&#xff0c; 我们诚挚地邀请您加入我们的专业社区——时序数据挖掘社区&#xff0c;一个专注于时序数据分析、挖掘与应…

QT 信号和槽 多对一关联示例,多个信号,一个槽函数响应,多个信号源如何绑定一个槽函数

三个顾客 Anderson、Bruce、Castiel 都要订饭&#xff0c;分别对应三个按钮&#xff0c;点击一个按钮&#xff0c;就会弹出给该顾客送饭的消息。注意这个例子只使用一个槽函数&#xff0c;而三个顾客名称是不一样的&#xff0c;弹窗时显示的消息不一样&#xff0c;这需要一些 技…

什么情况下要配置DNS服务

什么是DNS 一、DNS就是域名解析 我们上网的方式通常都由ip地址组成&#xff0c;但是为了有个规范&#xff0c;而且我们也不可能去记住那么多一串Ip数字&#xff0c;首先域名就会比ip好记很多&#xff0c;其次固定性&#xff0c;一旦服务器换了&#xff0c;只要重新绑定域名对…

【Flutter】 TextField限制长度时, 第三方手写输入法、ios原始拼音输入法输入被吞问题

问题描述 TextField限制长度时&#xff0c; 当你的输入字符长度已经到了最大值-1时&#xff0c;使用第三方手写输入法或者ios原生拼音输入法输入liang&#xff08;什么拼音都行&#xff0c;这里只是举例&#xff09;&#xff0c;输到i那么li都会消失。 原因分析 这是因为第三…

C++青少年简明教程:C++函数

C青少年简明教程&#xff1a;C函数 C函数是一段可重复使用的代码&#xff0c;用于执行特定的任务&#xff0c;可以提高代码的可读性和可维护性。函数可以接受参数&#xff08;输入&#xff09;并返回一个值&#xff08;输出&#xff09;&#xff0c;也可以没有参数和返回值。 …

Nvidia/算能 +FPGA+AI大算力边缘计算盒子:中国舰船研究院

中国舰船研究院又称中国船舶重工集团公司第七研究院&#xff0c;隶属于中国船舶重工集团公司&#xff0c;是专门从事舰船研究、设计、开发的科学技术研究机构&#xff0c;是中国船舶重工集团公司的军品技术研究中心、科技开发中心&#xff1b;主要从事舰船武器装备发展战略研究…

【spring】第一篇 IOC和DI入门案例

Spring到底是如何来实现IOC和DI的&#xff0c;那接下来就通过一些简单的入门案例&#xff0c;来演示下具体实现过程。 目录 前期准备 一、IOC入门案例 思路分析 代码实现 二、DI入门案例 思路分析 代码实现 总结 前期准备 使用IDEA创建Maven项目&#xff0c;首先需要配…