clients(PV操作共享内核内存进行输入输出分屏) - server(进程间通信)模型实现
1、拓扑结构

2、PV操作共享内核内存进行输入输出分屏
(1)
int semop(int semid,struct sembuf *sops,size_t nsops);
功能描述
struct sembuf
{
unsigned short sem_num; /* semaphore number */
short sem_op; /* semaphore operation */
short sem_flg; /* operation flags */
};
这三个字段的意义分别为:
void P(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = - ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, );
}
void V(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, ); }
(2)
|
shmget(得到一个共享内存标识符或创建一个共享内存对象)
|
||
|
所需头文件
|
#include <sys/ipc.h>
#include <sys/shm.h>
|
|
|
函数说明
|
得到一个共享内存标识符或创建一个共享内存对象并返回共享内存标识符
|
|
|
函数原型
|
int shmget(key_t key, size_t size, int shmflg)
|
|
|
函数传入值
|
key
|
0(IPC_PRIVATE):会建立新共享内存对象
|
|
大于0的32位整数:视参数shmflg来确定操作。通常要求此值来源于ftok返回的IPC键值
|
||
|
size
|
大于0的整数:新建的共享内存大小,以字节为单位
|
|
|
0:只获取共享内存时指定为0
|
||
|
shmflg
|
0:取共享内存标识符,若不存在则函数会报错
|
|
|
IPC_CREAT:当shmflg&IPC_CREAT为真时,如果内核中不存在键值与key相等的共享内存,则新建一个共享内存;如果存在这样的共享内存,返回此共享内存的标识符
|
||
|
IPC_CREAT|IPC_EXCL:如果内核中不存在键值 与key相等的共享内存,则新建一个共享内存;如果存在这样的共享内存则报错
|
||
|
函数返回值
|
成功:返回共享内存的标识符
|
|
|
出错:-1,错误原因存于error中
|
||
(3)
shm_key = (key_t)atoi(argv[]);
sem_key = (key_t)atoi(argv[]); my_shm = shmget(shm_key, sizeof(MBUF), |IPC_CREAT); my_sem = semget(sem_key, , | IPC_CREAT);
(4)
int semctl(int semid,int semnum,int cmd, /*union semun arg*/);
SETVAL设置信号量集中的一个单独的信号量的值。
semctl(my_sem, , SETVAL, );
(5)
|
shmat(把共享内存区对象映射到调用进程的地址空间)
|
||
|
所需头文件
|
#include <sys/types.h>
#include <sys/shm.h>
|
|
|
函数说明
|
连接共享内存标识符为shmid的共享内存,连接成功后把共享内存区对象映射到调用进程的地址空间,随后可像本地空间一样访问
|
|
|
函数原型
|
void *shmat(int shmid, const void *shmaddr, int shmflg)
|
|
|
函数传入值
|
shmid |
共享内存标识符
|
|
shmaddr
|
指定共享内存出现在进程内存地址的什么位置,直接指定为NULL让内核自己决定一个合适的地址位置
|
|
|
shmflg
|
SHM_RDONLY:为只读模式,其他为读写模式
|
|
|
函数返回值
|
成功:附加好的共享内存地址
|
|
|
出错:-1,错误原因存于error中
|
||
p = (pMBUF)shmat(my_shm, NULL, );
(6)
client_in PV操作:
while( P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
strcpy(p ->m_buf, line);
p ->m_flag = ;
V(my_sem);
(7)
|
shmdt(断开共享内存连接)
|
|
|
所需头文件
|
#include <sys/types.h>
#include <sys/shm.h>
|
|
函数说明
|
与shmat函数相反,是用来断开与共享内存附加点的地址,禁止本进程访问此片共享内存
|
|
函数原型
|
int shmdt(const void *shmaddr)
|
|
函数传入值
|
shmaddr:连接的共享内存的起始地址
|
|
函数返回值
|
成功:0
|
|
出错:-1,错误原因存于error中
|
|
shmdt(p);
(8)
删除内存 删除 信号量
|
shmctl(共享内存管理)
|
||
|
所需头文件
|
#include <sys/types.h>
#include <sys/shm.h>
|
|
|
函数说明
|
完成对共享内存的控制
|
|
|
函数原型
|
int shmctl(int shmid, int cmd, struct shmid_ds *buf)
|
|
|
函数传入值
|
shmid
|
共享内存标识符
|
|
cmd
|
IPC_STAT:得到共享内存的状态,把共享内存的shmid_ds结构复制到buf中
|
|
|
IPC_SET:改变共享内存的状态,把buf所指的shmid_ds结构中的uid、gid、mode复制到共享内存的shmid_ds结构内
|
||
|
IPC_RMID:删除这片共享内存
|
||
|
buf
|
共享内存管理结构体。具体说明参见共享内存内核结构定义部分
|
|
|
函数返回值
|
成功:0
|
|
|
出错:-1,错误原因存于error中
|
||
shmctl(my_shm, IPC_RMID, NULL) ;
semctl(my_sem, , IPC_RMID);//0为信号量编号
3、clients - server 通信
(1)上下线信息管道
server:
char file_name[] = "";
char client_r[], client_w[];
char line[] ;
int fd_read ;
int client_id ;
int fd_cr , fd_cw ;
FILE* fp ;
sprintf(file_name, "%s/%s", PATH, NAME);
mkfifo(file_name, ) ;
fd_read = open(file_name, O_RDONLY);
open(file_name, O_WRONLY); fp = fdopen(fd_read, "r");
client_in:
char server_name[]= "" ;
char read_file[], write_file[] ;
char msg[] ="" ;
int fd_r, fd_w ;
sprintf(server_name, "%s/%s", PATH, SERVER);
int fd_server = open(server_name, O_WRONLY);
sprintf(msg, "%d\n", getpid());
write(fd_server, msg, strlen(msg));
(2) 读写管道
client_in:
memset(read_file, , );
memset(write_file, , ); sprintf(read_file, "%s/%d_r.fifo", PATH, getpid());
sprintf(write_file, "%s/%d_w.fifo", PATH, getpid()); mkfifo(read_file, );
mkfifo(write_file, ); fd_r = open(read_file, O_RDONLY);
fd_w = open(write_file, O_WRONLY);
server:
while(memset(line, , ), fgets(line, , fp) != NULL)// "pid\n"
{// cr cw pid_r.fifo pid_w.fifo
sscanf(line, "%d", &client_id);
printf("client: %d request !\n", client_id) ;
memset(client_r, , );
memset(client_w, , ); sprintf(client_r, "%s/%d_r.fifo", PATH, client_id); sprintf(client_w, "%s/%d_w.fifo", PATH, client_id); fd_cw = open(client_r, O_WRONLY);
fd_cr = open(client_w, O_RDONLY);
/*
****************
*/
close(fd_cr);
close(fd_cw);
}
(3)信息传递
client_in:
while(memset(line, , ), fgets(line, , stdin) != NULL)
{
write(fd_w, line, strlen(line)); memset(line, , ); read(fd_r, line, );
/**
************
*/ }
server:
write(fd_wr, buf, strlen(buf));
4、server fork 子进程对客户端传来的信息进行处理
(1)fork
if(fork() == )
{
child_main(fd_cr, fd_cw);
close(fd_cr);
close(fd_cw);
exit();
}
(2)reserve 函数
void reverse(char* str)
{
int bg, end ;
char tmp ;
bg = ;
end = strlen(str) - ;
while(bg < end)
{
tmp = str[bg] ;
str[bg] = str[end] ;
str[end] = tmp ;
bg ++ ;
end -- ;
} }
(3) child_main read and write
void child_main(int fd_rd, int fd_wr)
{
char buf[] ;
while(memset(buf, , ), read(fd_rd, buf, ) != )
{
reverse(buf);
write(fd_wr, buf, strlen(buf));
} }
(4)防止僵尸进程
如果子进程先退出,系统不会自动清理掉子进程的环境,而必须由父进程调用wait或waitpid函数来完成清理工作,如果父进程不做清理工作,则已经退出的子进程将成为僵尸进程(defunct)。
signal(, child_handle);
下次收到17号信号(子进程退出信号) 就 调用 child_handle 函数
void child_handle(int sig_num)
{
printf("a child exit!\n");
wait(NULL);
}
5、详细代码:
server:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/stat.h>
#include<sys/types.h>
#include<fcntl.h>
#include<signal.h>
#define PATH "/home/comst/pipe"
#define NAME "server.fifo" void child_handle(int sig_num)
{
printf("a child exit!\n");
wait(NULL);
}
void reverse(char* str)
{
int bg, end ;
char tmp ;
bg = ;
end = strlen(str) - ;
while(bg < end)
{
tmp = str[bg] ;
str[bg] = str[end] ;
str[end] = tmp ;
bg ++ ;
end -- ;
} }
void child_main(int fd_rd, int fd_wr)
{
char buf[] ;
while(memset(buf, , ), read(fd_rd, buf, ) != )
{
reverse(buf);
write(fd_wr, buf, strlen(buf));
} }
int main(int argc, char* argv[])
{
signal(, child_handle);
char file_name[] = "";
char client_r[], client_w[];
char line[] ;
int fd_read ;
int client_id ;
int fd_cr , fd_cw ;
FILE* fp ;
sprintf(file_name, "%s/%s", PATH, NAME);
mkfifo(file_name, ) ;
fd_read = open(file_name, O_RDONLY);
open(file_name, O_WRONLY); fp = fdopen(fd_read, "r"); while(memset(line, , ), fgets(line, , fp) != NULL)// "pid\n"
{// cr cw pid_r.fifo pid_w.fifo
sscanf(line, "%d", &client_id);
printf("client: %d request !\n", client_id) ;
memset(client_r, , );
memset(client_w, , ); sprintf(client_r, "%s/%d_r.fifo", PATH, client_id); sprintf(client_w, "%s/%d_w.fifo", PATH, client_id); fd_cw = open(client_r, O_WRONLY);
fd_cr = open(client_w, O_RDONLY);
if(fork() == )
{
child_main(fd_cr, fd_cw);
close(fd_cr);
close(fd_cw);
exit();
}
close(fd_cr);
close(fd_cw);
} memset(file_name, , );
sprintf(file_name, "%s/%s", PATH, NAME);
unlink(file_name);
return ;
}
client.h:
#ifndef __CLINET_H__
#define __CLINET_H__ #include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/sem.h>
typedef struct tag
{
int m_flag ;
char m_buf[] ;
}MBUF, *pMBUF;
void P(int semid) ;
void V(int semid);
#endif
client_in:
#include "client.h"
#include<sys/stat.h>
#include<sys/types.h>
#include<sys/fcntl.h>
#define PATH "/home/comst/pipe"
#define SERVER "server.fifo"
int main(int argc, char* argv[])//shm_key sem_key
{ char server_name[]= "" ;
char read_file[], write_file[] ;
char msg[] ="" ;
int fd_r, fd_w ;
sprintf(server_name, "%s/%s", PATH, SERVER);
int fd_server = open(server_name, O_WRONLY);
sprintf(msg, "%d\n", getpid());
write(fd_server, msg, strlen(msg)); memset(read_file, , );
memset(write_file, , ); sprintf(read_file, "%s/%d_r.fifo", PATH, getpid());
sprintf(write_file, "%s/%d_w.fifo", PATH, getpid()); mkfifo(read_file, );
mkfifo(write_file, ); fd_r = open(read_file, O_RDONLY);
fd_w = open(write_file, O_WRONLY); key_t shm_key, sem_key ;
int my_shm, my_sem ; char line[] ; pMBUF p ;
shm_key = (key_t)atoi(argv[]);
sem_key = (key_t)atoi(argv[]); my_shm = shmget(shm_key, sizeof(MBUF), |IPC_CREAT); my_sem = semget(sem_key, , | IPC_CREAT);
semctl(my_sem, , SETVAL, ); p = (pMBUF)shmat(my_shm, NULL, );
memset(p, , sizeof(MBUF)); while(memset(line, , ), fgets(line, , stdin) != NULL)
{
write(fd_w, line, strlen(line)); memset(line, , ); read(fd_r, line, );
while( P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
strcpy(p ->m_buf, line);
p ->m_flag = ;
V(my_sem); } while( P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
strcpy(p ->m_buf, "over");
p ->m_flag = ;
V(my_sem); sleep(); shmdt(p);
shmctl(my_shm, IPC_RMID, NULL) ; semctl(my_sem, , IPC_RMID); }
client_out:
#include "client.h"
int main(int argc, char* argv[])//shm_key sem_key
{
key_t shm_key, sem_key ;
int my_shm, my_sem ; char line[] ; pMBUF p ;
shm_key = atoi(argv[]);
sem_key = atoi(argv[]); my_shm = shmget(shm_key, sizeof(MBUF), ); my_sem = semget(sem_key, , );
semctl(my_sem, , SETVAL, ); p = (pMBUF)shmat(my_shm, NULL, );
memset(p, , sizeof(MBUF)); while()
{
while(P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
printf("%d : %s\n", getpid(), p -> m_buf);
if(strcmp(p ->m_buf, "over") == )
{ V(my_sem);
break ;
}
p -> m_flag = ;
V(my_sem); } shmdt(p); }
P_V func
#include "client.h"
void P(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = - ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, );
}
void V(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, ); }
clients(PV操作共享内核内存进行输入输出分屏) - server(进程间通信)模型实现的更多相关文章
- 24小时学通Linux内核之如何处理输入输出操作
真的是悲喜交加呀,本来这个寒假早上8点都去练车,两个小时之后再来实验室陪伴Linux内核,但是今天教练说没名额考试了,好纠结,不过想想就可以睡懒觉了,哈哈,自从大三寒假以来还没睡过懒觉呢,现在也有更多 ...
- 内存操作相关内核 API 的使用
1.RtlCopyMemory .RtlCopyBytes.RtlMoveMemory: 2.RtlZeroMemory.RtlFillMemory: 3.RtlEqualMemory: 4.ExAl ...
- php操作共享内存shmop类及简单使用测试(代码)
SimpleSHM 是一个较小的抽象层,用于使用 PHP 操作共享内存,支持以一种面向对象的方式轻松操作内存段.在编写使用共享内存进行存储的小型应用程序时,这个库可帮助创建非常简洁的代码.可以使用 3 ...
- windows内核 内存管理
一.几个基本的概念 1.存储器的金字塔结构 存储器从下之上依次是磁盘/flash.DRAM(内存).L2-cache.L1-cache.寄存器,越在上面的存储器访问速度越快,同时价格也越昂贵,每一级都 ...
- Linux内核内存管理
<Linux内核设计与实现>读书笔记(十二)- 内存管理 内核的内存使用不像用户空间那样随意,内核的内存出现错误时也只有靠自己来解决(用户空间的内存错误可以抛给内核来解决). 所有内核 ...
- Java使用wait() notify()方法操作共享资源
Java多个线程共享资源: 1)wait().notify()和notifyAll()方法是本地方法,并且为final方法,无法被重写. 2)调用某个对象的wait()方法能让当前线程阻塞,并且当前线 ...
- LINUX内核内存屏障
================= LINUX内核内存屏障 ================= By ...
- Java使用wait() notify()方法操作共享资源详解_java - JAVA
文章来源:嗨学网 敏而好学论坛www.piaodoo.com 欢迎大家相互学习 Java多个线程共享资源: 1)wait().notify()和notifyAll()方法是本地方法,并且为final方 ...
- 看完了进程同步与互斥机制,我终于彻底理解了 PV 操作
尽人事,听天命.博主东南大学硕士在读,热爱健身和篮球,乐于分享技术相关的所见所得,关注公众号 @ 飞天小牛肉,第一时间获取文章更新,成长的路上我们一起进步 本文已收录于 CS-Wiki(Gitee 官 ...
随机推荐
- UrlPathEncode与UrlEncode的区别
UrlEncode与UrlPathEncode 的基本作用都是对 URL 字符串进行编码 不同点总结如下: 不同点 UrlEncode UrlPathEncode 处理空格的方式 替换成“+” 替换成 ...
- SSIS 学习(7):包配置(下)【转】
经过前面几个章节的学习,我们开发的ETL包算已经完成一大半了,但是还不够完美,正如一场足球比赛,前面大家打得很辛苦,传接得也很漂亮,但 是临门一脚的技术不过关,进不了球,一切都是白搭.今天我们就来为大 ...
- HDU 1016 Prime Ring Problem (DFS)
Prime Ring Problem Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Other ...
- Java操作Wrod文档的工具类
需要有jacob的jar包支持 import java.util.Iterator; import java.util.List; import java.util.HashMap; import c ...
- 微信小程序(原名微信应用号)开发工具0.9版安装教程
微信小程序全称微信公众平台·小程序,原名微信公众平台·应用号(简称微信应用号) 声明 微信小程序开发工具类似于一个轻量级的IDE集成开发环境,目前仅开放给了少部分受微信官方邀请的人士(据说仅200个名 ...
- 如何设置win7任务栏的计算机快速启动
win7默认会有一个资源管理器的快速启动栏,但是点击的时候会打开“库”,你可能一般不会用这个库,想打开计算机怎么办呢? 其实很简单,再按住shift的同时右键资源管理器的这个快速启动项,然后会出现菜单 ...
- chattr实现文件不可删除
用自己的话解释清楚这件事儿~ 目前问题: Android手机,在/system/app 目录下的apk,使用chmod 修改权限失败,rm命令也删除不掉. 现象: rm failed for wand ...
- 第四十六篇、UICollectionView广告轮播控件
这是利用人的视觉错觉来实现无限轮播,UICollectionView 有很好的重用机制,这只是部分核心代码,后期还要继续完善和代码重构. #import <UIKit/UIKit.h> # ...
- 第三十八篇、给UITabBar按钮的动画效果
在很多情况下,我们也时常有这样的需求,就是在UITabBar切换的时候,添加一些动画效果 1.在UITabBar触发点击方法的时候捕获当前点击的item 2.使用coreAnimation设置动画效果 ...
- JavaScript高级程序开发3笔记
Js对象 注意:js基本数据类型不是对象,但是"abc".match()这种,可以调用对象的方法,是因为调用方法是临时产生了一个wrapper的包装对象,this指向它: Js ...