计费 API v3 IabHelper NullPointerException

Posted

技术标签:

【中文标题】计费 API v3 IabHelper NullPointerException【英文标题】:Billing API v3 IabHelper NullPointerException 【发布时间】:2013-03-05 14:03:41 【问题描述】:

编辑 4/15: 在 IabHelper 中捕获空指针似乎已经解决了这个问题。我不再看到抛出的异常,我将接受这个作为答案。


编辑 4/04: 稍微深入一点。有用于处理 queryPurchases 方法的 RemoteExceptions 和 JSONExceptions 的 try catch 块,但没有 NullPointerException 处理。我要尝试的是包括 NullPointer 异常处理,因此 IabHelper 在尝试查询 SkuDetails 时看起来像这样:

    catch (NullPointerException e) 
        throw new IabException(IABHELPER_UNKNOWN_ERROR, "NullPointer while refreshing inventory.", e);
    

我刚刚提交了一个错误:

https://code.google.com/p/marketbilling/issues/detail?id=114


编辑 3/25: 好吧,仍然收到这个崩溃...现在它发生在尝试从 IabHelper 摘录的以下第 3 行获取上下文时:

int queryPurchases(Inventory inv, String itemType) throws JSONException, RemoteException 
    logDebug("Querying owned items, item type: " + itemType);
    logDebug("Package name: " + mContext.getPackageName());

这令人沮丧,因为在我的清单中,我总是使用应用程序的完整路径名作为“名称”。

示例“com.myappname.blah.ClassName”

我也尝试过将这个、MyClass.this、getApplicationContext() 传递给 mHelper。然而,它们都从野外设备随机产生相同的 NullPointer 结果。我还在清单中尝试了 name=".MyClass" 。这是目前的样子:

mHelper = new IabHelper(MyClass.this, myKey);

编辑 3/18/13:我仍然收到异常,即使在 3/17 部署了新的 IabHelper 版本。

我开始在这里看到一种模式,即崩溃都是在执行 mContext.getPackageName() 时尝试获取上下文时发生的。我很好奇为什么这在我所有的测试设备上都有效,而且我无法重现这个崩溃,而且似乎只在少数设备上。

这是新的崩溃:

java.lang.NullPointerException
    at com.myapp.util.IabHelper.queryPurchases(SourceFile:836)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:558)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:522)
    at com.myapp.util.IabHelper$2.run(SourceFile:617)
    at java.lang.Thread.run(Thread.java:1019)

由 IabHelper 引起...

line 836: logDebug("Package name: " + mContext.getPackageName());

edit 2013 年 3 月 17 日:我看到过去几个月发布了许多错误修复,我将尝试此处提供的最新代码,看看是否能解决问题:

https://code.google.com/p/marketbilling/source/browse/v3/src/com/example/android/trivialdrivesample/util


在我的一个应用程序中,我使用了计费 API 和其中包含的样板代码。

我正在使用截至 2013 年 3 月 16 日通过 SDK 管理器提供的最新版本的计费 API。

在我的活动中,我使用以下方式查询库存:

final List<String> skuList = new ArrayList<String>();
skuList.add("sku1");
skuList.add("sku2");
skuList.add("sku3");
if (skuList != null) 
    if (skuList.size() > 0) 
        try 
            mHelper.queryInventoryAsync(true, skuList, mGotInventoryListener);
         catch (Exception e)  
            ACRA.getErrorReporter().handleException(e);
        
    

我从以下设备的 IabHelper 类中收到多个 NullPointerException 报告。我无法重现该问题,也找不到有关这些崩溃的任何信息,这就是我发布此问题的原因。

我在计费 API 的“面向开发人员”部分(包括在 onQueryInventoryFinished 中)对 null 和 try/catch 块进行了无数其他检查,所以我知道这个异常不是从“我的代码”中抛出的(因为我没有从我的任何应用程序的类中捕获崩溃),而是从 IabHelper 本身中抛出。除了这个推荐的修复之外,我没有修改 IabHelper:https://***.com/a/14737699

崩溃 #1 Galaxy Nexus

java.lang.NullPointerException
    at com.myapp.util.IabHelper.querySkuDetails(SourceFile:802)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:471)
    at com.myapp.util.IabHelper$2.run(SourceFile:521)
    at java.lang.Thread.run(Thread.java:856)

由 IabHelper 引起...

line 802: Bundle skuDetails = mService.getSkuDetails(3, mContext.getPackageName(), ITEM_TYPE_INAPP, querySkus);    

崩溃 #2 三星 GT-S5570L

java.lang.NullPointerException
    at com.myapp.util.IabHelper.queryPurchases(SourceFile:735)
    at com.myapp.util.IabHelper.queryInventory(SourceFile:465)
    at com.myapp.util.IabHelper$2.run(SourceFile:521)
    at java.lang.Thread.run(Thread.java:1019)

由 IabHelper 引起...

line 735: Bundle ownedItems = mService.getPurchases(3, mContext.getPackageName(), ITEM_TYPE_INAPP, continueToken);

【问题讨论】:

我收到以下崩溃日志。 java.lang.NullPointerException at com.myapp.util.IabHelper.queryPurchases(SourceFile:735) at com.myapp.util.IabHelper.queryInventory(SourceFile:465) at com.myapp.util.IabHelper$2.run(SourceFile:521) at java.lang.Thread.run(Thread.java:1019) 有没有人弄清楚可能导致这种情况的原因。我正在检查是否在查询购买之前建立了连接。 我仍然看到同样的事情,即使是几周前的最新代码。这是非常令人沮丧的,但它似乎与上下文有关。我尝试了将上下文传递给 IabHelper 的不同方法,但它们似乎都崩溃了。在我的最新版本中,我注释掉了引发异常的日志行,但现在它只是在不同的地方崩溃了。 @glo 我刚刚提交了一个错误:code.google.com/p/marketbilling/issues/detail?id=114 【参考方案1】:

编辑 4/15: 在 IabHelper 中捕获空指针似乎已经解决了这个问题。我不再看到抛出的异常,我将接受这个作为答案。


编辑 4/04: 稍微深入一点。有用于处理 queryPurchases 方法的 RemoteExceptions 和 JSONExceptions 的 try catch 块,但没有 NullPointerException 处理。我要尝试的是包括 NullPointer 异常处理,因此 IabHelper 在尝试查询 SkuDetails 时看起来像这样:

    catch (NullPointerException e) 
        throw new IabException(IABHELPER_UNKNOWN_ERROR, "NullPointer while refreshing inventory.", e);
    

我刚刚提交了一个错误:

https://code.google.com/p/marketbilling/issues/detail?id=114


改变

        if (querySkuDetails) 
            r = querySkuDetails(ITEM_TYPE_INAPP, inv, moreItemSkus);
            if (r != BILLING_RESPONSE_RESULT_OK) 
                throw new IabException(r, "Error refreshing inventory (querying prices of items).");
            
        

        if (querySkuDetails) 
            try 
                r = querySkuDetails(ITEM_TYPE_INAPP, inv, moreItemSkus);
                if (r != BILLING_RESPONSE_RESULT_OK) 
                    throw new IabException(r, "Error refreshing inventory (querying prices of items).");
                
             catch (NullPointerException e) 
                throw new IabException(IABHELPER_UNKNOWN_ERROR, "NPE while refreshing inventory.", e);
            
        

改变

            if (querySkuDetails) 
                r = querySkuDetails(ITEM_TYPE_SUBS, inv, moreSubsSkus);
                if (r != BILLING_RESPONSE_RESULT_OK) 
                    throw new IabException(r, "Error refreshing inventory (querying prices of subscriptions).");
                
            

            if (querySkuDetails) 
                try 
                    r = querySkuDetails(ITEM_TYPE_SUBS, inv, moreSubsSkus);
                    if (r != BILLING_RESPONSE_RESULT_OK) 
                        throw new IabException(r, "Error refreshing inventory (querying prices of subscriptions).");
                    
                 catch (NullPointerException e) 
                    throw new IabException(IABHELPER_UNKNOWN_ERROR, "NPE while refreshing inventory.", e);
                
            

【讨论】:

提交错误可能没有帮助,因为该 repo 已被弃用。 我应该把这个抓到哪里? 根据我对代码的阅读,捕获错误应该只是消除崩溃,而不是修复在这种情况下无法购买的源问题;那是对的吗?如果是这样,如何解决源问题(确保必要的对象在内存中)?【参考方案2】:

您可能正在使用异步操作。如果您使用 ...async 方法,当前的 IabHelper 是不安全的。问题是,在任何时候运行异步操作的时候,都可以在主线程上调用 dispose。在这种情况下,您将获得 NullPointerExceptions 和 IllegalStateExceptions。

这是修复它的补丁:

Index: src/com/evotegra/aCoDriver/iabUtil/IabHelper.java
===================================================================
--- src/com/evotegra/aCoDriver/iabUtil/IabHelper.java   (revision 1162)
+++ src/com/evotegra/aCoDriver/iabUtil/IabHelper.java   (working copy)
@@ -86,7 +86,10 @@

     // Is an asynchronous operation in progress?
     // (only one at a time can be in progress)
-    boolean mAsyncInProgress = false;
+    volatile boolean mAsyncInProgress = false;
+    
+    // is set to true if dispose is called while a thread is running. Allows graceful shutdown
+    volatile boolean mDisposeRequested = false;

     // (for logging/debugging)
     // if mAsyncInProgress == true, what asynchronous operation is in progress?
@@ -285,6 +288,12 @@
      * disposed of, it can't be used again.
      */
     public void dispose() 
+       // do not dispose while an async Thread is running. Will cause all kinds of exceptions.
+       // In this case dispose must be called from thread after setting mAsyncInProgress to true
+       if (mAsyncInProgress) 
+           mDisposeRequested = true;
+           return;
+       
         logDebug("Disposing.");
         mSetupDone = false;
         if (mServiceConn != null) 
@@ -827,6 +836,7 @@
         logDebug("Ending async operation: " + mAsyncOperation);
         mAsyncOperation = "";
         mAsyncInProgress = false;
+        if (mDisposeRequested) IabHelper.this.dispose();
     

或者在这里下载补丁。 http://code.google.com/p/marketbilling/issues/detail?id=139&thanks=139&ts=1375614409

【讨论】:

我使用单线程执行器来安排随后的所有异步操作。也是dispose方法。另请参阅***.com/questions/16222015/…。 不幸的是,IabHelper 还有其他几个缺陷。一般来说,它还没有准备好生产,并且 imo 应该在任何公共应用程序中使用。我通过 Google trevorjohns@google.com 通知了代码的所有者,我什至收到了回复说“开发人员告诉我它在当前版本中已修复”,但实际上并非如此。也许你给他写一封电子邮件,请求他们修复它。 这个补丁是否修复了 this 错误? 它修复了 IabHelper 中 NullPointerException 的来源。但是这段代码有很多bug。 如何在 Android 项目中使用这个补丁?谢谢。【参考方案3】:

稍微修改queryPurchases方法的开头,如下所示:

int queryPurchases(Inventory inv, String itemType) throws JSONException, RemoteException 
        // Query purchases
        //logDebug("Querying owned items, item type: " + itemType);
       //logDebug("Package name: " + mContext.getPackageName());
       boolean verificationFailed = false;
       String continueToken = null;

        do 
//            logDebug("Calling getPurchases with continuation token: " + continueToken);
            if(mDisposed || mService==null) return IABHELPER_UNKNOWN_ERROR;
            Bundle ownedItems = mService.getPurchases(3, mContext.getPackageName(),
                    itemType, continueToken);

感谢sebastie 指出原因。

【讨论】:

【参考方案4】:

tmantheypatch也需要

mDisposeRequested = false;

处理后

【讨论】:

【参考方案5】:

如果您在模拟器上遇到此错误,这可能是一件非常简单的事情,发生在一半以上的情况下。

检查您使用的是 Google API SDK 而不是常规 SDK!!!

【讨论】:

【参考方案6】:

IabHelper 已过时,已被 BillingClient 取代。 见https://developer.android.com/google/play/billing/billing_library.html

【讨论】:

它并没有过时,而是一种替代方案。如前所述,“为了简化应用内计费 API 的开发,您可以使用 Play 计费库。这个库是一个应用内计费客户端,作为 Android 接口定义语言文件之上的包装器开发,它与 In-应用计费 API。" @ChallengeAccepted IabHelper 已过时“我们没有更多计划更新使用 IABHelper 的 Trivial Drive v1。” github.com/googlesamples/android-play-billing/issues/… @ChallengeAccepted "但是,不会更新" github.com/googlesamples/android-play-billing/issues/…

以上是关于计费 API v3 IabHelper NullPointerException的主要内容,如果未能解决你的问题,请参考以下文章

使用IabHelper时无法绑定InAppBillingService

有没有办法增加 YouTube API v3 配额限制?

IabHelper queryInventory() 返回产品的基本价格

Paypal REST API - 计费协议

PayPal REST API 计费计划和协议(订阅)

PayPal API 如何设置自定义计费频率?