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 官 ...
随机推荐
- [Java] HashMap、TreeMap、Hashtable排序
Java中对Map(HashMap,TreeMap,Hashtable等)的排序时间 首先简单说一下他们之间的区别: HashMap: 最常用的Map,它根据键的HashCode 值存储数据,根据键可 ...
- 【安卓面试题】在一个Activity启动另一个Activity和在Service中启动一个Activity有什么区别
在Activity中可以直接使用Intent启动另一个Activity 显式Intent intent = new Intent(context, activity.class) 隐式 Intent ...
- ios -- NSdata 与 NSString,Byte数组,UIImage 的相互转换(转)
1. NSData 与 NSStringNSData-> NSStringNSString *aString = [[NSString alloc] initWithData:adata enc ...
- ActiveMQ(5.10.0) - Message Redelivery and DLQ Handling
When messages expire on the ActiveMQ broker (they exceed their time-to-live, if set) or can’t be red ...
- django 学习-2 模板
如何使用渲染模板的方法来显示内容. 1.创建一个项目dream django-admin.py startproject dream cd dream 再创建一个应用 python m ...
- android用异步操作AsyncTask编写文件查看器
Activity程序 package com.example.fileasynctaskproject; import java.io.File; import java.util.ArrayList ...
- SharpZipLib 压缩后传输给第三方平台无法识别问题
问题描述:在项目中需要将文件压缩然后传输给三方进行彩信发送,使用SharpZipLib 进行压缩,原先使用J#进行压缩处理,但是用SharpZipLib压缩后的zip文件传输过去之后,总会报发送失败. ...
- angular的post请求,SpringMVC后台接收不到参数值的解决方案
这是我后台SpringMVC控制器接收isform参数的方法,只是简单的打出它的值: @RequestMapping(method = RequestMethod.POST) @ResponseBod ...
- 百练_2945 拦截导弹(DP)
描述 某国为了防御敌国的导弹袭击,开发出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹来袭 ...
- Ubuntu环境变量——添加与删除
转自:http://beanocean.diandian.com/post/2013-11-09/40060047963 注: 1.作者的系统是Ubuntu 13.10,在其他linux发行版中环境变 ...