ios - 购买应用程序后,应用程序在启动时崩溃。产品标识符=无?

标签 ios in-app-purchase null storekit

我有一些用户报告说,在尝试应用内购买后,应用现在在启动时崩溃。我已要求他们删除并重新安装未运行的应用程序,并试图要求他们进入飞行模式以停止任何未运行的网络通信。

我无法在我的设备上以任何方式复制错误,我的应用程序内购买在沙盒和生产模式下运行良好。我的想法是,他们的交易以某种方式收到了一个 nil productIdentifier,这导致了启动崩溃,但我不确定在应用程序启动时调用了哪些交易观察器方法,我可以为他们解决这个问题。

是否有某种方法可以“清除”事务队列或以其他方式在启动时测试 nil productidentifiers 并允许这些用户至少再次运行该应用程序?我已经使用下面的代码进行了数百次应用内购买,而这最近才开始发生。应用程序启动时会调用哪些辅助方法?

在 AppDelegate.m 中

[[SKPaymentQueue defaultQueue] addTransactionObserver:[MovieClockIAPHelper sharedHelper]];

在应用程序帮助代码中:

- (id)initWithProductIdentifiers:(NSSet *)productIdentifiers {
    if ((self = [super init])) {

        // Store product identifiers
        _productIdentifiers = [productIdentifiers retain];

        // Check for previously purchased products

        NSMutableSet * purchasedProducts = [NSMutableSet set];
        for (NSString * productIdentifier in _productIdentifiers) {

            BOOL productPurchased = [[NSUserDefaults standardUserDefaults] boolForKey:productIdentifier];

            if (productPurchased) {
                [purchasedProducts addObject:productIdentifier];
                NSLog(@"Previously purchased: %@", productIdentifier);
            }
            else{
            NSLog(@"Not purchased: %@", productIdentifier);
            }
        }
        self.purchasedProducts = purchasedProducts;

    }
    return self;
}

- (void)requestProducts {

    self.request = [[[SKProductsRequest alloc] initWithProductIdentifiers:_productIdentifiers] autorelease];
    _request.delegate = self;
    [_request start];

}

- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response {

    NSLog(@"Received products results...");   
    self.products = response.products;
    self.request = nil;    

    [[NSNotificationCenter defaultCenter] postNotificationName:kProductsLoadedNotification object:_products];    
}


- (void)restoreCompletedTransactions {
    [[SKPaymentQueue defaultQueue] restoreCompletedTransactions];
}

- (void)provideContent:(NSString *)productIdentifier {

    NSLog(@"Toggling flag for: %@", productIdentifier);
    [[NSUserDefaults standardUserDefaults] setBool:TRUE forKey:productIdentifier];
    [[NSUserDefaults standardUserDefaults] synchronize];
    [_purchasedProducts addObject:productIdentifier];

    [[NSNotificationCenter defaultCenter] postNotificationName:kProductPurchasedNotification object:productIdentifier];

}

- (void)completeTransaction:(SKPaymentTransaction *)transaction {

    NSLog(@"completeTransaction...");

    [self recordTransaction: transaction];
    [self provideContent: transaction.payment.productIdentifier];
    [[SKPaymentQueue defaultQueue] finishTransaction: transaction];

}

- (void)restoreTransaction:(SKPaymentTransaction *)transaction {

    NSLog(@"restoreTransaction...");


    [self recordTransaction: transaction];
    [self provideContent: transaction.originalTransaction.payment.productIdentifier];
    [[SKPaymentQueue defaultQueue] finishTransaction: transaction];

}

- (void)failedTransaction:(SKPaymentTransaction *)transaction {

    if (transaction.error.code != SKErrorPaymentCancelled)
    {
        NSLog(@"Transaction error: %@", transaction.error.localizedDescription);
    }

    [[NSNotificationCenter defaultCenter] postNotificationName:kProductPurchaseFailedNotification object:transaction];

    [[SKPaymentQueue defaultQueue] finishTransaction: transaction];

}

- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions
{
    NSLog(@"in the payment queue");

    for (SKPaymentTransaction *transaction in transactions)
    {

        switch (transaction.transactionState)
        {
            case SKPaymentTransactionStatePurchased:
                [self completeTransaction:transaction];
                break;
            case SKPaymentTransactionStateFailed:
                [self failedTransaction:transaction];
                break;
            case SKPaymentTransactionStateRestored:
                [self restoreTransaction:transaction];
            default:
                break;
        }
        }

}

-(void)buyProduct:(SKProduct *)product
{
    NSLog(@"In buyproduct Buying %@...", product.productIdentifier);

    SKPayment *payment = [SKPayment paymentWithProduct:product];

    [[SKPaymentQueue defaultQueue] addPayment:payment];
}

- (void)dealloc
{
    [_productIdentifiers release];
    _productIdentifiers = nil;
    [_products release];
    _products = nil;
    [_purchasedProducts release];
    _purchasedProducts = nil;
    [_request release];
    _request = nil;
    [super dealloc];
}

@end

最佳答案

当交易在 SKPaymentTransactionStateRestored 中时,我遇到了类似的问题。我的测试表明这可能是 IOS 7.0.3 的问题,但我无法验证这一点。

StoreKit 会保存一份您的应用程序必须完成的持久事务列表。正如您所注意到的,交易将在每次启动时报告,直到它完成。

我们实现的解决方案是在使用前从入口点检查产品标识符是否为 nil:

- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions

即使我们收到一个产品标识符为零的交易,我们也能够成功调用 finishTransaction

希望对您有所帮助。

关于ios - 购买应用程序后,应用程序在启动时崩溃。产品标识符=无?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19817130/

相关文章:

ios - 存储为变量的 Swift 闭包会产生内存泄漏

ios - 如何从 UITabBarItem 获取 RootViewController

ios - 与在后台运行应用程序的一台设备进行多点连接

c# - json.net : why is a null JToken replaced with a non-null value? 中的奇怪行为

java - FontMetrics 生成 NullPointerException

java - 如何使用 null 对集合进行排序并在之后反转列表?

ios - 如何使用 PHCachingImageManager().stopCachingImages

iphone - 是否可以在两个应用程序之间共享一项应用程序内购买?

ios - 处理应用购买 - 从我自己的服务器获取产品标识符

swift - 尝试使用 StoreKit 恢复应用内购买时未收到预期的委托(delegate)调用