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 filename
fname可以是任意一个文件的路径,甚至你可以不用ftok这个函数,直接写key_t key = 0x34542126; 随便写一个数
write:
#include <sys/types.h> #include <sys/ipc.h> #include <sys/shm.h> #include <stdio.h> #include <string.h> struct data_in_out { int rw_flag; char data[100]; }; int main() { key_t key = ftok("/work/projects/shm/shared_mem", 38); if (key < 0) { printf("ftok error "); return -1; } int shm_id = shmget(key, sizeof(struct data_in_out), 0666 | IPC_CREAT); //获得共享内存 if (shm_id < 0) { printf("shmget error "); return -1; } struct data_in_out *addr = (struct data_in_out *)shmat(shm_id, NULL, 0); //映射的内存地址 if (addr == (struct data_in_out *)-1) { printf("shmat error "); return -1; } addr->rw_flag = 0; while(1) { while (addr->rw_flag) { sleep(1); } if (addr->rw_flag == 0) { char *ret = fgets(addr->data, 100, stdin); if (!ret) { printf("fgets error "); return -1; } addr->rw_flag = 1;
if (strncmp(data_io->data, "end", 3) == 0) {
break; } } shmdt(addr); return 0; }
read:
#include <sys/types.h> #include <sys/ipc.h> #include <sys/shm.h> #include <stdio.h> #include <string.h> struct data_in_out { int rw_flag; char data[100]; }; int main() { key_t key = ftok("/work/projects/shm/shared_mem", 38); if (key < 0) { printf("ftok error "); return -1; } printf("key = 0x%x ", key); int shm_id = shmget(key, sizeof(struct data_in_out), 0666 | IPC_CREAT); //获得共享内存 if (shm_id < 0) { printf("shmget error "); return -1; } void *addr = shmat(shm_id, NULL, 0660); //映射的内存地址 if (addr == (void *)-1) { printf("shmat error "); return -1; } struct data_in_out *data_io = (struct data_in_out *)addr; while(1) { while (data_io->rw_flag == 0) { sleep(1); } if (strncmp(data_io->data, "end", 3) == 0) { break; } printf("data : %s ", data_io->data); data_io->rw_flag = 0; } shmdt(addr); shmctl(shm_id, IPC_RMID, 0); return 0; }