Swift MultipeerConnectivity 无条件地使 DateComponents 崩溃 BridgeFromObjectiveC
Swift MultipeerConnectivity crash DateComponents unconditionallyBridgeFromObjectiveC
我有一个以 EXC_BREAKPOINT 结尾的可重复崩溃,如下图所示:
重现崩溃的步骤:
- 连接两个设备
- 使用 func sendResource 开始传输(在资源URL:URL,
withName 资源名称:字符串,
toPeer peerID:MCPeerID,
withCompletionHandler completionHandler: ((Error?) -> Void)? = 无)-> 进度?
- 通过调用 func disconnect() 断开启动传输的设备
编辑:重现崩溃的另一种方法,通过调用 Progress.cancel() 步骤:
- 连接两个设备
- 开始传输并存储 Progress 对象 让进度:Progress = session.sendResource(...)
- 在进度对象上调用取消,导致另一台设备崩溃progress.cancel()
我的代码在 didFinishReceivingResourceWithName:
行
func session(_ session: MCSession, didFinishReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL, withError error: Error?) {
// transfer to local URL
MusicDownloadRequestor.sharedInstance.finishReceivingSongUploadAtLocalURL(tempUrl: localURL)
}
在堆栈跟踪中似乎没有调用我的代码...
* thread #25: tid = 0x806ec, 0x0000000100944af4 libswiftFoundation.dylib`static Foundation.DateComponents._unconditionallyBridgeFromObjectiveC (Swift.Optional<__ObjC.NSDateComponents>) -> Foundation.DateComponents with unmangled suffix "_merged" + 96, queue = 'com.apple.MCSession.callbackQueue', stop reason = EXC_BREAKPOINT (code=1, subcode=0x100944af4)
frame #0: 0x0000000100944af4 libswiftFoundation.dylib`static Foundation.DateComponents._unconditionallyBridgeFromObjectiveC (Swift.Optional<__ObjC.NSDateComponents>) -> Foundation.DateComponents with unmangled suffix "_merged" + 96
frame #1: 0x0000000100114c60 MyAppSwift`@objc NetworkManager.session(MCSession, didFinishReceivingResourceWithName : String, fromPeer : MCPeerID, at : URL, withError : Error?) -> () + 168 at NetworkManager.swift:0
frame #2: 0x00000001a1dda028 MultipeerConnectivity`__79-[MCSession syncCloseIncomingStream:forPeer:state:error:reason:removeObserver:]_block_invoke + 208
frame #3: 0x0000000100c05258 libdispatch.dylib`_dispatch_call_block_and_release + 24
frame #4: 0x0000000100c05218 libdispatch.dylib`_dispatch_client_callout + 16
frame #5: 0x0000000100c12aec libdispatch.dylib`_dispatch_queue_serial_drain + 1136
frame #6: 0x0000000100c08ce0 libdispatch.dylib`_dispatch_queue_invoke + 672
frame #7: 0x0000000100c14e2c libdispatch.dylib`_dispatch_root_queue_drain + 584
frame #8: 0x0000000100c14b78 libdispatch.dylib`_dispatch_worker_thread3 + 140
frame #9: 0x000000018c2a32a0 libsystem_pthread.dylib`_pthread_wqthread + 1288
frame #10: 0x000000018c2a2d8c libsystem_pthread.dylib`start_wqthread + 4
更新 #1:添加堆栈跟踪作为文本
更新 #2:找到了导致崩溃的可能线索,here's another crash with unconditionallyBridgeFromObjectiveC
我认为问题是 URL 被作为 nil 传递给 didFinishReceivingResourceWithName 但参数是非可选的。这是有道理的,因为如果文件传输失败,URL 将没有最后的归宿。有什么办法可以解决这个问题或拦截错误吗?
我认为这是一个 Apple 错误。有人有解决方法的建议吗?
使用混合和匹配功能,我能够将我的 MCSessionDelegate 转换为 Objective-C,然后将参数传递给 Swift。检查传入值后,我确认 URL 类型的变量 "localURL" 实际上是 nil,这意味着它应该被声明为可选的。我向 Apple 提交了错误报告。
作为解决方法,请在 Objective-C 中编写 MCSessionDelegate。有几个步骤可以让 Objective-C 和 Swift 在同一个项目中一起工作,this page explains it very well.
这是我的代码。
我的 MCSessionDelegate header:
#import <Foundation/Foundation.h>
#import <MultipeerConnectivity/MultipeerConnectivity.h>
@class NetworkMCSessionTranslator;
@interface NetworkMCSessionDelegate : NSObject <MCSessionDelegate>
-(instancetype)initWithTranslator:(NetworkMCSessionTranslator*) trans;
@end
我的 MCSessionDelegate 实现文件:
#import "NetworkMCSessionDelegate.h"
#import "MusicAppSwift-Swift.h"
@interface NetworkMCSessionDelegate()
@property (nonatomic) NetworkMCSessionTranslator *translator;
@end
@implementation NetworkMCSessionDelegate
/*
create a NetworkMCSessionDelegate and pass in a swift translator
*/
-(instancetype)initWithTranslator:(NetworkMCSessionTranslator*) trans{
if(self = [super init]){
self.translator = trans;
}
return self;
}
/*
Indicates that an NSData object has been received from a nearby peer.
*/
- (void)session:(MCSession *)session didReceiveData:(NSData *)data fromPeer:(MCPeerID *)peerID{
[self.translator networkSession:session didReceive:data fromPeer:peerID];
}
/*
Indicates that the local peer began receiving a resource from a nearby peer.
*/
- (void)session:(MCSession *)session didStartReceivingResourceWithName:(NSString *)resourceName fromPeer:(MCPeerID *)peerID withProgress:(NSProgress *)progres{
[self.translator networkSession:session didStartReceivingResourceWithName:resourceName fromPeer:peerID with:progres];
}
/*
Indicates that the local peer finished receiving a resource from a nearby peer.
*/
- (void)session:(MCSession *)session didFinishReceivingResourceWithName:(NSString *)resourceName fromPeer:(MCPeerID *)peerID atURL:(NSURL *)localURL withError:(NSError *)error{
[self.translator networkSession:session didFinishReceivingResourceWithName:resourceName fromPeer:peerID at:localURL withError:error];
}
/*
Called when the state of a nearby peer changes.
*/
- (void)session:(MCSession *)session peer:(MCPeerID *)peerID didChangeState:(MCSessionState)state{
[self.translator networkSession:session peer:peerID didChange:state];
}
/*
Called when a nearby peer opens a byte stream connection to the local peer.
*/
- (void)session:(MCSession *)session didReceiveStream:(NSInputStream *)stream withName:(NSString *)streamName fromPeer:(MCPeerID *)peerID{
// not expecting to see any of this
}
@end
我的 Swift 文件收集来自代理的流量:
import Foundation
import MultipeerConnectivity
@objc class NetworkMCSessionTranslator: NSObject{
public func networkSession(_ session: MCSession, peer peerID: MCPeerID, didChange state: MCSessionState){
...
}
public func networkSession(_ session: MCSession, didReceive data: Data, fromPeer peerID: MCPeerID){
...
}
public func networkSession(_ session: MCSession, didFinishReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL?, withError error: Error?){
// !!! Notice localURL is now an OPTIONAL !!!
}
public func networkSession(_ session: MCSession, didStartReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, with progress: Progress){
...
}
}
祝你好运!
只是让人们知道这个错误已经随着 xcode 9 beta 1 的发布得到修复。根据文档和我的测试,at localurl 参数现在按预期是可选的:https://developer.apple.com/documentation/multipeerconnectivity/mcsessiondelegate/1406984-session
只需更新到 xcode9 的最新版本即可解决此问题。
在 iOS 7 和 xcode 8 及更高版本中,将 localURL 更改为可选的函数
使用这个功能
func session(_ session: MCSession, didFinishReceivingResourceWithName
resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL?,
withError error: Error?) {
if (localURL != nil) {
//enter code here
}
}
而不是
func session(_ session: MCSession, didFinishReceivingResourceWithName
resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL,
withError error: Error?) {
}
所以,这个功能解决了我的崩溃问题。
xcode 8 如果有警告,忽略。
我有一个以 EXC_BREAKPOINT 结尾的可重复崩溃,如下图所示:
重现崩溃的步骤:
- 连接两个设备
- 使用 func sendResource 开始传输(在资源URL:URL, withName 资源名称:字符串, toPeer peerID:MCPeerID, withCompletionHandler completionHandler: ((Error?) -> Void)? = 无)-> 进度?
- 通过调用 func disconnect() 断开启动传输的设备
编辑:重现崩溃的另一种方法,通过调用 Progress.cancel() 步骤:
- 连接两个设备
- 开始传输并存储 Progress 对象 让进度:Progress = session.sendResource(...)
- 在进度对象上调用取消,导致另一台设备崩溃progress.cancel()
我的代码在 didFinishReceivingResourceWithName:
行func session(_ session: MCSession, didFinishReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL, withError error: Error?) {
// transfer to local URL
MusicDownloadRequestor.sharedInstance.finishReceivingSongUploadAtLocalURL(tempUrl: localURL)
}
在堆栈跟踪中似乎没有调用我的代码...
* thread #25: tid = 0x806ec, 0x0000000100944af4 libswiftFoundation.dylib`static Foundation.DateComponents._unconditionallyBridgeFromObjectiveC (Swift.Optional<__ObjC.NSDateComponents>) -> Foundation.DateComponents with unmangled suffix "_merged" + 96, queue = 'com.apple.MCSession.callbackQueue', stop reason = EXC_BREAKPOINT (code=1, subcode=0x100944af4)
frame #0: 0x0000000100944af4 libswiftFoundation.dylib`static Foundation.DateComponents._unconditionallyBridgeFromObjectiveC (Swift.Optional<__ObjC.NSDateComponents>) -> Foundation.DateComponents with unmangled suffix "_merged" + 96
frame #1: 0x0000000100114c60 MyAppSwift`@objc NetworkManager.session(MCSession, didFinishReceivingResourceWithName : String, fromPeer : MCPeerID, at : URL, withError : Error?) -> () + 168 at NetworkManager.swift:0
frame #2: 0x00000001a1dda028 MultipeerConnectivity`__79-[MCSession syncCloseIncomingStream:forPeer:state:error:reason:removeObserver:]_block_invoke + 208
frame #3: 0x0000000100c05258 libdispatch.dylib`_dispatch_call_block_and_release + 24
frame #4: 0x0000000100c05218 libdispatch.dylib`_dispatch_client_callout + 16
frame #5: 0x0000000100c12aec libdispatch.dylib`_dispatch_queue_serial_drain + 1136
frame #6: 0x0000000100c08ce0 libdispatch.dylib`_dispatch_queue_invoke + 672
frame #7: 0x0000000100c14e2c libdispatch.dylib`_dispatch_root_queue_drain + 584
frame #8: 0x0000000100c14b78 libdispatch.dylib`_dispatch_worker_thread3 + 140
frame #9: 0x000000018c2a32a0 libsystem_pthread.dylib`_pthread_wqthread + 1288
frame #10: 0x000000018c2a2d8c libsystem_pthread.dylib`start_wqthread + 4
更新 #1:添加堆栈跟踪作为文本
更新 #2:找到了导致崩溃的可能线索,here's another crash with unconditionallyBridgeFromObjectiveC
我认为问题是 URL 被作为 nil 传递给 didFinishReceivingResourceWithName 但参数是非可选的。这是有道理的,因为如果文件传输失败,URL 将没有最后的归宿。有什么办法可以解决这个问题或拦截错误吗?
我认为这是一个 Apple 错误。有人有解决方法的建议吗?
使用混合和匹配功能,我能够将我的 MCSessionDelegate 转换为 Objective-C,然后将参数传递给 Swift。检查传入值后,我确认 URL 类型的变量 "localURL" 实际上是 nil,这意味着它应该被声明为可选的。我向 Apple 提交了错误报告。
作为解决方法,请在 Objective-C 中编写 MCSessionDelegate。有几个步骤可以让 Objective-C 和 Swift 在同一个项目中一起工作,this page explains it very well.
这是我的代码。
我的 MCSessionDelegate header:
#import <Foundation/Foundation.h>
#import <MultipeerConnectivity/MultipeerConnectivity.h>
@class NetworkMCSessionTranslator;
@interface NetworkMCSessionDelegate : NSObject <MCSessionDelegate>
-(instancetype)initWithTranslator:(NetworkMCSessionTranslator*) trans;
@end
我的 MCSessionDelegate 实现文件:
#import "NetworkMCSessionDelegate.h"
#import "MusicAppSwift-Swift.h"
@interface NetworkMCSessionDelegate()
@property (nonatomic) NetworkMCSessionTranslator *translator;
@end
@implementation NetworkMCSessionDelegate
/*
create a NetworkMCSessionDelegate and pass in a swift translator
*/
-(instancetype)initWithTranslator:(NetworkMCSessionTranslator*) trans{
if(self = [super init]){
self.translator = trans;
}
return self;
}
/*
Indicates that an NSData object has been received from a nearby peer.
*/
- (void)session:(MCSession *)session didReceiveData:(NSData *)data fromPeer:(MCPeerID *)peerID{
[self.translator networkSession:session didReceive:data fromPeer:peerID];
}
/*
Indicates that the local peer began receiving a resource from a nearby peer.
*/
- (void)session:(MCSession *)session didStartReceivingResourceWithName:(NSString *)resourceName fromPeer:(MCPeerID *)peerID withProgress:(NSProgress *)progres{
[self.translator networkSession:session didStartReceivingResourceWithName:resourceName fromPeer:peerID with:progres];
}
/*
Indicates that the local peer finished receiving a resource from a nearby peer.
*/
- (void)session:(MCSession *)session didFinishReceivingResourceWithName:(NSString *)resourceName fromPeer:(MCPeerID *)peerID atURL:(NSURL *)localURL withError:(NSError *)error{
[self.translator networkSession:session didFinishReceivingResourceWithName:resourceName fromPeer:peerID at:localURL withError:error];
}
/*
Called when the state of a nearby peer changes.
*/
- (void)session:(MCSession *)session peer:(MCPeerID *)peerID didChangeState:(MCSessionState)state{
[self.translator networkSession:session peer:peerID didChange:state];
}
/*
Called when a nearby peer opens a byte stream connection to the local peer.
*/
- (void)session:(MCSession *)session didReceiveStream:(NSInputStream *)stream withName:(NSString *)streamName fromPeer:(MCPeerID *)peerID{
// not expecting to see any of this
}
@end
我的 Swift 文件收集来自代理的流量:
import Foundation
import MultipeerConnectivity
@objc class NetworkMCSessionTranslator: NSObject{
public func networkSession(_ session: MCSession, peer peerID: MCPeerID, didChange state: MCSessionState){
...
}
public func networkSession(_ session: MCSession, didReceive data: Data, fromPeer peerID: MCPeerID){
...
}
public func networkSession(_ session: MCSession, didFinishReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL?, withError error: Error?){
// !!! Notice localURL is now an OPTIONAL !!!
}
public func networkSession(_ session: MCSession, didStartReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, with progress: Progress){
...
}
}
祝你好运!
只是让人们知道这个错误已经随着 xcode 9 beta 1 的发布得到修复。根据文档和我的测试,at localurl 参数现在按预期是可选的:https://developer.apple.com/documentation/multipeerconnectivity/mcsessiondelegate/1406984-session
只需更新到 xcode9 的最新版本即可解决此问题。
在 iOS 7 和 xcode 8 及更高版本中,将 localURL 更改为可选的函数
使用这个功能
func session(_ session: MCSession, didFinishReceivingResourceWithName
resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL?,
withError error: Error?) {
if (localURL != nil) {
//enter code here
}
}
而不是
func session(_ session: MCSession, didFinishReceivingResourceWithName
resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL,
withError error: Error?) {
}
所以,这个功能解决了我的崩溃问题。 xcode 8 如果有警告,忽略。