通过从主应用程序发送消息,我成功地创建了 XPC 服务并与 XPC 服务通信。但我想知道的是,是否可以启动从 XPC 服务到主应用程序的通信。Apple 文档说 XPC 是双向的。如果有人能用一个例子指出我正确的方向,将不胜感激。
请注意,
- 我想从主应用程序启动 XPC。
- 从主应用程序与 XPC 通信。
- 当某些事件发生时,XPC 应该向主应用程序发送消息。
我在前两个中成功了,但在第三个中找不到任何资源。
谢谢。:)
通过从主应用程序发送消息,我成功地创建了 XPC 服务并与 XPC 服务通信。但我想知道的是,是否可以启动从 XPC 服务到主应用程序的通信。Apple 文档说 XPC 是双向的。如果有人能用一个例子指出我正确的方向,将不胜感激。
请注意,
我在前两个中成功了,但在第三个中找不到任何资源。
谢谢。:)
想通了一切。下面应该是一个不错的例子:
ProcessorListener.h(包含在客户端和服务器中):
@protocol Processor
- (void) doProcessing: (void (^)(NSString *response))reply;
@end
@protocol Progress
- (void) updateProgress: (double) currentProgress;
- (void) finished;
@end
@interface ProcessorListener : NSObject<NSXPCListenerDelegate, Processor>
@property (weak) NSXPCConnection *xpcConnection;
@end
ProcessorListener.m:(仅包含在服务器中)
#import "ProcessorListener.h"
@implementation ProcessorListener
- (BOOL)listener:(NSXPCListener *)listener shouldAcceptNewConnection:(NSXPCConnection *)newConnection
{
[newConnection setExportedInterface: [NSXPCInterface interfaceWithProtocol:@protocol(Processor)]];
[newConnection setExportedObject: self];
self.xpcConnection = newConnection;
newConnection.remoteObjectInterface = [NSXPCInterface interfaceWithProtocol: @protocol(Progress)];
// connections start suspended by default, so resume and start receiving them
[newConnection resume];
return YES;
}
- (void) doProcessing: (void (^)(NSString *g))reply
{
dispatch_async(dispatch_get_global_queue(0,0), ^{
for(int index = 0; index < 60; ++index)
{
[NSThread sleepWithTimeInterval: 1];
[[self.xpcConnection remoteObjectProxy] updateProgress: (double)index / (double)60 * 100];
}
[[self.xpcConnection remoteObjectProxy] finished];
}
// nil is a valid return value.
reply(@"This is a reply!");
}
@end
MainApplication.m(您的主应用程序):
#import "ProcessorListener.h"
- (void) executeRemoteProcess
{
// Create our connection
NSXPCInterface * myCookieInterface = [NSXPCInterface interfaceWithProtocol: @protocol(Processor)];
NSXPCConnection * connection = [[NSXPCConnection alloc] initWithServiceName: kServiceName];
[connection setRemoteObjectInterface: myCookieInterface];
connection.exportedInterface = [NSXPCInterface interfaceWithProtocol:@protocol(Progress)];
connection.exportedObject = self;
[connection resume];
id<Processor> theProcessor = [connection remoteObjectProxyWithErrorHandler:^(NSError *err)
{
NSAlert *alert = [[NSAlert alloc] init];
[alert addButtonWithTitle: @"OK"];
[alert setMessageText: err.localizedDescription];
[alert setAlertStyle: NSWarningAlertStyle];
[alert performSelectorOnMainThread: @selector(runModal) withObject: nil waitUntilDone: YES];
}];
[theProcessor doProcessing: ^(NSString * response)
{
NSLog(@"Received response: %@", response);
}];
}
#pragma mark -
#pragma mark Progress
- (void) updateProgress: (double) currentProgress
{
NSLog(@"In progress: %f", currentProgress);
}
- (void) finished
{
NSLog(@"Has finished!");
}
@end
请注意,此代码是基于我的工作代码的精简版本。它可能无法 100% 编译,但显示了使用的关键概念。在示例中,doProcessing
异步运行以表明Progress
即使在初始方法返回并Received response: This is a reply!
记录后,协议中定义的回调仍会执行。