我們可以看到上面的拷貝次數(shù)是4次,下面則是2次。
接下來我們看看使用,其實(shí)網(wǎng)上和書上都有了很多資料,API就是那么幾個。
int shmget(key_t key,size_t size,int shmflag)。
key:可以指定IPC_PRIVATE,那么系統(tǒng)會為你創(chuàng)建一個key,并且返回一個id號。也可以通過ftok函數(shù)生成一個key(不了解ftok的童鞋可以動手man一下)。那么為什么要一個IPC既要一個key又有一個ID呢。這里我覺得是為了方便其他進(jìn)程訪問。進(jìn)程A創(chuàng)建了一個共享內(nèi)存,內(nèi)核為其分配一個ID,這個時候進(jìn)程B想要訪問,他怎么獲取這個ID(難道需要A把ID發(fā)送給B??)。但是我們用一個key就很方便了。事先A,B進(jìn)程都知道這個key,那么A創(chuàng)建了,B就可以通過事先知道key找到這塊內(nèi)存。
size:共享內(nèi)存的大小。如果是獲得一塊內(nèi)存,則該值應(yīng)該為0。
shmflag:讀寫權(quán)限的組合,可以與IPC_CREAT和IPC_EXCL按位或。當(dāng)指定一個key時,IPC_CREAT和IPC_EXCL配合使用可以在存在該key的共享內(nèi)存時返回-1。
當(dāng)該函數(shù)調(diào)用成功后,返回一個系統(tǒng)分配的共享內(nèi)存,并且size大小的字節(jié)被初始化為0
void *shmat(int shmid, const void *shmaddr, int shmflg)
有了一塊共享內(nèi)存后,進(jìn)程需要映射該內(nèi)存到進(jìn)程的地址空間。這個函數(shù)就是作用。
shmid就是之前獲得ID,shmaddr如果指定了,就會配合shmflg確定映射地址,不過一般都不這么干的。返回值就是獲得的地址,你可以往里面寫你或者讀你需要的數(shù)據(jù)了。同時調(diào)用了該函數(shù),系統(tǒng)會修改shmid_ds數(shù)據(jù)。
int shmdt(const void *shmaddr)
解除綁定關(guān)系,參數(shù)就是我們之前獲取的那個返回地址。(其實(shí)我覺得這里參數(shù)如果為ID貌似更統(tǒng)一些吧)
簡單介紹了API,還要說一下一些注意的東西
1.共享內(nèi)存不會把數(shù)據(jù)寫入磁盤文件中,這個區(qū)別于mmap
2.即使沒有進(jìn)程綁定在共享內(nèi)存,共享內(nèi)存也不會消失的。必須通過shmctl或者ipcrm刪除(或者更暴力的方式關(guān)掉電腦)
另外我們可能會考慮,系統(tǒng)最多創(chuàng)建多少個共享內(nèi)存,一個進(jìn)程最多可以綁定多少個內(nèi)存,一個共享內(nèi)存創(chuàng)建的size最大最小值是多少。其實(shí)這些設(shè)置在/proc/sys/kernel下面,我們也可以自己寫程序來讀取。貼一段代碼用來獲取上面的信息
有了一塊共享內(nèi)存后,進(jìn)程需要映射該內(nèi)存到進(jìn)程的地址空間。這個函數(shù)就是作用。
shmid就是之前獲得ID,shmaddr如果指定了,就會配合shmflg確定映射地址,不過一般都不這么干的。返回值就是獲得的地址,你可以往里面寫你或者讀你需要的數(shù)據(jù)了。同時調(diào)用了該函數(shù),系統(tǒng)會修改shmid_ds數(shù)據(jù)。
int shmdt(const void *shmaddr)
解除綁定關(guān)系,參數(shù)就是我們之前獲取的那個返回地址。(其實(shí)我覺得這里參數(shù)如果為ID貌似更統(tǒng)一些吧)
int shmctl(int shmid, int cmd, struct shmid_ds *buf)
這個函數(shù)主要做一些修改和查詢。比如設(shè)置鎖,解鎖,移除一塊共享內(nèi)存。簡單介紹了API,還要說一下一些注意的東西
1.共享內(nèi)存不會把數(shù)據(jù)寫入磁盤文件中,這個區(qū)別于mmap
2.即使沒有進(jìn)程綁定在共享內(nèi)存,共享內(nèi)存也不會消失的。必須通過shmctl或者ipcrm刪除(或者更暴力的方式關(guān)掉電腦)
另外我們可能會考慮,系統(tǒng)最多創(chuàng)建多少個共享內(nèi)存,一個進(jìn)程最多可以綁定多少個內(nèi)存,一個共享內(nèi)存創(chuàng)建的size最大最小值是多少。其實(shí)這些設(shè)置在/proc/sys/kernel下面,我們也可以自己寫程序來讀取。貼一段代碼用來獲取上面的信息
#define MAX_SHMIDS 8196
int main(int argc,char *argv[])
{
int i,j;
int shmid[MAX_SHMIDS] = {0};
void *addr[MAX_SHMIDS] = {0};
//測試可以創(chuàng)建多少個共享內(nèi)存
for ( i = 0;i < MAX_SHMIDS;++i )
{
shmid[i] = shmget(IPC_PRIVATE,1024,0666|IPC_CREAT);
if ( shmid[i] == -1 )
{
printf("create shared memory failed,max create num[%d],%s\r\n",i,strerror(errno));
break;
}
}
for ( int j = 0;j < i;++j )
{
shmctl(shmid[j],IPC_RMID,NULL);
}
//測試每個進(jìn)程可以attach的最大數(shù)
for ( i = 0;i < MAX_SHMIDS;++i )
{
shmid[i] = shmget(IPC_PRIVATE,1024,0666|IPC_CREAT);
if ( shmid[i] != -1 )
{
addr[i] = shmat(shmid[i],0,0);
if ( addr[i] == (void *)-1 )
{
printf("process attach shared memory failed,max num[%d],%s\r\n",i,strerror(errno));
shmctl(shmid[i],IPC_RMID,NULL);
break;
}
}
else
{
printf("max num of process attach shared memory is[%d]\r\n",i-1);
break;
}
}
for ( j = 0;j < i;++j )
{
shmdt(addr[j]);
shmctl(shmid[j],IPC_RMID,NULL);
}
//測試一個共享內(nèi)存創(chuàng)建最小的size
size_t size = 0;
for ( ;;size++ )
{
shmid[0] = shmget(IPC_PRIVATE,size,0666|IPC_CREAT);
if ( shmid[0] != -1 )
{
printf("create shared memory succeed,min size[%d]\r\n",size);
shmctl(shmid[0],IPC_RMID,NULL);
break;
}
}
//測試共享內(nèi)存創(chuàng)建最大的size
for ( size = 65536;;size += 1024 )
{
shmid[0] = shmget(IPC_PRIVATE,size,0666|IPC_CREAT);
if ( shmid[0] == -1 )
{
printf("create shared memory failed,max size[%ld],%s\r\n",size,strerror(errno));
break;
}
shmctl(shmid[0],IPC_RMID,NULL);
}
exit(0);
}
好了,下篇開始介紹如何控制讀寫。int main(int argc,char *argv[])
{
int i,j;
int shmid[MAX_SHMIDS] = {0};
void *addr[MAX_SHMIDS] = {0};
//測試可以創(chuàng)建多少個共享內(nèi)存
for ( i = 0;i < MAX_SHMIDS;++i )
{
shmid[i] = shmget(IPC_PRIVATE,1024,0666|IPC_CREAT);
if ( shmid[i] == -1 )
{
printf("create shared memory failed,max create num[%d],%s\r\n",i,strerror(errno));
break;
}
}
for ( int j = 0;j < i;++j )
{
shmctl(shmid[j],IPC_RMID,NULL);
}
//測試每個進(jìn)程可以attach的最大數(shù)
for ( i = 0;i < MAX_SHMIDS;++i )
{
shmid[i] = shmget(IPC_PRIVATE,1024,0666|IPC_CREAT);
if ( shmid[i] != -1 )
{
addr[i] = shmat(shmid[i],0,0);
if ( addr[i] == (void *)-1 )
{
printf("process attach shared memory failed,max num[%d],%s\r\n",i,strerror(errno));
shmctl(shmid[i],IPC_RMID,NULL);
break;
}
}
else
{
printf("max num of process attach shared memory is[%d]\r\n",i-1);
break;
}
}
for ( j = 0;j < i;++j )
{
shmdt(addr[j]);
shmctl(shmid[j],IPC_RMID,NULL);
}
//測試一個共享內(nèi)存創(chuàng)建最小的size
size_t size = 0;
for ( ;;size++ )
{
shmid[0] = shmget(IPC_PRIVATE,size,0666|IPC_CREAT);
if ( shmid[0] != -1 )
{
printf("create shared memory succeed,min size[%d]\r\n",size);
shmctl(shmid[0],IPC_RMID,NULL);
break;
}
}
//測試共享內(nèi)存創(chuàng)建最大的size
for ( size = 65536;;size += 1024 )
{
shmid[0] = shmget(IPC_PRIVATE,size,0666|IPC_CREAT);
if ( shmid[0] == -1 )
{
printf("create shared memory failed,max size[%ld],%s\r\n",size,strerror(errno));
break;
}
shmctl(shmid[0],IPC_RMID,NULL);
}
exit(0);
}