gpt4 book ai didi

ios - 如果不在 ViewController 中,则估计不会调用 didChangeAuthorizationStatus

转载 作者:行者123 更新时间:2023-11-28 06:53:54 24 4
gpt4 key购买 nike

我想在方法 didChangeAuthorizationStatus 中调用 startMonitoringForRegion 并且一切正常,主要是因为代码是从 Estimote 示例中复制的。此代码在 ViewController 中。当我使用类似的代码但放置在单独的 Controller 中时会出现问题。

这段代码工作正常,方法 didChangeAuthorizationStatus 每次都会调用 didStartMonitoringForRegion,所以我打开应用程序或 changeStatus。所以成功。ViewController.m

- (void)viewDidAppear:(BOOL)animated{
[super viewDidAppear:animated];
self.region = [[CLBeaconRegion alloc] initWithProximityUUID:[[NSUUID alloc] initWithUUIDString:@"myUUID"]
identifier:@"EstimoteSampleRegion"];
self.beaconManager = [[ESTBeaconManager alloc] init];
self.beaconManager.delegate = self;

[self startRangingBeacons];
}

-(void)startRangingBeacons
{
if ([ESTBeaconManager authorizationStatus] == kCLAuthorizationStatusNotDetermined)
{
[self.beaconManager requestAlwaysAuthorization];
}
else if([ESTBeaconManager authorizationStatus] == kCLAuthorizationStatusDenied)
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Location Access Denied"
message:@"You have denied access to location services. Change this in app settings."
delegate:nil
cancelButtonTitle:@"OK"
otherButtonTitles: nil];
[alert show];
}
else if([ESTBeaconManager authorizationStatus] == kCLAuthorizationStatusRestricted)
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Location Not Available"
message:@"You have no access to location services."
delegate:nil
cancelButtonTitle:@"OK"
otherButtonTitles: nil];
[alert show];
}
}

- (void)beaconManager:(id)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
if (status != kCLAuthorizationStatusNotDetermined && status != kCLAuthorizationStatusDenied )
{
[self.beaconManager startMonitoringForRegion:self.region];
}
}

- (void)beaconManager:(id)manager didStartMonitoringForRegion:(CLBeaconRegion *)region{
NSLog(@"didStartMonitoringForRegion %@",region);
}

此代码部分有效。 BeaconManager 中的方法 requestAlwaysAuthorization 被调用一次,但随后 didChangeAuthorizationStatus 甚至在我手动调用 startMonitoringForRegion 然后方法时也没有didStartMonitoringForRegion 未被调用。

分离的BeaconManager

ViewController.m

- (void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
self.region = [[CLBeaconRegion alloc] initWithProximityUUID:[[NSUUID alloc] initWithUUIDString:@"myUUID"]
identifier:@"EstimoteSampleRegion"];
self.beaconManager = [[BeaconManager alloc]initWithRegion:self.region];
[self.beaconManager startMonitoring];

}

BeaconManager.swift

class BeaconManager: NSObject,ESTBeaconManagerDelegate {
private lazy var secureBeaconManager:ESTSecureBeaconManager = {
let beaconManagerForReturn = ESTSecureBeaconManager()
beaconManagerForReturn.delegate = self
return beaconManagerForReturn
}()
private lazy var beaconsArray = [CLBeacon]()

var region:CLBeaconRegion!
var delegate:BeaconDelegate?

init(region:CLBeaconRegion) {
super.init()
self.region = region
}

/**
Method which check AuthorizationStatus for application, and will start monitoring if status is equal .AuthorizedAlways
*/
func startMonitoring() {
setupLocalizationAuthorization()
}

private func setupLocalizationAuthorization(){
let status = ESTSecureBeaconManager.authorizationStatus()
if (status == .NotDetermined){
secureBeaconManager.requestAlwaysAuthorization()
}

if (status == .Denied){
delegate?.beaconManager(self, locationAutorizationFail: .Denied)
}

if (status == .Restricted){
delegate?.beaconManager(self, locationAutorizationFail: .Restricted)
}
}


// MARK: - BeaconManager

func beaconManager(manager: AnyObject, didChangeAuthorizationStatus status: CLAuthorizationStatus) {
if (status != .NotDetermined && status != .Denied){
secureBeaconManager.startMonitoringForRegion(region)
}
}
func beaconManager(manager: AnyObject, didStartMonitoringForRegion region: CLBeaconRegion) {
print("didStartMonitoring for \(region)")
}
}

问题

为什么会这样?是线程问题?我不知道这里出了什么问题。从我可以看出这两个代码是相似的,但说明放在不同的地方。无论我使用 Swift 还是 Obj-c(已经尝试过)都没有关系。

最佳答案

分析第一次启动应用时发生了什么:

您实例化您的 BeaconManager 并调用 startMonitoring,后者又调用 setupLocalizationAuthorization。在那里,您检查应用程序的授权状态,如果它是 NotDetermined,则实例化 secureBeaconManager(因为您将其声明为惰性属性),分配委托(delegate)(通过您的惰性实例代码),并调用 requestAlwaysAuthorization。这会提示用户允许该应用程序访问位置服务,如果他们同意,iOS 将调用 didChangeAuthorizationStatus 并使用新的 Always 状态,您终于开始了监控。如您所述,此流程按预期工作。

但是请考虑应用已经拥有Always 授权的流程:

您实例化您的 BeaconManager 并调用 startMonitoring,后者又调用 setupLocalizationAuthorization。在那里,您可以检查应用程序的授权状态。但是对于 Always 状态,您没有 if 子句……因此 BeaconManager 就此结束。

解决方案:将 Always 状态的 if 子句附加到您的 setupLocalizationAuthorization:

if status == .AuthorizedAlways {
secureBeaconManager.startMonitoringForRegion(region)
}

让我们最后回顾一下当您实现建议的解决方案时会发生什么:

您实例化您的 BeaconManager 并调用 startMonitoring,后者又调用 setupLocalizationAuthorization。在那里,您可以检查应用程序的授权状态。如果它是 Always,您将延迟实例化 secureBeaconManager,然后开始监视。之前的流程(对于 NotDetermined)保持不变。

关于ios - 如果不在 ViewController 中,则估计不会调用 didChangeAuthorizationStatus,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34224351/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com