问题描述
我恢复了我的问题:我想进行网络服务调用(三个调用),对接收到的数据执行一些操作,最后更新UI(许多textview)。 我使用Retrofit2和rxjava2来执行Web服务调用,特别是Observable.zip()对这三个Web服务进行一次调用,以便在执行某些计算之前获得所有这三个数据。 webservice调用工作良好,但是我注意到并不是我的计算方法(在onComplete中)中的所有指令都被调用,仅第一个textView.setText()被调用,之后的所有内容均不执行。即使仅使用Log.i填充方法,也不会全部调用。当我将方法放在onNext()上时,也会发生同样的事情。不会引发任何错误。我做错了什么? 这是我的代码:
// Web服务
@GET("url 1")
Observable<List<Data1>> getData1();
@GET("url 2")
Observable<List<Data2>> getData2();
@GET("url 3")
Observable<List<Data3>> getData3();
// MainActivity
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//getTextViews
getData();
}
private void getData(){
Retrofit retrofit = Retrofit.Builder()
.baseUrl("apiURL")
.addCallAdapterFactory(RxJava2CallAdapterFactory.createWithScheduler(Schedulers.io()))
.client(httpClient.build())
.addConverterFactory(GsonConverterFactory.create(gson))
.build();
WebService service = retrofit.create(WebService.class);
Observable.zip(service.getData1(),service.getData2(),service.getData3(),new Function3<List<Data1>,List<Data2>,List<Data3>,MergedResponse>() {
int i;
@Override
public MergedResponse apply(List<Data1> data1,List<Data2> data2,List<Data3> data3) throws Exception {
return new MergedResponse(data1,data2,data3);
}
}).subscribe(new Observer<MergedResponse>() {
@Override
public void onSubscribe(disposable d) {
}
@Override
public void onNext(MergedResponse mergedResponse) {
// dispatch my merged data
}
@Override
public void onError(Throwable e) {
}
@Override
public void onComplete() {
computeReceivedData();
}
});
}
private void computeReceivedData(){
// List of expressions to compute data and update the view (textview)
...
textView1.setText()
textView2.setText()
...
// Only the first expression textView1.setText() here is called.If I put it at the begining of the method,it stop after executing it
}
retrofit和rxjava2 vresion
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter-gson:2.9.0'
implementation 'com.squareup.okhttp3:logging-interceptor:3.9.0'
implementation "com.squareup.retrofit2:adapter-rxjava2:2.9.0"
implementation 'io.reactivex.rxjava2:rxjava:2.2.2'
implementation 'io.reactivex.rxjava2:rxandroid:2.1.0'
implementation 'androidx.appcompat:appcompat:1.2.0'
implementation 'androidx.legacy:legacy-support-v13:1.0.0'
解决方法
我认为,您正在尝试使用MainThread从后台线程更新UI。
因此您必须在代码中添加以下行
.subscribeOn(Schedulers.io()) // will do operation on background thread
.observeOn(AndroidSchedulers.mainThread()) // will notify UI for changes
例如。
Observable.zip(service.getData1(),service.getData2(),service.getData3(),new Function3<List<Data1>,List<Data2>,List<Data3>,MergedResponse>() {
int i;
@Override
public MergedResponse apply(List<Data1> data1,List<Data2> data2,List<Data3> data3) throws Exception {
return new MergedResponse(data1,data2,data3);
}
}).subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Observer<MergedResponse>() {
@Override
public void onSubscribe(Disposable d) {
}
@Override
public void onNext(MergedResponse mergedResponse) {
// dispatch my merged data
}
@Override
public void onError(Throwable e) {
}
@Override
public void onComplete() {
computeReceivedData();
}
});
,
我认为该错误不在onComplete()方法中。 检查computeReceivedData()方法以确保其正常工作,然后调试textView2.setText()以查找错误。 另外,请检查rxjava中zip方法返回的MergedResponse对象。
,谢谢@yousef,我检查了textView.setText()调用并发现此错误:
Unable to evaluate the expression "totalJ1.setText(""value");" : Method threw 'android.view.ViewRootImpl$CalledFromWrongThreadException' exception.
为解决此问题,我强制调用在onComplete()的mainThread中执行:
MainActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
computeReceivedData();
}
});
一切正常。