首页 | 资讯动态 | linux基础 | 系统管理 | 网络管理 | 编程开发 | linux数据库 | 服务器技术 | linux相关 | linux认证 | 嵌入式 | 下载中心 | 专题 | linux招聘 | 镜像站
OKLinux中文技术站
·设为首页
·加入收藏
·联系我们
系统管理: 中文环境 系统管理 桌面应用 内核技术 | Linux基础: 基础入门 安装配置 常用命令 经验技巧 软件应用 | Linux数据库: Mysql Postgre Oracle DB2 Sybase other
网络管理: 网络安全 网络应用 Linux服务器 环境配置 黑客安全 | 编程开发: PHP CC++ Python Perl Shell 嵌入式开发 java jsp | PHP技术: PHP基础 PHP技巧 PHP应用 PHP文摘
Linux资讯 Linux招聘 Linux专题 Apache | Linux相关: 硬件相关 Linux解决方案 Linux认证 企业应用 其它Unix | 相关下载: 资料下载 参考手册 开发工具 服务器类 软路由 其它
 技术搜索:
会员中心 注册会员 高级搜索  
  → 当前位置:首页>编程开发>cc++>正文

在 Linux 中使用共享对象

http://www.oklinux.cn  2008-03-07  来源: IBM DW Sachin O. Agrawal  会员收藏  游客收藏  【 】 

充分利用共享内存并不总是容易的。在本文中,IBM 的 Sachin Agrawal 与我们共享了他的 C 专门技术,展示了面向对象如何去利用一个独特而实用的进程间通信通道的关键优势。

就时间和空间而言,共享内存可能是所有现代操作系统都具备的最高效的进程间通信通道。共享内存 同时将地址空间映射到多个进程:一个进程只需依附到共享内存并像使用普通内存一样使用它,就可以开始与其他进程进行通信。

不过,在面向对象编程领域中,进程更倾向于使用共享对象而不是原始的信息。通过对象,不需要再对对象中容纳的信息进行序列化、传输和反序列化。共享对象也驻留在共享内存中,尽管这种对象“属于”创建它们的进程,但是系统中的所有进程都可以访问它们。因此,共享对象中的所有信息都应该是严格与特定进程无关的。

这与当前所有流行的编译器所采用的 C 模型是直接矛盾的:C 对象中总是包含指向各种 Vee-Table 和子对象的指针,这些是 与特定进程相关的。要让这种对象可以共享,您需要确保在所有进程中这些指针的目标都驻留在相同的地址。

在一个小的示例的帮助下,本文展示了在哪些情况下 C 模型可以成功使用共享内存模型,哪些情况下不能,以及可能从哪里着手。讨论和示例程序都只限于非静态数据成员和虚函数。其他情形不像它们这样与 C 对象模型关系密切:静态的和非静态非虚拟的成员函数在共享环境中没有任何问题。每个进程的静态成员不驻留在共享内存中(因此没有问题),而共享的静态成员的问题与这里讨论到的问题类似。

环境假定

本文仅局限于用于 32 位 x86 Interl 体系结构的 Red Hat Linux 7.1,使用版本 2.95 的 GNU C 编译器及相关工具来编译和测试程序。不过,您同样可以将所有的思想应用到任意的机器体系结构、操作系统和编译器组合。

示例程序

示例程序由两个客户机构成:shm_client1 和 shm_client2,使用由共享对象库 shm_server 提供的共享对象服务。对象定义在 common.h 中:

清单 1. common.h 中的定义

#ifndef __COMMON_H__
   #define __COMMON_H__

   class A {
   public:
       int m_nA;
       virtual void WhoAmI();

       static void * m_sArena;
       void * operator new (unsigned int);
   };

   class B : public A {
   public:
       int m_nB;
       virtual void WhoAmI();
   };

   class C : virtual public A {
   public:
       int m_nC;
       virtual void WhoAmI();
   };

   void GetObjects(A ** pA, B ** pB, C ** pC);

   #endif //__COMMON_H__

清单 1 定义了三个类(A、B 和 C),它们有一个共同的虚函数 WhoAmI()。基类 A 有一个名为 m_nA 的成员。定义静态成员 m_sArena 和重载操作 new() 是为了可以在共享内存中构造对象。类 B 简单地从 A 继承,类 C 从 A 虚拟地继承。为了确保 A、B 和 C 的大小明显不同,定义了 B::m_nB 和 C::m_nC。这样就简化了 A::operator new() 的实现。GetObjects() 接口返回共享对象的指针。

共享库的实现在 shm_server.cpp 中:
清单 2. 库 - shm_server.cpp

共3页: 上一页 1 [2] [3] 下一页
#include <sys/types.h>
   #include <sys/ipc.h>
   #include <sys/shm.h>
   #include <errno.h>
   #include <stdio.h>
   #include <iostream>

   #include "common.h"

   void * A::m_sArena = NULL;

   void * A::operator new (unsigned int size)
   {
       switch (size)
       {
       case sizeof(A):
           return m_sArena;

       case sizeof(B):
           return (void *)((int)m_sArena   1024);

       case sizeof(C):
           return (void *)((int)m_sArena   2048);

       default:
           cerr << __FILE__ << ":" << __LINE__ << " Critical error" << 
endl;
       }
   }

   void A::WhoAmI() {
       cout << "Object type: A" << endl;
   }


   void B::WhoAmI() {
       cout << "Object type: B" << endl;
   }


   void C::WhoAmI() {
       cout << "Object type: C" << endl;
   }


   void GetObjects(A ** pA, B ** pB, C ** pC) {
       *pA = (A *)A::m_sArena;
       *pB = (B *)((int)A::m_sArena   1024);
       *pC = (C *)((int)A::m_sArena   2048);
   }


   class Initializer {
   public:
       int m_shmid;
       Initializer();

       static Initializer m_sInitializer;
   };

   Initializer Initializer::m_sInitializer;

   Initializer::Initializer()
   {
       key_t key = 1234;
       bool bCreated = false;

       m_shmid = shmget(key, 3*1024, 0666);

       if (-1 == m_shmid) {

           if (ENOENT != errno) {
               cerr << __FILE__ << ":" << __LINE__ << " Critical error" << 
endl;
               return;
           }

           m_shmid = shmget(key, 3*1024, IPC_CREAT | 0666);

           if (-1 == m_shmid) {
               cerr << __FILE__ << ":" << __LINE__ << " Critical error" <<
endl;
               return;
           }

           cout << "Created the shared memory" << endl;
           bCreated = true;
       }

       A::m_sArena = shmat(m_shmid, NULL, 0);

       if (-1 == (int)A::m_sArena) {
           cerr << __FILE__ << ":" << __LINE__ << " Critical error" <<
endl;
           return;
       }

       if (bCreated) {
           // Construct objects on the shared memory
           A * pA;
           pA = new A;
           pA->m_nA = 1;
           pA = new B;
           pA->m_nA = 2;
           pA = new C;
           pA->m_nA = 3;
       }

       return;
   }

上一篇:VS:针对Java开发人员的C#编程语言   下一篇:开发框架--Hibernate中各个包的作用


收藏于收藏夹】 【评论】 【推荐】 【打印】 【关闭
相关文档
·VS:针对Java开发人员的C#编程语言
·使用智能设备扩展在C#中开发自定义控件
·用Eclipse平台进行C/C 开发
·Visual C# 常见问题
·二级C语言实例解答
·一种被忽视的构造和整数溢出重现
·轻轻松松C to C
·与用于 C 的ISO标准保持一致
·用C 的托管扩展针对Windows编程
·运行时和编译时的安全性检查
·轻轻松松C to C (二)
·轻轻松松C to C (三)
·Web Browser Express 概述
·例解用 VC 6.0 实现 JNI
·送给初学者的礼物:游戏编程起源连载二
·送给初学者的礼物:游戏编程起源连载一
发表评论
密码: 匿名评论
评论内容:

(不超过250字,需审核后才会公布,请自觉遵守互联网相关政策法规)
 
  最新文档
·用Eclipse平台进行C/C 开发
·VS:针对Java开发人员的C#编程语言
·使用智能设备扩展在C#中开发自定义控件
·Visual C# 常见问题
·二级C语言实例解答
·一种被忽视的构造和整数溢出重现
·轻轻松松C to C
·与用于 C 的ISO标准保持一致
·用C 的托管扩展针对Windows编程
·运行时和编译时的安全性检查
·轻轻松松C to C (二)
·轻轻松松C to C (三)
  阅读排行
·c/c++ 学习-read 函数和 write 函数
·程序员眼中的qmail(qmail源代码分析)
·Awk 基础入门:Awk 实例编程
·autoconf 和automake生成Makefile文件
·Linux下的多进程编程
·入门文章:教你学会编写Linux设备驱动
·C++自动化(模板元)编程基础与应用
·使用 GDB 调试多进程程序
·Qt 不规则窗体的实现
·嵌入式程序员应知道的几个基本问题
·Linux操作系统中GCC的应用介绍
·C语言中的指针和内存泄漏
·用GNU profiler提高代码运行速度
·vi 中的正则表达式 (Regular Expressio
·Linux 套接字编程中的 5 个隐患
网摘收藏: