本文转载至 http://www.cocoachina.com/industry/20140606/8701.html

iOS系统是出了名的封闭,每个应用的活动范围被严格地限制在各自的沙盒中。尽管如此,iOS还是提供了若干进程间通信机制,CFMessagePort就是其中之一。

“”
 
阅读器

iOSCFMessagePort

 
iOS系统是出了名的封闭,每个应用的活动范围被严格地限制在各自的沙盒中。尽管如此,iOS还是提供了若干进程间通信机制,CFMessagePort就是其中之一。
 
从类名可以看出,CFMessagePort属于Core Foundation层的东西,其实现部分是开源的,代码在可以在苹果的开源代码库中找到。
 
使用方式
1、消息接收者
CFMessagePort端口消息的接收者需要实现以下功能:
 
1.1 注册监听
消息接收者需要通过以下方式注册消息监听:
  1. -(void)startListenning
  2. {
  3. if (0 != mMsgPortListenner && CFMessagePortIsValid(mMsgPortListenner))
  4. {
  5. CFMessagePortInvalidate(mMsgPortListenner);
  6. }
  7. mMsgPortListenner = CFMessagePortCreateLocal(kCFAllocatorDefault,CFSTR(LOCAL_MACH_PORT_NAME),onRecvMessageCallBack, NULL, NULL);
  8. CFRunLoopSourceRef source = CFMessagePortCreateRunLoopSource(kCFAllocatorDefault, mMsgPortListenner, 0);
  9. CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopCommonModes);
  10. NSLog(@"start listenning");
  11. }
 
其中LOCAL_MACH_PORT_NAME的定义为:
  1. #define LOCAL_MACH_PORT_NAME    "com.wangzz.demo"
 
经过查看源码发现,CFMessagePort实际上是通过mach port实现的。Mach port是iOS系统提供的基于端口的输入源,可用于线程或进程间通讯。而Runloop支持的输入源类型中就包括基于端口的输入源,因此可以使用Runloop做为CFMessagePort端口源事件的监听者。
 
上述代码有几点需要说明:
1. 通过CFMessagePortCreateLocal可以创建一个本地CFMessagePortRef对象
2. CFMessagePort对象是靠一个字符串来唯一标识的,这一点非常重要,在这里字符串是由宏LOCAL_MACH_PORT_NAME定义的;
3. 创建CFMessagePort对象的同时设置了端口源事件的回调函数onRecvMessageCallBack,用于处理端口源事件;
4. 将创建的对象作为输入源添加到Runloop中,从而实现对端口源事件的监听,当Runloop收到对应的端口源事件时,会调用上一步中指定的回调芳芳;
 
1.2 实现回调方法
回调函数为CFMessagePortCallBack类型,其定义部分为: 
  1. typedef CFDataRef (*CFMessagePortCallBack) (
  2. CFMessagePortRef local,
  3. SInt32 msgid,
  4. CFDataRef data,
  5. void *info
  6. );
 
各个参数的含义为:
CFMessagePortRef local:当前接收消息的CFMessagePortRef对象。
 
SInt32 msgid:这个字段非常有用,用于标识消息。如果通信双方进程约定号每个msgid对应的数据结构,即可实现较为复杂的通信。
 
CFDataRef data:通信的真正数据部分。
 
void *info:为使用CFMessagePortCreateLocal方法创建port端口时指定的CFMessagePortContext对象的info字段,通常为空。
 
该回调方法可以返回一个CFDataRef类型的数据给port消息的发送者,以实现有效的双方通信,这一点也非常重要。
 
我的回调函数onRecvMessageCallBack的实现:
  1. CFDataRef onRecvMessageCallBack(CFMessagePortRef local,SInt32 msgid,CFDataRef cfData, void*info)
  2. {
  3. NSLog(@"onRecvMessageCallBack is called");
  4. NSString *strData = nil;
  5. if (cfData)
  6. {
  7. const UInt8  * recvedMsg = CFDataGetBytePtr(cfData);
  8. strData = [NSString stringWithCString:(char *)recvedMsg encoding:NSUTF8StringEncoding];
  9. /**
  10. 实现数据解析操作
  11. **/
  12. NSLog(@"receive message:%@",strData);
  13. }
  14. //为了测试,生成返回数据
  15. NSString *returnString = [NSString stringWithFormat:@"i have receive:%@",strData];
  16. const char* cStr = [returnString UTF8String];
  17. NSUInteger ulen = [returnString lengthOfBytesUsingEncoding:NSUTF8StringEncoding];
  18. CFDataRef sgReturn = CFDataCreate(NULL, (UInt8 *)cStr, ulen);
  19. return sgReturn;
  20. }
 
该方法实现的较为简单,解析约定的数据(测试代码中约定传送的是string),为了测试,同时生成一个CFDataRef数据返回给port消息的发送者。
 
1.3 取消端口监听
可以通过如下方式取消对port端口的监听:
  1. - (void)endLisenning
  2. {
  3. CFMessagePortInvalidate(mMsgPortListenner);
  4. CFRelease(mMsgPortListenner);
  5. }
 
CFMessagePortInvalidate会停止port消息的发送和接收操作,而只有调用了CFRelease,CFMessagePortRef对象才真正的被释放掉。
 
2、消息发送者
发送部分代码如下:
  1. -(NSString *)sendMessageToDameonWith:(id)msgInfo msgID:(NSInteger)msgid
  2. {
  3. // 生成Remote port
  4. CFMessagePortRef bRemote = CFMessagePortCreateRemote(kCFAllocatorDefault, CFSTR(MACH_PORT_REMOTE));
  5. if (nil == bRemote) {
  6. NSLog(@"bRemote create failed");
  7. return nil;
  8. }
  9. // 构建发送数据(string)
  10. NSString    *msg = [NSString stringWithFormat:@"%@",msgInfo];
  11. NSLog(@"send msg is :%@",msg);
  12. const char *message = [msg UTF8String];
  13. CFDataRef data,recvData = nil;
  14. data = CFDataCreate(NULL, (UInt8 *)message, strlen(message));
  15. // 执行发送操作
  16. CFMessagePortSendRequest(bRemote, msgid, data, 0, 100 , kCFRunLoopDefaultMode, &recvData);
  17. if (nil == recvData) {
  18. NSLog(@"recvData date is nil.");
  19. CFRelease(data);
  20. CFMessagePortInvalidate(bRemote);
  21. CFRelease(bRemote);
  22. return nil;
  23. }
  24. // 解析返回数据
  25. const UInt8  * recvedMsg = CFDataGetBytePtr(recvData);
  26. if (nil == recvedMsg) {
  27. NSLog(@"receive date err.");
  28. CFRelease(data);
  29. CFMessagePortInvalidate(bRemote);
  30. CFRelease(bRemote);
  31. return nil;
  32. }
  33. NSString    *strMsg = [NSString stringWithCString:(char *)recvedMsg encoding:NSUTF8StringEncoding];
  34. NSLog(@"%@",strMsg);
  35. CFRelease(data);
  36. CFMessagePortInvalidate(bRemote);
  37. CFRelease(bRemote);
  38. CFRelease(recvData);
  39. return strMsg;
  40. }
 
其中MACH_PORT_REMOTE的定义为:
  1. #define MACH_PORT_REMOTE    "com.wangzz.demo"
 
发送消息时要相对简单,首先通过CFMessagePortCreateRemote生成一个Remote的CFMessagePortRef,这里需要注意的是CFMessagePortCreateRemote时传入的字符串唯一标识MACH_PORT_REMOTE必须和消息接收者创建local的CFMessagePortRef时使用的字符串唯一标识是同一个!
 
通过查看源码发现,CFMessagePortCreateRemote会根据MACH_PORT_REMOTE定义的字符串为唯一标识获取消息接收者通过CFMessagePortCreateLocal使用相同字符串创建的底层mach port端口,从而实现向消息接收者发送信息。
 
如果消息接收者还没有创建或者通过CFMessagePortCreateLocal创建local端口失败时,想要通过CFMessagePortCreateRemote去创建remote端口肯定是失败的。
 
说明
1. 很遗憾的是,在iOS7及以后系统中,CFMessagePort的通信机制不再可用。
 
在使用CFMessagePortCreateLocal/CFMessagePortCreateRemote创建CFMessagePortRef对象时会失败,官方文档中是这么说的:
This method is not available on iOS 7 and later—it will return NULL and log a sandbox violation in syslog. See Concurrency Programming Guide for possible replacement technologies. 
 
2. CFMessagePort只能用于本地进程通信。
 
3. CFMessagePort是基于mach port端口的通信方式,不但可以用于进程通信,也可以用于线程间通信,只是线程间通信有了GCD和Cocoa提供的原生方法,已经能很方便的实现了,没必要再使用CFMessagePort。
 
4. 进程通信使用场景
 
iOS系统多任务机制,使得进程间通信基本都只能用于越狱开发。常用的场景是前端有一个UI程序用于界面展示,后端有一个daemo精灵程序用于任务处理。
 
demo工程
特地做了了个demo工程,以便更好地演示CFMessagePort的使用,可以到CSDN下载
 
为了模拟进程间通信场景,我将消息接收进程CFMessagePortReceive做成了能够后台播放音乐的程序,以便其切到后台后能继续存活。
 
由于CFMessagePort不再支持iOS7及以后系统,本demo实在iOS6系统上测试的。
 
demo使用方式
1. CFMessagePortReceive启动后,点击Start Listenning创建CFMessagePort接口并开始监听port消息,然后将CFMessagePortReceive切到后台;
 
2. 启动CFMessagePortSend程序,在输入框中写入内容,点击发送按钮即可和CFMessagePortReceive通信。
 
3. MessagePort通信过程中会有日志输出,可以使用以下方式查看日志:
 
1. 真机
选择:Xcode->Window->Organizer->Devices,然后选中窗口左侧当前设备的Console窗口查看。
 
2. 模拟器
选择:模拟器->调试->打开系统日志,或者直接使用快捷键?/直接打开系统控制台查看日志。
 
参考文档:
 
 
 

iOS进程间通信之CFMessagePort的更多相关文章

  1. iOS刨根问底-深入理解RunLoop

    开源的RunloopRef 通常所说的RunLoop指的是NSRunloop或者CFRunloopRef,CFRunloopRef是纯C的函数,而NSRunloop仅仅是CFRunloopRef的OC ...

  2. iOS 系统分析(一) 阅读内核准备知识

    ➠更多技术干货请戳:听云博客 0x01 iOS体系架构 1.1 iOS 系统的整体体系架构 用户体验( The User Experience layer ):SpringBoard 同时支持 Spo ...

  3. IOS随机随学

    1.Objective-C是一种面向对象的语言. 2.Objective-C类声明和实现包括两个部分:接口部分和实现部分. 3.Objective-C中方法不是在“.”运算符,而是采用“[]”运算符. ...

  4. iOS多线程实现1-pthread

    1 操作系统.进程.线程简单介绍 现在的程序都是在操作系统上跑,很少有裸机的,而且大部分的嵌入式应用也都支持操作系统,当然还有一些很低端的嵌入式设备没有操作系统. iPhone手机跑的是iOS操作系统 ...

  5. iOS的TCP/IP协议族剖析&&Socket

    原创文章,版权声明:自由转载-非商用-非衍生-保持署名 | Creative Commons BY-NC-ND 3.0 简介 该篇文章主要回顾--TCP/IP协议族中的TCP/UDP.HTTP:还有S ...

  6. iOS中多线程原理与runloop介绍

    一.线程概述 有些程序是一条直线,起点到终点:有些程序是一个圆,不断循环,直到将它切断.直线的如简单的Hello World,运行打印完,它的生命周期便结束了,像昙花一现那样:圆如操作系统,一直运行直 ...

  7. IOS基础面试题

    最近离职了,找工作,光会做项目,对基础不熟,今天就总结了一点面试题. 废话不多说,上题吧: 1.objective-c中的数字对象都有哪些,简述它们与基本数据类型的区别是什么. 基本类型和C一样,主要 ...

  8. iOS多线程介绍

    一.线程概述 有些程序是一条直线,起点到终点:有些程序是一个圆,不断循环,直到将它切断.直线的如简单的Hello World,运行打印完,它的生命周期便结束了,像昙花一现那样:圆如操作系统,一直运行直 ...

  9. ios开发--tcp/ip

    简介 该篇文章主要回顾--TCP/IP协议族中的TCP/UDP.HTTP:还有Socket.(--该文很干,酝酿了许久!你能耐心看完吗?) 我在这个文章中,列举了常见的TCP/IP族中的协议,今天主角 ...

随机推荐

  1. 阿狸的打字机(bzoj 2434)

    Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...

  2. VMware锁定文件失败开启模块diskearly的操作失败未能启动虚拟机

      删除虚拟机目录下的(如图中标明的就是D:\VMWorks\YeZiZxWeb这个目录)三个 *.lck文件夹,启动正常

  3. [转载][FPGA]有限状态机FSM学习笔记(二)

    1. Mealy和Moore状态机的互换 对于给定的时序逻辑功能,可以用Mealy机实现,也可以用Moore机实现.根据Moore机比Mealy机输出落后一个周期的特性,可以实现两种状态机之间的转换. ...

  4. [转载][FPGA]Quartus代码保护-生成网表文件

    0. 简介 当项目过程中,不想给甲方源码时,该如何?我们可以用网表文件qxp或者vqm对资源进行保护. 下面讲解这两个文件的具体生成步骤: 1. 基本概念 QuartusII的qxp文件为Quartu ...

  5. mybatis学习网站

    http://www.mybatis.org/mybatis-3/zh/index.html

  6. MongoDb 出现配置服务不同步的处理

    主要片方法就是用正常的配置文件的数据覆盖有问题的就行. 引用: http://dba.stackexchange.com/questions/48232/mongodb-config-servers- ...

  7. jquery 查找子元素的几种方法

    <div class="tm-clear tb-hidden tm_brandAttr" id="J_BrandAttr" style="dis ...

  8. python为不同的对象如何分配内存的小知识

    id方法的返回值就是对象的内存地址. python中会为每个出现的对象分配内存,哪怕他们的值完全相等(注意是相等不是相同).如执行a=2.0,b=2.0这两个语句时会先后为2.0这个Float类型对象 ...

  9. hdu5340 Three Palindromes(manacher算法)

    题目描写叙述: 推断能否将字符串S分成三段非空回文串. 解题思路: 源码: #include <cstdio> #include <algorithm> #define MAX ...

  10. 更改已经签名的app中的内容

    转载请说明出处http://blog.csdn.net/andywuchuanlong 记得上次在南昌中兴的一个项目中遇到过一个这种需求:一个app能够给多个渠道商去运营,渠道商推广出去能够获得对应的 ...