共享内存

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\n");
        return -1;
    }
    int shm_id = shmget(key, sizeof(struct data_in_out), 0666 | IPC_CREAT);   //获得共享内存
    if (shm_id < 0) {
        printf("shmget error\n");
        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\n");
        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\n");
                   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\n");
        return -1;
    }
    printf("key = 0x%x \n", key);
    int shm_id = shmget(key, sizeof(struct data_in_out), 0666 | IPC_CREAT);   //获得共享内存
    if (shm_id < 0) {
        printf("shmget error\n");
        return -1;
    }
    void *addr = shmat(shm_id, NULL, 0660);   //映射的内存地址
    if (addr == (void *)-1) {
        printf("shmat error\n");
        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 \n", data_io->data);    
        data_io->rw_flag = 0;
    }
    shmdt(addr);
    shmctl(shm_id, IPC_RMID, 0);
    return 0;
}

 

posted @ 2019-03-30 15:38  一条水煮鱼  阅读(214)  评论(0编辑  收藏  举报