繁体   English   中英

改造 - android.os.NetworkOnMainThreadException

[英]Retrofit - android.os.NetworkOnMainThreadException

我正在使用Retrofit 2来获取json并将其解析为POJO。 我的目的是获得该对象的一个​​值。

compile 'com.squareup.retrofit2:retrofit:2.0.0-beta4'
compile 'com.squareup.retrofit2:converter-gson:2.0.0-beta4'

我的REST客户端:

public interface MyClient {

    @GET("/part1/part2")
    Call<MyItem> getMyItem(@Query("param1") String param1,
                                                 @Query("param2") String param2,
                                                 @Query("param3") String param3);

}

在这里,我找到了创建服务的绝佳工具:

public class ServiceGenerator {

    public static final String API_BASE_URL = "http://my.api.com";

    private static OkHttpClient.Builder httpClient = new OkHttpClient.Builder();

    private static Retrofit.Builder builder =
            new Retrofit.Builder()
                    .baseUrl(API_BASE_URL)
                    .addConverterFactory(GsonConverterFactory.create());

    public static <S> S createService(Class<S> serviceClass) {
        Retrofit retrofit = builder.client(httpClient.build()).build();
        return retrofit.create(serviceClass);
    }
}

然后我使用服务生成器类创建新服务:

MyClient api = ServiceGenerator.createService(MyClient.class);
        Call<MyItem> call = api.getMyItem(param1, param2, param3);
        MyItem myItem= null;
        try {
            myItem= call.execute().body();
            Log.d("MyTag", myItem.getValue());
        } catch (IOException e) {
            e.printStackTrace();
        }

当我尝试运行此代码时,我收到此错误:

android.os.NetworkOnMainThreadException位于java.net.InetAddress.getAllByNameImpl的java.net.InetAddress.lookupHostByName(InetAddress.java:418)上的android.os.StrictMode $ AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1147)(InetAddress.java: 252)在okhttp3.Dns的java.net.InetAddress.getAllByName(InetAddress.java:215),在okhttp3的okhttp3.internal.http.RouteSelector.resetNextInetSocketAddress(RouteSelector.java:173)$ 1.lookup(Dns.java:39) .internal.http.RouteSelector.nextProxy(RouteSelector.java:139)位于okhttp3.internal.http.RouteSelector.next(RouteSelector.java:81)的okhttp3.internal.http.StreamAllocation.findConnection(StreamAllocation.java:174)at at okhttp3.internal.http.StreamAllocation.findHealthyConnection(StreamAllocation.java:127)at okhttp3.internal.http.StreamAllocation.newStream(StreamAllocation.java:97)at okhttp3.internal.http.HttpEngine.connect(HttpEngine.java:289) at okhttp3.internal.http.HttpEngine.sendRequest(HttpEngine.java:241)at okhttp3.RealCall.getResponse(RealCall) .java:240)okhttp3.RealCall $ ApplicationInterceptorChain.proceed(RealCall.java:198)at okhttp3.RealCall.getResponseWithInterceptorChain(RealCall.java:160)at okhttp3.RealCall.execute(RealCall.java:57)at retrofit2.OkHttpCall .execute(OkHttpCall.java:177)在uof.cp.ox上的uc.cp.ox.data.MyRepository.getMyItem(MyRepository.java:31)上的retrofit2.ExecutorCallAdapterFactory $ ExecutorCallbackCall.execute(ExecutorCallAdapterFactory.java:87)。 presenters.MyPresenter.do(MyPresenter.java:30)at uz.cp.ox.activities.MyActivity.onClick(MyActivity.java:52)at android.view.View.performClick(View.java:4756)at android.view 。在android.os.Handler.dispatchMessage(Handler.java:95)的android.os.Looper上的android.os.Handler.handleCallback(Handler.java:739)上查看$ PerformClick.run(View.java:19749)。在java.lang.reflect.Method.invoke的java.lang.reflect.Method.invoke(Native Method)的android.app.ActivityThread.main(ActivityThread.java:5221)上循环(Looper.java:135)(方法。 java:372)在com.android.internal.os.ZygoteIn 在com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)的$ MethodAndArgsCaller.run(ZygoteInit.java:899)

我认为Retrofit会自动在后台线程中完成工作。 或者我误解了一些事情。 这种情况有什么问题以及如何解决?

我认为Retrofit会自动在后台线程中完成工作。

如果您使用异步版本 - enqueue 您正在使用在调用线程上运行的同步版本。

这样称呼它:

MyClient api = ServiceGenerator.createService(MyClient.class);
Call<MyItem> call = api.getMyItem(param1, param2, param3);
call.enqueue(new Callback<MyItem>() {
    @Override
    public void onResponse(Call<MyItem> call, Response<MyItem> response) {
        MyItem myItem=response.body();
    }

    @Override
    public void onFailure(Call<MyItem> call, Throwable t) {
        //Handle failure
    }
});

在onResponse()中,使用response.body()来获取响应,例如:
MyItem myItem=response.body();

编辑:修复onResponse()和onFailure()签名并将示例添加到onRespnose()。

你在主线程上调用webservice,这就是为什么你得到这个错误' android.os.NetworkOnMainThreadException '。

您可以使用NightSkyDev回答的上述编码来获取您想要的数据。

在改造方法执行的两种需要地方- 排队 -适用于后台线程通过改装自动处理。 执行 - 手动在UI线程上工作,需要放入后台线程或获取异步任务的帮助。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM