我正在开发一款使用蓝牙2.1进行通信的iOS应用程序。应用程序在激活时连接到BT。如果应用程序转到后台并再次激活,它会正常工作。
但我刚刚注意到一个问题:
如果我关闭BT模块电源,应用程序会收到通知,我会执行以下操作:
- (void)accessoryDidDisconnect:(EAAccessory *)accessory
{
NSLog(@"EAController::accessoryDidDisconnect:");
_selectedAccessory = nil;
UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"Alert" message:@"Lost connection. " delegate:nil cancelButtonTitle:@"OK" otherButtonTitles: nil];
[alert show];
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
[defaults setBool:YES forKey:@"Lost Communication"];
[self closeSession];
}
然后我打开模块电源,进入设置->蓝牙,将BT模块连接到iPhone,激活应用程序(从后台返回),EASession无法启动:
if (_session == nil)
{
NSLog(@"EAController::openSession");
[_selectedAccessory setDelegate:self];
_session = [[EASession alloc] initWithAccessory:[self selectedAccessory] forProtocol:_protocolString];
if (_session)
{
// Set up delegate........
}
else
{
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
[defaults setBool:NO forKey:Unit_Has_Connection_UserDefault_Key];
NSLog(@"creating session failed");
}
}
启动_accessoryList和_selectedAccessory:
_accessoryList = [[NSMutableArray alloc] initWithArray:[[EAAccessoryManager sharedAccessoryManager] connectedAccessories]];
_selectedAccessory = [_accessoryList objectAtIndex:0];
我发现_accessoryList有我正在使用的附件,但它的"协议"是空的。
$12 = 0x1dd58050 <__NSArrayI 0x1dd58050>(
<EAAccessory: 0x1dd1dce0> {
connected:YES
connectionID:XXX
name: XXX
manufacturer: XXX
modelNumber: XXX
serialNumber:
firmwareRevision: XXX
hardwareRevision: XXX
macAddress: XXX
protocols: (
)
delegate: (null)
}
)
如果我终止应用程序并重新启动,它会正常工作。
有人知道怎么解决这个问题吗?
这个问题和这个问题类似,但发生在不同的情况下,这个问题的答案似乎并不能解决我的问题。
经过几天的寻找解决方案,我终于解决了这个问题(或者我希望我解决了它。
你永远不能相信EAAccessoryManager,它可能有幽灵配件。因此,使用以下命令启动_accessoryList并不总是有效的。
_accessoryList = [[NSMutableArray alloc] initWithArray:[[EAAccessoryManager sharedAccessoryManager] connectedAccessories]];
如果您使用重影附件启动accessoryList,请确保您无法成功启动EASession。
正确的方法是使用EAAccessory Notification,它在这个答案中进行了描述,但没有详细说明如何做到这一点。
首先,您需要创建后两个通知:
[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(accessoryConnected:) name:EAAccessoryDidConnectNotification object:nil];
[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(accessoryDisconnected:) name:EAAccessoryDidConnectNotification object:nil];
[[EAAccessoryManager sharedAccessoryManager]registerForLocalNotifications];
然后在accessoryConnected和accessoryDisconnected中,可以执行以下操作。
- (void)accessoryConnected:(NSNotification *)notification
{
NSLog(@"EAController::accessoryConnected");
EAAccessory *connectedAccessory = [[notification userInfo] objectForKey:EAAccessoryKey];
[[self accessoryList] addObject:connectedAccessory];
if ([_accessoryList count])
{
_selectedAccessory = [_accessoryList objectAtIndex:0];
NSArray *protocolStrings = [_selectedAccessory protocolStrings];
if ([protocolStrings count]) {
self.protocolString = [protocolStrings objectAtIndex:0];
[self openSession];
}
}
}
- (void)accessoryDisconnected:(NSNotification *)notification
{
EAAccessory *disconnectedAccessory = [[notification userInfo] objectForKey:EAAccessoryKey];
int disconnectedAccessoryIndex = 0;
for(EAAccessory *accessory in [self accessoryList]) {
if ([disconnectedAccessory connectionID] == [accessory connectionID]) {
break;
}
disconnectedAccessoryIndex++;
}
if (disconnectedAccessoryIndex < [[self accessoryList] count]) {
[[self accessoryList ] removeObjectAtIndex:disconnectedAccessoryIndex];
} else {
NSLog(@"could not find disconnected accessory in accessory list");
}
NSLog(@"_accessory did disconnect: %@",_accessoryList);
}
从后台返回时,我也遇到了重新连接配件的问题,尤其是在8小时以上空闲之后。在大多数情况下,仅依靠EAAccessoryDidConnectNotification/Disconnect(而不是可以提供重影设备的EAAccessoryManager)进行细化看起来工作良好
然而,如果两个应用程序正在使用我们的配件,一旦手机/平板电脑被唤醒,它可能会出现故障。第一个进入前台的应用程序最终将收到连接通知。但当在第二个应用程序中交换时,它只会收到断开连接的通知,而不会收到连接。在这里,我很难确定我们是否有连接的附件,因为我不能信任EAAccessoryManager。
我发现,当EAAccessory重新连接时,您会看到附件对象的两次刷新。第一个没有协议字符串,而几秒钟后,相同的附件(具有相同的连接ID)再次出现,并带有协议字符串。结构化我的代码s.t.它忽略了任何没有协议字符串的附件,按照苹果推荐的技术。
我相信您的配件必须与MFI(专为iPhone、ipAd和iPod touch设计)芯片一起定义协议,以支持iDevice&附件你需要检查你的固件团队以及苹果MFI程序。该协议需要添加到应用程序plist文件的"支持的外部附件协议"字段中。