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 官 ...
随机推荐
- javascript的变量、作用域和内存问题
基本类型和引用类型的值执行环境垃圾收集 ECMAScript 变量可能包含两种不同数据类型的值:基本类型值和引用类型值.基本类型值指的是简单的数据段引用类型值指那些可能由多个值构成的对象 基本数据类型 ...
- plsql常用快捷键
plsql使用技巧 1.类SQL PLUS窗口:File->New->Command Window,这个类似于oracle的客户端工具sql plus,但比它好用多了. 2.设置关键字自动 ...
- flex数据交互方式 转
Flex数据交互方法- httpservice, webservice, RemoteObject, socket. 写在前面: 使用SOAP Web Service同Flex交互有很多好处,但是它很 ...
- hdu 1195 广度搜索
这题我们可以用优先队列,每次弹出队列中操作次数最少的一个,那么当找到匹配数时,该值一定是最优的.需要注意的时,加个vi[]数组,判读当前数是否已经存在于队列中.我做的很烦啊~~~ #include&l ...
- 核心概念 —— 服务提供者
1.简介 服务提供者是Laravel应用启动的中心,你自己的应用以及所有Laravel的核心服务都是通过服务提供者启动. 但是,我们所谓的"启动"指的是什么?通常,这意味着注册事物 ...
- Unity3D导出的EXE不用显示分辨率选择界面
在导出游戏的时候,选择Build-setting ->Player-setting; 具体如图: resolution 是选分辨率和屏幕大小display resolution 选disable ...
- HDOJ2029Palindromes _easy version
Palindromes _easy version Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Jav ...
- C#winform MDI子窗体打开时内容显示不全
出现这种情况一般是 打开了多个MDI的子窗体,打开新窗体的时候关闭其他的子窗体就OK了, 具体代码: foreach (Form form in main.MdiChildren) ...
- CF下的BackgroudWorker组件优化.
.net compact framwork(2.0/3.5)下没有Backgroundworder组件,在网上找了一个类 经过使用发现了一些问题,主要有一个问题:在一个Dowork事件中对Report ...
- [转] Portable Trac 简单介绍 - 兼谈为什么不选择 Redmine
Portable Trac 简单介绍 - 兼谈为什么不选择 Redmine Trac是一个轻量级的软件项目管理环境,如果在工作中涉及一个开发团队的管理并且关心项目管理工具的话,相信都在 Trac. ...