天天看點

優雅地調用接口--手把手帶你從0到1實作retrofit架構

retrofit作為http請求的接口架構而被衆人熟知道,它獨特且優雅的接口使用習慣被無數的程式員所認可。 http://square.github.io/retrofit/

介紹

A type-safe HTTP client for Android and Java

Retrofit是一個基于AOP思想,對RestfulApi注解進行動态代理的網絡架構。

public interface GitHubService {
  @GET("users/{user}/repos")
  Call<List<Repo>> listRepos(@Path("user") String user);
}
Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .build();


GitHubService service = retrofit.create(GitHubService.class);
           

我們的JT架構

JTHttpClient
                .create()
                .setSecLevel(JtRequest.NORMAL)
                .url(Urls.index_V)
                .build()
                .post(HomeModule.class)
                .subscribe(new JtSingObserver<HomeModule>(this) {
                    @Override
                    public void onSuccess(HomeModule homeModule) {
                        if (homeModule.header.app_login_status == 0) {
                            JTApi.get().logout();
                        }
                        sHomeModule = homeModule;
                        mHomeModule.setValue(homeModule);

                    }

                    @Override
                    public void onError(Throwable e) {
                        sHomeModule = null;
                        mHomeModule.setValue(null);
                        UIUtil.INSTANCE.showExceptionMsg(e);
                    }
                });

           

我在深思熟慮中打造的JT架構雖然簡單易用性和性能上其實已經很優秀,但是經過我再一次的深思熟慮覺得依然存在着一些問題。

  • 模闆類代碼太多
  • 泛型傳遞太頻發(兩次)
  • 鍊式的調用過程某種程度意味效率的低下(選擇遺忘的太多)

Retrofit的實作過程

retrofit的主要技術細節包括:

1.接口的執行個體化

2.擷取泛型的傳回值type(資料的解析和執行個體化觀察者需要)

3.擷取Method之後通過反射擷取方法注解和參數注解

我們忽略了一些AOP程式設計的細節,而是主要去關注一個注解方法對象(Method)的解析過程。

retrofit将Method對象解析成對應http request;将request的參數給下層的網絡擴充卡adapter(可以認為是OkhttpAdapter),擴充卡調用自身execute方法,調用成功後通過線程分發器發送到對應的線程。至此,完成了retrofit對網絡的完整的抽象。

優雅地調用接口--手把手帶你從0到1實作retrofit架構

retrofit_stay.png

public <T> T create(final Class<T> service) {//函數的入口,執行個體化接口對象
        Utils.validateServiceInterface(service);//檢查service合法性
        if (validateEagerly) {
            eagerlyValidateMethods(service);
        }
        //動态代理執行個體化對象
        return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] {service},
                new InvocationHandler() {

                    @Override
                    public Object invoke(Object proxy, Method method, Object... args)
                            throws Throwable {
                        // If the method is a method from Object then defer to normal invocation.
                        //解析method注解并交給call對象(網絡執行者,預設okhttpclient)
                        // 觸發http請求的方法。invoke會傳回Call對象,調用該對象的方法即可完成網絡請求
                        return loadMethodHandler(method).invoke(args);

                    }
                });
    }

    MethodHandler loadMethodHandler(Method method) {
        MethodHandler handler;
        synchronized(methodHandlerCache) {//檢視靜态緩存區,是否有對應method可用,加快執行的速度
            handler = methodHandlerCache.get(method);
            if (handler == null) {
                //建立MethodHandler對象,
                //該對象用來管理
                // requestFactory請求構造器,
                // callFactory(http執行棧預設OkHttpClient),
                // callAdapter回調擴充卡(用于線程的分發),
                // responseConverter内容轉換器
                // 可以認為一種接口method與http請求過程的媒介
                // new MethodHandler(retrofit.callFactory(), requestFactory, callAdapter,responseConverter);

                handler = MethodHandler.create(this, method);
                methodHandlerCache.put(method, handler);
            }
        }
        return handler;
    }

    static MethodHandler create(Retrofit retrofit, Method method) {
        //建立callAdapter, callAdapter是一個可以用來觸發網絡成功或失敗的回調的對象,該方法主要
        // 1.查找目前隊列中可用的callAdapter
        // 2.不管有沒有查找成功都會擷取method傳回值(method.getGenericReturnType())的泛型type
        // 3.驗證合法性
        CallAdapter<?> callAdapter = createCallAdapter(method, retrofit);
        Type responseType = callAdapter.responseType();
        if (responseType == Response.class || responseType == okhttp3.Response.class) {
            throw Utils.methodError(method, "'"
                    + Types.getRawType(responseType).getName()
                    + "' is not a valid response body type. Did you mean ResponseBody?");
        }
        //建立responseConverter,這個方法不關注
        Converter<ResponseBody, ?> responseConverter =
                createResponseConverter(method, retrofit, responseType);
        //解析獲得請求的參數,建立requestFactory,最為核心且最複雜的方法
        //        RequestFactory:
        //        private final String method;
        //        private final BaseUrl baseUrl;
        //        private final String relativeUrl;
        //        private final Headers headers;
        //        private final MediaType contentType;
        //        private final boolean hasBody;
        //        private final boolean isFormEncoded;
        //        private final boolean isMultipart;
        //        private final RequestAction[] requestActions;
        RequestFactory requestFactory = RequestFactoryParser.parse(method, responseType, retrofit);
        return new MethodHandler(retrofit.callFactory(), requestFactory, callAdapter,
                responseConverter);
    }

  
    static RequestFactory parse(Method method, Type responseType, Retrofit retrofit) {
        RequestFactoryParser parser = new RequestFactoryParser(method);

        Annotation[] methodAnnotations = method.getAnnotations();
        //解析method方法的注解annotations[]
        parser.parseMethodAnnotations(responseType, methodAnnotations);
        //解析method參數的注解annotations[][],此處是二維數組
        parser.parseParameters(retrofit, methodAnnotations);

        return parser.toRequestFactory(retrofit.baseUrl());
    }
//該方法用于通過returnType查找對應的callAdapter并擷取執行個體化的callAdapter對象
    private static CallAdapter<?> createCallAdapter(Method method, Retrofit retrofit) {
        Type returnType = method.getGenericReturnType();
        if (Utils.hasUnresolvableType(returnType)) {
            throw Utils.methodError(method,
                    "Method return type must not include a type variable or wildcard: %s", returnType);
        }
        if (returnType == void.class) {
            throw Utils.methodError(method, "Service methods cannot return void.");
        }
        Annotation[] annotations = method.getAnnotations();
        try {
            return retrofit.callAdapter(returnType, annotations);
        } catch (RuntimeException e) { // Wide exception range because factories are user code.
            throw Utils.methodError(e, method, "Unable to create call adapter for %s", returnType);
        }
    }



           

JT的實作

主要實作的步驟:

  1. 動态代理執行個體化接口對象
  2. 在invoke函數中擷取被調用的Method對象
  3. 解析并拿到Method中的request參數和傳回值泛型的type
//return type
Type returnType = method.getGenericReturnType();
            if (method.getReturnType() == Single.class) {
            } else {
                throw new IllegalArgumentException("you must return a type == Single.class");
            }
            if (returnType instanceof ParameterizedType) {
                Type[] actualTypeArguments = ((ParameterizedType) returnType)
                        .getActualTypeArguments();
                if (actualTypeArguments.length == 1) {
                    mReturnType = actualTypeArguments[0];
                } else {
                    throw new IllegalArgumentException();
                }
            }
//緩存Method的request.builder參數,調用perform方法去注入builder
private List<RequestAction> requestActionList = new ArrayList<>();
           
  1. 通過jtclient發送http請求

    5.根據3中的傳回值泛型type解析response

    6.最後,通知觀察者,處理所訂閱的事件

apiService = Proxy.newProxyInstance(service.getClassLoader(),
                new Class<?>[] {service},
                new InvocationHandler() {

                    //這個invoke方法會在代理對象的方法中調用,第一個參數就是代理對象
                    //第二個參數是代理對象調用的方法
                    //第三個參數方法的參數
                    @Override
                    public Object invoke(Object proxy, Method method, Object... args)
                            throws Throwable {
                        validateServiceInterface(service);

                        // If the method is a method from Object then defer to normal invocation.
                        if (method.getDeclaringClass() == Object.class) {
                            return method.invoke(this, args);
                        }
                        //調用loadMethodHandler
                        return loadMethodHandler(method).invoke(args);
                    }
                });

Type returnType = method.getGenericReturnType();
            if (method.getReturnType() == Single.class) {
            } else {
                throw new IllegalArgumentException("you must return a type == Single.class");
            }
            if (returnType instanceof ParameterizedType) {
                Type[] actualTypeArguments = ((ParameterizedType) returnType)
                        .getActualTypeArguments();
                if (actualTypeArguments.length == 1) {
                    mReturnType = actualTypeArguments[0];
                } else {
                    throw new IllegalArgumentException();
                }
            }






        MethodHandler(Method method) {
            if (JTApi.get().isDebug()) {
                Log.e("MethodHandler", "MethodHandler-method " + method.getName());
            }
            Type returnType = method.getGenericReturnType();
            if (method.getReturnType() == Single.class) {
            } else {
                throw new IllegalArgumentException("you must return a type == Single.class");
            }
            if (returnType instanceof ParameterizedType) {
                Type[] actualTypeArguments = ((ParameterizedType) returnType)
                        .getActualTypeArguments();
                if (actualTypeArguments.length == 1) {
                    mReturnType = actualTypeArguments[0];
                } else {
                    throw new IllegalArgumentException();
                }
            }
            Annotation[] methodAnnotation = method.getAnnotations();
            for (Annotation annotation : methodAnnotation) {
                if (annotation instanceof Url) {
                    mUrl = ((Url) annotation).value();
                } else if (annotation instanceof Security) {
                    security = ((Security) annotation).value();
                }
            }

            Annotation[][] parameterAnnotations = method.getParameterAnnotations();
            if (parameterAnnotations == null || parameterAnnotations.length == 0) {
                return;
            }
            for (Annotation[] annotations : parameterAnnotations) {
                if (annotations != null && annotations.length != 0) {
                    for (Annotation aMethodAnnotation : annotations) {
                        if (aMethodAnnotation instanceof Field) {
                            Field field = (Field) aMethodAnnotation;
                            requestActionList.add(new FieldRequestAction(field.value()));
                            break;

                        } else if (aMethodAnnotation instanceof FieldMap) {
                            requestActionList.add(new FieldMapRequestAction());
                        } else if (aMethodAnnotation instanceof Path) {
                            Path path = (Path) aMethodAnnotation;
                            requestActionList.add(new PathRequestAction(path.value()));
                        }
                    }
                }
            }
        }
        public Object invoke(Object... objects) {
            JtRequest.Builder builder = JTHttpClient.create().url(mUrl);
            if (objects != null && objects.length > 0) {
                if (objects.length != requestActionList.size()) {
                    throw new NullPointerException("url: " + mUrl + "\nrequestActionList length is "
                            + "not equals objects length");
                }
                for (int i = 0; i < requestActionList.size(); i++) {
                    requestActionList.get(i).perform(builder, objects[i]);
                }
            }

            return builder.setSecLevel(security).build().post(mReturnType);

        }

//execute
builder.setSecLevel(security).build().post(mReturnType);

           

繼續閱讀