本文转载至 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. 莫比乌斯函数之和(51nod 1244)

    莫比乌斯函数,由德国数学家和天文学家莫比乌斯提出.梅滕斯(Mertens)首先使用μ(n)(miu(n))作为莫比乌斯函数的记号.具体定义如下: 如果一个数包含平方因子,那么miu(n) = 0.例如 ...

  2. 2017 UESTC Training for Data Structures-解题报告

    题目链接:http://acm.uestc.edu.cn/#/contest/show/155 这个数据结构训练主要针对线段树,树转数组和并查集.比较适合刚入门数据结构的同学. 注意,因为后面题的代码 ...

  3. 常用的XMPP服务器

    1. Openfire (Wildfire) 3.x 底层通讯采用的mina框架,minak框架其实性能一般,netty早已经超越它,虽然最初都是Doug Lea写的.3.4版本之后支持集群,单台服务 ...

  4. dedecms--后台添加会员栏目(批量添加)

    最近在用dedecms二次开发会员功能,一开始做了一个会员添加,但是领导要求可以批量添加,最好是可以输入添加个数:这样我想添加几个就添加几个了 1:会员添加的htm页面 <html> &l ...

  5. IOS-<input>表单元素只能读,设置readonly时光标仍然可见的解决办

    在HTML中,如果把一个<input>的readonly属性设置为"readonly",表示这个表单元素不能编辑. 但是,鼠标点击之后,这个表单元素还是有光标存在的. ...

  6. R语言实战读书笔记(八)回归

    简单线性:用一个量化验的解释变量预测一个量化的响应变量 多项式:用一个量化的解决变量预测一个量化的响应变量,模型的关系是n阶多项式 多元线性:用两个或多个量化的解释变量预测一个量化的响应变量 多变量: ...

  7. Codeforces 786B Legacy(线段树优化建图)

    题目链接  Legacy 首先对于输入的$n$,建立一棵线段树. 显然线段树有大概$2n$个结点,每个节点对应一段区间 我们把这$2n$个结点加入我们的无向图中,一起跑最短路. 具体连边方案: 我们把 ...

  8. JavaScript 中那些关于坐标和距离的属性与方法

    一 前言 在前端开发中总会遇到各种各样需要使用或计算坐标和距离的情况,但是这些属性和方法众多,全部熟练地记下来并非是一件易事,大多只能现查,耗费不少时间精力,于是便有了整理记录的想法,即加深了印象,又 ...

  9. 洛谷——P2706 巧克力

    P2706 巧克力 题目背景 王7的生日到了,他的弟弟准备送他巧克力. 题目描述 有一个被分成n*m格的巧克力盒,在(i,j)的位置上有a[i,j]块巧克力.就在送出它的前一天晚上,有老鼠夜袭巧克力盒 ...

  10. Boost Asio介绍--之一

    原文:http://www.tuicool.com/articles/YbeYR3 Boost Asio介绍--之一 时间 2014-03-26 17:57:39  CSDN博客 原文  http:/ ...