我正在尝试将 MultipeerConnectivity 框架与 Swift 一起使用。我有以下属性:
var peerId: MCPeerID?;
let advertiser: MCNearbyServiceAdvertiser;
let browser: MCNearbyServiceBrowser;
var session: MCSession?
....
在我的 init 方法中,我初始化所有存储的属性,如下所示:
init() {
let defaults = NSUserDefaults.standardUserDefaults();
let dataToShow = defaults.dataForKey("kPeerID");
peerId = NSKeyedUnarchiver.unarchiveObjectWithData(dataToShow) as? MCPeerID;
if !peerId {
peerId = MCPeerID(displayName: UIDevice.currentDevice().name);
let data: NSData = NSKeyedArchiver.archivedDataWithRootObject(peerId);
defaults.setObject(data, forKey: "kPeerID");
defaults.synchronize();
}
advertiser = MCNearbyServiceAdvertiser(peer: peerId, discoveryInfo:nil, serviceType: "stc-classroom");
browser = MCNearbyServiceBrowser(peer: peerId, serviceType: "stc-classroom");
session = MCSession(peer: self.peerId);
super.init();
session!.delegate = self;
advertiser.delegate = self;
browser.delegate = self;
}
现在,正如您所看到的,有相当多的代码行可以为peerId赋值(Apple推荐的方式),所以我认为最好将该逻辑放入计算属性中,因此我用计算属性替换了peerId存储属性,并且初始化方法如下:
var peerId: MCPeerID {
let defaults = NSUserDefaults.standardUserDefaults();
let dataToShow = defaults.dataForKey("kPeerID");
var peer = NSKeyedUnarchiver.unarchiveObjectWithData(dataToShow) as? MCPeerID;
if peer == nil {
peer = MCPeerID(displayName: UIDevice.currentDevice().name);
let data: NSData = NSKeyedArchiver.archivedDataWithRootObject(peer);
defaults.setObject(data, forKey: "kPeerID");
defaults.synchronize();
}
return peer!;
}
现在我的 init 方法更加干净和简短,如下所示:
init() {
advertiser = MCNearbyServiceAdvertiser(peer: peerId, discoveryInfo:nil, serviceType: "stc-classroom");
browser = MCNearbyServiceBrowser(peer: peerId, serviceType: "stc-classroom");
session = MCSession(peer: self.peerId);
super.init();
session!.delegate = self;
advertiser.delegate = self;
browser.delegate = self;
}
但现在编译器抱怨,因为我在浏览器/广告商/会话初始化中访问peerId时尝试访问self。我知道在 Swift 中,所有属性都必须先初始化,然后才能使用它们,并且 self 被隐式调用,因此会出现错误。但显然这意味着我无法在 init 方法中调用计算属性,所以我想知道是否有更好的方法来实现我想要做的事情?我知道我可以为每个计算属性创建存储属性并将其视为 iVar,但这似乎也不是一个优雅的解决方案。
我非常感谢你对这些人的帮助。