Rxjava+Retrofit取消网络请求

一、解决办法
刚开始接触Rxjava的朋友可能不知道怎么取消网络请求。
其实直接调用unsubscribe()就可以了。
下面是验证测试

private OkHttpClient okHttpClient;
private Retrofit retrofit; 

public interface HttpLogin {                                             
    @POST("account/login")                                               
    retrofit2.Call<HashMap<String, Object>> login(@Body Account account);
}        

@Before                                                                                                  
public void setUp() {                                                                                    
    OkHttpClient.Builder builder = new OkHttpClient.Builder();                                           
    HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor(new MyLogger());              
    loggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY);                                      
    builder.addInterceptor(loggingInterceptor);                                                          
    okHttpClient = builder.build();                                                                      

    retrofit = new Retrofit.Builder().client(okHttpClient).baseUrl("http://localhost:8080/campus/")      
            .addCallAdapterFactory(RxJavaCallAdapterFactory.createWithScheduler(Schedulers.newThread())) 
            .addConverterFactory(JacksonConverterFactory.create())                                       
            .build();                                                                                    
}       

 @Test                                                                    
 public void testRxJava() {                                               
     final Subscription subscription = retrofit                           
             .create(HttpLoginRx.class)                                   
             .login(new Account("yincs", "123456"))                       
             .subscribe(new Subscriber<HashMap<String, Object>>() {       

                 public void onCompleted() {                              
                     System.out.println("onCompleted");                   
                 }                                                        

                 public void onError(Throwable arg0) {                    
                     System.out.println("onError  " + arg0.getMessage()); 
                 }                                                        

                 public void onNext(HashMap<String, Object> arg0) {       
                     System.out.println("onNext   " + arg0.get("des"));   
                 }                                                        
             });                                                          
     subscription.unsubscribe();                                          

     try {        
         //因为网络请求在子线程,所以不能让宿主线程过早的结束。                                                        
         Thread.sleep(120000);                                            
     } catch (InterruptedException e1) {                                  
         e1.printStackTrace();                                            
     }                                                                    
 }                                                                     

返回结果:
message:–> POST http://localhost:8080/campus/account/login http/1.1
message:Content-Type: application/json; charset=UTF-8
message:Content-Length: 48
message:
message:{“account”:”yincs”,”passwd”:”123456”,”userid”:0}
message:–> END POST (48-byte body)
message:<– HTTP FAILED: java.io.IOException: Canceled

看最后一行。这是okhttp抛出的异常。
在okhttp3.RealCall.java中可以看到下面这段代码

  @Override public Response execute() throws IOException {
    synchronized (this) {
      if (executed) throw new IllegalStateException("Already Executed");
      executed = true;
    }
    try {
      client.dispatcher().executed(this);
      Response result = getResponseWithInterceptorChain();
      //如果请求取消了就抛出一个该异常
      if (result == null) throw new IOException("Canceled");
      return result;
    } finally {
      client.dispatcher().finished(this);
    }

二、实现原理
在上面创建retrofit的时候,我们传入了一个RxJavaCallAdapterFactory的适配器工厂。
RxJavaCallAdapterFactory.createWithScheduler(Schedulers.newThread())这时传入了一个Scheduler线程控件器。
我们知道Retrofit最后是通过CallAdapter.get方法创建适配器,然后再通过这个适配器.adapt方法创建返回对象(这里也就是创建Observable)
现在看下retrofit2.adapter.rxjava.RxJavaCallAdapterFactory.get()方法

 @Override
  public CallAdapter<?> get(Type returnType, Annotation[] annotations, Retrofit retrofit) {
    Class<?> rawType = getRawType(returnType);
    String canonicalName = rawType.getCanonicalName();
    //isSingle = flase
    boolean isSingle = "rx.Single".equals(canonicalName);
    //isCompletable  = flase
    boolean isCompletable = "rx.Completable".equals(canonicalName);
    if (rawType != Observable.class && !isSingle && !isCompletable) {
      return null;
    }
    if (!isCompletable && !(returnType instanceof ParameterizedType)) {
      String name = isSingle ? "Single" : "Observable";
      throw new IllegalStateException(name + " return type must be parameterized"
          + " as " + name + "<Foo> or " + name + "<? extends Foo>");
    }

    if (isCompletable) {
      // Add Completable-converter wrapper from a separate class. This defers classloading such that
      // regular Observable operation can be leveraged without relying on this unstable RxJava API.
      // Note that this has to be done separately since Completable doesn't have a parametrized
      // type.
      return CompletableHelper.createCallAdapter(scheduler);
    }

    CallAdapter<Observable<?>> callAdapter = getCallAdapter(returnType, scheduler);
    if (isSingle) {
      // Add Single-converter wrapper from a separate class. This defers classloading such that
      // regular Observable operation can be leveraged without relying on this unstable RxJava API.
      return SingleHelper.makeSingle(callAdapter);
    }
    return callAdapter;
  }

上面可以看出是直接返回的getCallAdapter(returnType, scheduler);

  private CallAdapter<Observable<?>> getCallAdapter(Type returnType, Scheduler scheduler) {
    Type observableType = getParameterUpperBound(0, (ParameterizedType) returnType);
    Class<?> rawObservableType = getRawType(observableType);
    //返回参数里的泛型,这里上面的接口返回的泛型类型是一个HashMap
    if (rawObservableType == Response.class) {
      if (!(observableType instanceof ParameterizedType)) {
        throw new IllegalStateException("Response must be parameterized"
            + " as Response<Foo> or Response<? extends Foo>");
      }
      Type responseType = getParameterUpperBound(0, (ParameterizedType) observableType);
      return new ResponseCallAdapter(responseType, scheduler);
    }
    //返回参数里的泛型,这里上面的接口返回的泛型类型是一个HashMap
    if (rawObservableType == Result.class) {
      if (!(observableType instanceof ParameterizedType)) {
        throw new IllegalStateException("Result must be parameterized"
            + " as Result<Foo> or Result<? extends Foo>");
      }
      Type responseType = getParameterUpperBound(0, (ParameterizedType) observableType);
      return new ResultCallAdapter(responseType, scheduler);
    }
    //返回了个SimpleCallAdapter
    return new SimpleCallAdapter(observableType, scheduler);
  }

好,继续看下SimpleCallAdapter类的源码,我们一开始的目标是要找到adapt方法。不多说直接看

  static final class SimpleCallAdapter implements CallAdapter<Observable<?>> {  
   private final Type responseType;                                            
   private final Scheduler scheduler;                                          

   SimpleCallAdapter(Type responseType, Scheduler scheduler) {                 
     this.responseType = responseType;                                         
     this.scheduler = scheduler;                                               
   }                                                                           

   @Override public Type responseType() {                                      
     return responseType;                                                      
   }                                                                           

   @Override public <R> Observable<R> adapt(Call<R> call) {                    
     Observable<R> observable = Observable.create(new CallOnSubscribe<>(call)) 
         //异常处理的一个订阅操作。即如果产生了网络异常之类的,就回调订阅者的onError操作
         .lift(OperatorMapResponseToBodyOrError.<R>instance());                
     if (scheduler != null) {                                                  
       return observable.subscribeOn(scheduler);                               
     }                                                                         
     return observable;                                                        
   }                                                                           
 }                                                                             

上面可以看到Observable观察了了一个CallOnSubscribe订阅者。这里需要对Rxjava机制有大致了解

static final class CallOnSubscribe<T> implements Observable.OnSubscribe<Response<T>> {
  private final Call<T> originalCall;                                                 

  CallOnSubscribe(Call<T> originalCall) {                                             
    this.originalCall = originalCall;                                                 
  }                                                                                   

  @Override public void call(final Subscriber<? super Response<T>> subscriber) {      
    // Since Call is a one-shot type, clone it for each new subscriber.               
    Call<T> call = originalCall.clone();                                              

    // Wrap the call in a helper which handles both unsubscription and backpressure.  
    RequestArbiter<T> requestArbiter = new RequestArbiter<>(call, subscriber);        
    subscriber.add(requestArbiter);                                                   
    subscriber.setProducer(requestArbiter);                                           
  }                                                                                   
}                                                                                     

继续看RequestArbiter

static final class RequestArbiter<T> extends AtomicBoolean implements Subscription, Producer {     
  private final Call<T> call;                                                                      
  private final Subscriber<? super Response<T>> subscriber;                                        

  RequestArbiter(Call<T> call, Subscriber<? super Response<T>> subscriber) {                       
    this.call = call;                                                                              
    this.subscriber = subscriber;                                                                  
  }                                                                                                

  @Override public void request(long n) {                                                          
    if (n < 0) throw new IllegalArgumentException("n < 0: " + n);                                  
    if (n == 0) return; // Nothing to do when requesting 0.                                        
    if (!compareAndSet(false, true)) return; // Request was already triggered.                     

    try {                                                                                          
      Response<T> response = call.execute();                                                       
      if (!subscriber.isUnsubscribed()) {                                                          
        subscriber.onNext(response);                                                               
      }                                                                                            
    } catch (Throwable t) {                                                                        
      Exceptions.throwIfFatal(t);                                                                  
      if (!subscriber.isUnsubscribed()) {                                                          
        subscriber.onError(t);                                                                     
      }                                                                                            
      return;                                                                                      
    }                                                                                              

    if (!subscriber.isUnsubscribed()) {                                                            
      subscriber.onCompleted();                                                                    
    }                                                                                              
  }                                                                                                
   //取消订阅的时候就调用了retrofit2.OkHttpCall.cancel()方法
   //该方法也就执行了okhttp里call.cancel方法取消网络请求了。                                                             
  @Override public void unsubscribe() {                                                            
    call.cancel();                                                                                 
  }                                                                                                

  @Override public boolean isUnsubscribed() {                                                      
    return call.isCanceled();                                                                      
  }                                                                                                
}                                                                                                  

三、总结。
完~

  • 14
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: OkHttpRxJavaRetrofit是个非常常用的组合,用来进行网络请求和处理。 首先,OkHttp是一个开源的HTTP客户端,提供了简洁的接口,用于与服务器进行通信,并且支持HTTP/2协议,拥有连接池、请求重试和缓存等功能。 然后,RxJava是一个基于观察者模式的异步编程库,可以用于处理异步操作,如网络请求、文件IO等。它的核心是Observable(被观察者)和Subscriber(订阅者),通过各种操作符可以对数据进行变换和处理。 最后,Retrofit是一个RESTful风格的HTTP请求库,它基于OkHttp,使用了Retrofit的注解和接口定义的方式,可以方便地进行网络请求。它支持动态代理,可以将网络请求接口转化为对应的HTTP请求,支持同步和异步请求,并且可以将响应数据转化为Java对象。 综上所述,我们可以使用OkHttp作为底层网络库,然后结合RxJavaRetrofit进行网络请求和数据处理。使用Retrofit的注解和接口定义方式,可以简化网络请求的代码,并且通过RxJava的操作符可以对请求结果进行变换和处理,使得代码更加清晰和可读性。 在使用过程中,可以先创建一个Retrofit实例,并指定OkHttpClient作为网络客户端,然后定义一个接口,在该接口中使用Retrofit的注解,定义网络请求的方法。最后,通过创建该接口的实例,即可进行网络请求,并结合RxJava进行操作。 总之,使用OkHttpRxJavaRetrofit组合进行网络请求可以提高效率和可读性,并且可以处理各种复杂的网络场景,是一种非常实用的方式。 ### 回答2: OKHTTPRXJavaRetrofitAndroid开发中常用的三个库,可以一起使用来进行网络请求和数据处理。 OKHTTP是一个用于处理网络请求的库,可以发送HTTP请求并获取服务器返回的数据。它提供了简洁的API和高效的网络堆栈,可以很好地处理网络请求。我们可以使用OKHTTP来发送SOAP请求到WebService,并获得响应。 RXJava是一个流编程库,它提供了一种被观察者和观察者模式,可以简化异步操作和事件处理。我们可以使用RXJava来处理OKHTTP返回的响应数据,在主线程或后台线程中进行处理,实现数据的异步处理和流式编程。 Retrofit是一个基于OKHTTP的RESTful风格的网络请求库,它提供了一种简洁的方式来定义和发送HTTP请求,并将响应转换为可用的Java对象。我们可以使用Retrofit来定义WebService接口,然后使用注解来指定请求方法、路径和参数,Retrofit会自动帮我们处理请求和响应。 通过OKHTTP的原生支持、RXJava的异步处理和Retrofit网络请求,我们可以很方便地使用OKHTTPRXJavaRetrofit一起发送WebService请求。首先,我们可以使用Retrofit定义WebService接口,再使用RXJava来处理OKHTTP返回的响应数据,实现简洁高效的网络请求和数据处理。 综上所述,OKHTTPRXJavaRetrofitAndroid开发中常用的网络请求库,它们能够很好地协作,实现对WebService的请求和响应的处理。通过使用它们,我们可以简化网络请求的编写,并实现高效的数据处理。 ### 回答3: OkHttpRxJavaRetrofit是三个在Android开发中常用的网络请求库,它们在一起能够提供更加便捷和高效的网络请求处理。 首先,OkHttp是一个开源的HTTP客户端,它能够处理网络请求、连接管理、请求重试等一系列的网络相关事务。它的特点是简单易用、性能优越、可定制性强。我们可以通过使用OkHttp来发送和接收基于HTTP的请求响应,并进行网络请求的管理和处理。 其次,RxJava是一个响应式编程框架,它基于观察者模式和函数式编程的思想,提供了一系列强大的操作符和线程切换的能力。我们可以使用RxJava来处理异步任务,加快网络请求的响应时间,并且提供方便的线程切换和错误处理机制。 最后,Retrofit是一个RESTful风格的网络请求框架,它结合了OkHttpRxJava的强大功能。它提供了一种简单的方式来定义和处理RESTful API的请求和响应。我们可以使用Retrofit来创建和处理webservice的请求,根据API的接口定义来发送请求,并将返回的结果映射到Java对象中。 综上所述,使用OkHttpRxJavaRetrofit能够方便地进行webservice的网络请求,并在处理过程中提供更好的性能和便利性。这三个库的结合能够大大简化网络请求的开发工作,提高开发效率,并提供更好的用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值