我正在尝试使用以下代码将NSNetService(名为“My_Mac”)解析为后台应用中的IP:
NSNetService *service = [[NSNetService alloc] initWithDomain:@"local." type:@"_daap._tcp" name:@"My_Mac"];
[service setDelegate:self];
[service resolveWithTimeout:5];
在同一个类中,我定义了这些委托方法:
- (void)netServiceDidResolveAddress:(NSNetService *)sender
- (void)netService:(NSNetService *)sender didNotResolve:(NSDictionary *)errorDict
这是一个奇怪的部分:除非我在“[service resolveWithTimeout:5];”之后运行NSAlert,否则不会调用委托方法。有什么想法吗?
答案 0 :(得分:25)
如果您使用ARC,则需要在某处保留服务对象。 在您的示例中,服务对象很可能超出范围,并且未在代码中的任何其他位置引用,因此编译器将尝试在解析调用后立即释放它。
添加属性:
@property (nonatomic, strong) NSMutableArray *services;
在您的代理中,或在您使用NSNetService的任何地方
- (void)netServiceBrowser:(NSNetServiceBrowser *)browser
didFindService:(NSNetService *)aNetService
moreComing:(BOOL)moreComing
{
if (!self.services) {
self.services = [[NSMutableArray alloc] init];
}
[self.services addObject:aNetService];
[aNetService setDelegate:self];
[aNetService resolveWithTimeout:3.0];
}
完成或删除代表后,不要忘记停止并释放这些服务:
for (NSNetService* service in self.services) {
[service stop];
}
[self.services removeAllObjects];
答案 1 :(得分:9)
我不确定,但看起来请求实际上并没有因为某种原因在运行循环中进行调度。也许尝试像this那样安排它?
NSNetService *service = [[[NSNetService alloc] initWithDomain:@"local." type:@"_daap._tcp." name:@"My_Mac"] autorelease];
[service setDelegate:self];
[service scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:@"PrivateMyMacServiceMode"];
[service resolveWithTimeout:8.0];
愚蠢的问题,但您是明确实施NSServiceDelegate协议还是只有方法?
编辑:我有另一种想法,这可能是某种竞争条件(更可能的情况)。代表们通常都是弱参考。如果您的对象超出范围并被自动释放,系统将以nil句柄结束并将消息触发为nil。在您展示NSAlert(或做其他工作)的情况下,您的对象可能会闲置很长时间,以便将消息反射回来。你可以确认你的物体持续整整8秒钟吗?答案 2 :(得分:0)
我刚刚遇到了这个问题,并且我已经保留了NSNetService
对象,所以这不是原因。
原来是因为我定义的委托方法的访问级别与class
的访问级别不匹配(class
是public
,而委托方法是internal
)。而且由于所有委托方法都是可选的,因此编译器只是发出一个静默警告。