linux ftok() shmget函数

本文详细介绍了Linux下使用IPC(Inter Process Communication)进行进程间通信的方法,特别是如何利用ftok函数生成用于消息队列和共享内存的唯一标识符,并通过shmget和shmctl函数创建、操作共享内存。通过示例展示了多进程间共享内存的过程及其应用。


      系统建立IPC通讯(如消息队列、共享内存时)必须指定一个ID值。通常情况下,该id值通过ftok函数得到。
ftok原型如下:
key_t ftok( char * fname, int id )

fname就时你指定的文件名(该文件必须是存在而且可以访问的),id是子序号, 虽然为int,但是只有8个比特被使用(0-255)。

当成功执行的时候,一个key_t值将会被返回,否则 -1 被返回。

   在一般的UNIX实现中,是将文件的索引节点号取出,前面加上子序号得到key_t的返回值。如指定文件的索引节点号为65538,换算成16进制为 0x010002,而你指定的ID值为38,换算成16进制为0x26,则最后的key_t返回值为0x26010002。
查询文件索引节点号的方法是: ls -i

以下为测试程序:
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>

#define IPCKEY 0x11
int main( void )
{
    int i=0;
    for ( i = 1; i < 256; ++ i )
        printf( "key = %x\n", ftok( "/tmp", i ) );

    return 0;
}

在成功获取到key之后,就可以使用该key作为某种方法的进程间通信的key值,例如shmget共享内存的方式。

shmget的函数原型为

int shmget( key_t, size_t, flag);

在创建成功后,就返回共享内存的描述符。在shmget中使用到的key_t就是通过ftok的方式生成的

实例:

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

#define SIZE 1024

extern int errno;

int main()
{
int shmid;
char *shmptr;

//创建共享内存
if((shmid = shmget(IPC_PRIVATE, SIZE, 0600)) < 0)
   {
    printf("shmget error:%s\n", strerror(errno));
    return -1;
   }

//将共享内存连接到 可用地址上

if((shmptr = (char*)shmat(shmid, 0, 0)) == (void*)-1)
{
    printf("shmat error:%s\n", strerror(errno));
    return -1;
}
memcpy(shmptr, "hello world", sizeof("hello world"));
printf("share memory from %lx to %lx, content:%s\n",(unsigned long)shmptr, (unsigned long)(shmptr + SIZE), shmptr);

//拆卸共享内存
if((shmctl(shmid, IPC_RMID, 0) < 0))
{
    printf("shmctl error:%s\n", strerror(errno));
    return -1;
}
}

多进程之间共享内存情况:

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

#define SIZE 1024

extern int errno;

int main()
{
int shmid;
char *shmptr;
key_t key;
pid_t pid;

if((pid = fork()) < 0)
{
    printf("fork error:%s\n", strerror(errno));
    return -1;
   }
else if(pid == 0)
   {
     sleep(2);
    if((key = ftok("/dev/null", 1)) < 0)
    {
      printf("ftok error:%s\n", strerror(errno));
      return -1;
    }
if((shmid = shmget(key, SIZE, 0600)) < 0)
{
    printf("shmget error:%s\n", strerror(errno));
    exit(-1);
   }

if((shmptr = (char*)shmat(shmid, 0, 0)) == (void*)-1)
{
    printf("shmat error:%s\n", strerror(errno));
    exit(-1);
}
//memcpy(shmptr, "hello world", sizeof("hello world")); 
printf("child:pid is %d,share memory from %lx to %lx, content:%s\n",getpid(), (unsigned long)shmptr, (unsigned long)(shmptr + SIZE
), shmptr);
printf("child process sleep 2 seconds\n");
sleep(2);
if((shmctl(shmid, IPC_RMID, 0) < 0))
{
    printf("shmctl error:%s\n", strerror(errno));
    exit(-1);
}
   exit(0);
   }
   //parent
   else
   {
    if((key = ftok("/dev/null", 1)) < 0)
    {
      printf("ftok error:%s\n", strerror(errno));
      return -1;
    }
if((shmid = shmget(key, SIZE, 0600|IPC_CREAT|IPC_EXCL)) < 0)

{
    printf("shmget error:%s\n", strerror(errno));
    exit(-1);
   }

if((shmptr = (char*)shmat(shmid, 0, 0)) == (void*)-1)
{
    printf("shmat error:%s\n", strerror(errno));
    exit(-1);
}
memcpy(shmptr, "hello world", sizeof("hello world"));
printf("parent:pid is %d,share memory from %lx to %lx, content:%s\n",getpid(),(unsigned long)shmptr, (unsigned long)(shmptr + SIZE
), shmptr);
printf("parent process sleep 2 seconds\n");
sleep(2);
if((shmctl(shmid, IPC_RMID, 0) < 0))
{
    printf("shmctl error:%s\n", strerror(errno));
    exit(-1);
}
   }

waitpid(pid,NULL,0);
exit(0);
}

输出为:


shmctl(shmid, IPC_RMID, 0)的作用是从系统中删除该恭喜存储段。因为每个共享存储段有一个连接计数(shmid_ds结构中的shm_nattch),所以除非使用该段的最后一个进程终止与该段脱接,否则不会实际上删除该存储段

分类:  Linux_C/C++


### shmget 函数功能 `shmget()` 是 Linux 系统调用函数,用于创建新的共享内存段或获取已有共享内存段。此函数返回一个整数类型的共享内存标识符 (ID),该 ID 可在后续系统调用中引用共享内存段[^1]。 ### 使用方法 #### 声明与参数说明 ```c #include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int shmflg); ``` - `key`: 由 `ftok()` 创建的唯一键值,用来区分不同的共享内存区。 - `size`: 请求分配的空间大小(字节单位)。 - `shmflg`: 控制标志位,通常设置为权限模式加上 IPC_CREAT 或者 IPC_EXCL 标志组合。 当成功执行时,`shmget()` 返回非负整数值作为新获得/现有共享内存段的标识符;如果失败,则返回 -1 并设置 errno 来指示错误原因[^4]。 #### 实际应用案例 下面是一个简单的例子来展示如何利用 `shmget()` 获取并操作共享内存: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/types.h> #include <sys/ipc.h> #include <sys/shm.h> #define SHM_KEY 0x1234 /* 随意定义的一个常量 */ #define PERMISSIONS 0666 | IPC_CREAT int main(void){ char *data; int shmid; // 获得共享内存id if ((shmid = shmget(SHM_KEY, sizeof(char)*50 ,PERMISSIONS)) == -1) { perror("shmget"); exit(EXIT_FAILURE); } printf("Shared memory segment id: %d\n", shmid); // 将共享内存连接到当前进程空间 data = shmat(shmid, NULL, 0); strcpy(data,"Hello Shared Memory!"); puts(data); // 断开共享内存链接 shmdt(data); return EXIT_SUCCESS; } ``` 这段程序展示了怎样通过指定关键字和尺寸请求一段共享内存,并将其附加至进程中以便读写数据。最后记得断开与共享内存之间的联系以释放资源。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值