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 官 ...
随机推荐
- 创建FILE GEODATABASE 和栅格目录及向栅格目录中添加影像
using System;using System.IO;using ESRI.ArcGIS.DataSourcesGDB;using ESRI.ArcGIS.DataSourcesRaster;us ...
- ASP.NET MVC 4 批量上传文件
上传文件的经典写法: <form id="uploadform" action="/Home/UploadFile" method="post& ...
- [改善Java代码]使用valueOf前必须进行校验
每个枚举都是java.lang.Enum的子类,都可以访问Enum类提供的方法,比如hashCode(),name(),valueOf()等..... 其中valueOf()方法会把一个String类 ...
- MSP430常见问题之IO端口类
Q1:请问430 的I/O 中断能不能可靠的响应60ns 的脉冲信号, 就是来了一个60ns 的脉冲,430 的中断会有丢失吗?A1:端口支持的最高8M的时钟,无法响应这么快的频率. Q2:430是3 ...
- sql常识-Join
SQL join 用于根据两个或多个表中的列之间的关系,从这些表中查询数据. Join 和 Key 有时为了得到完整的结果,我们需要从两个或更多的表中获取结果.我们就需要执行 join. 数据库中的表 ...
- Java之绘制五环图案
前面已经介绍过绘制方法.这里不再赘述. package com.caiduping; import java.awt.BasicStroke; import java.awt.Color; impor ...
- ASSERT报错:error C2664: “AfxAssertFailedLine”: 不能将参数 1 从“TCHAR []”转换为“LPCSTR”
转载请注明来源:崨雁嫀筝 http://www.cnblogs.com/xuesongshu 这个错误是我在把tinyxml修改为宽字符(Unicode)版本时候遇到的问题,我首先按关键字把所有有ch ...
- Dreamweaver标签库
.highlight .hll { background-color: #ffffcc } .highlight { background: #ffffff } .highlight .c { col ...
- angular的post请求,SpringMVC后台接收不到参数值的解决方案
这是我后台SpringMVC控制器接收isform参数的方法,只是简单的打出它的值: @RequestMapping(method = RequestMethod.POST) @ResponseBod ...
- DOM_节点层次_Document类型
一.Document类型 nodeType: 9; nodeName: "#document"; nodeValue: null; parentValue: null; owner ...