由主线程阻止的AdvertisingIdClient getAdvertisingIdInfo

时间:2015-01-15 10:44:00

标签: java android multithreading google-play-services

我试图等待AdvertisingIdClient.getAdvertisingIdInfo(活动)的响应没有成功。在主线程完成之前,此方法永远不会响应。

import android.app.Activity;
import android.os.Bundle;
import android.os.SystemClock;
import android.util.Log;
import com.google.android.gms.ads.identifier.AdvertisingIdClient;
import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.GooglePlayServicesNotAvailableException;
import com.google.android.gms.common.GooglePlayServicesUtil;
import java.io.IOException;


public class MyActivity extends Activity {

    private Activity m_activity = null;
    private AdvertisingIdClient.Info m_info = null;


    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        // start the thread with the getAdvertisingIdInfo()
        startGoogleAdvertisingIdRequest(this);

        // simulate a waiting loop, others app init, ...
        for (int i=0; i<20; i++) {
            SystemClock.sleep(100);
        }

        // get the uuid
        String uuid = getGoogleAdvertisingId();

        // call a method who need the uuid
        Log.i("UUID", "receive uuid: " + uuid);
    }


    public String getGoogleAdvertisingId() {
        String uuid = null;
        if (m_info != null) {
            if (!m_info.isLimitAdTrackingEnabled()) {
                uuid = m_info.getId();
            } else {
                uuid = "another uuid";
            }
        } else {
            uuid = "another uuid";
        }

        return uuid;
    }

    public void startGoogleAdvertisingIdRequest(final Activity activity) {
        m_activity = activity;
        if (GooglePlayServicesUtil.isGooglePlayServicesAvailable(activity) == ConnectionResult.SUCCESS) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    AdvertisingIdClient.Info adInfo = null;
                    try {
                        Log.i("UUID", "before google request");
                        adInfo = AdvertisingIdClient.getAdvertisingIdInfo(activity);
                        Log.i("UUID", "after google request");
                    } catch (IOException e) {
                        Log.w("UUID", "getAdvertisingIdInfo IOException: " + e.getMessage());
                    } catch (GooglePlayServicesNotAvailableException e) {
                        Log.w("UUID", "GooglePlayServicesNotAvailableException: " + e.getMessage());
                    } catch (Exception e) {
                        Log.w("UUID", "GooglePlayServicesException: " + e.getMessage());
                    } finally {
                        finished(adInfo);
                    }
                }
            }).start();
        }
    }

    private void finished(final AdvertisingIdClient.Info adInfo){
        if(adInfo != null){
            m_activity.runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    m_info = adInfo;
                    Log.i("UUID", "runOnUiThread id: " + adInfo.getId());
                }
            });
        }
    }
}

此代码的Logcat

11:29:52.103  30810-30828/com.example.testuuid I/UUID﹕ before google request
11:29:54.107  30810-30810/com.example.testuuid I/UUID﹕ receive uuid: another uuid
11:29:54.127  30810-30828/com.example.testuuid I/UUID﹕ after google request
11:29:54.151  30810-30810/com.example.testuuid I/UUID﹕ runOnUiThread id: d5dc3bfb-4756-490c-8f8e-2bedfb5e827a

相同的logcat,等待时间更长(5s)

11:36:14.215  31413-31436/com.example.testuuid I/UUID﹕ before google request
11:36:19.225  31413-31413/com.example.testuuid I/UUID﹕ receive uuid: another uuid
11:36:19.293  31413-31436/com.example.testuuid I/UUID﹕ after google request
11:36:19.315  31413-31413/com.example.testuuid I/UUID﹕ runOnUiThread id: d5dc3bfb-4756-490c-8f8e-2bedfb5e827a

每次处于另一个线程中的getAdvertisingIdInfo()被主线程阻止。

是什么原因?以及如何做到这一点?

3 个答案:

答案 0 :(得分:0)

在关于AdvertisingIdClient的文献中,它表示不在主线程上使用。它会引发异常。所以如果你把它放进自己的线程中,你很可能会很好。

AdvertisingIdClient reference

答案 1 :(得分:0)

根据文档(here):

  

不幸的是,在这些情况下,使用com.google.android.gms.iid   InstanceID API或用于创建应用程序范围ID的系统功能不是   适当的解决方案,因为可能需要在多个ID之间共享ID   应用。一种替代解决方案是使用广告标识符   可通过getId()从AdvertisingIdClient.Info类获得   方法。您可以使用来创建AdvertisingIdClient.Info对象   getAdvertisingIdInfo(Context)方法并调用getId()方法以   使用标识符。请注意,此方法正在阻止,因此您应该   不从主线程调用它;对此的详细说明   方法在这里可用。

here

  

公共静态AdvertisingIdClient.Info getAdvertisingIdInfo(上下文   上下文)

     

检索用户的广告ID并限制广告跟踪首选项。

     

此方法可能会阻塞,因此无法在主线程中调用   导致ANR。如果这是一个IllegalStateException将被抛出   在主线程上调用。

所以他们说它正在阻止... 您需要将此代码放在后台线程中。

答案 2 :(得分:0)

您所看到的问题似乎没有足够长的时间来完成getAdvertisingIdInfo调用,这是由您等待的方式以及runOnUiThread的工作方式引起的。关键是runOnUiThread将在ui线程当前正在运行的内容(在本例中为onCreate)之后将要运行的代码排队。用于“模拟等待”的sleep调用将让您的后台线程运行并完成其工作,但是设置m_info的最终操作将始终在onCreate完成后排队并执行。

一种解决方案是确保m_info可安全地从多个线程进行访问,并将其简单地分配给后台线程。不需要runOnUiThread。这将删除队列,并允许您的代码以最小的更改来工作。

更好的解决方案是继续使用runOnUiThread并删除用于等待的睡眠。您需要记住,m_info在onCreate中始终为null,但是其他事件可以检查该值是否为非null并根据需要使用它。

https://developer.android.com/reference/android/app/Activity#runOnUiThread(java.lang.Runnable)

在UI线程上运行指定的操作。如果当前线程是UI线程,则立即执行该操作。如果当前线程不是UI线程,则将操作发布到UI线程的事件队列。