Call<List> listRepos(@Path(“user”) String user);
 }
创建Retrofit并生成API的实现
Retrofit retrofit = new Retrofit.Builder()
 .baseUrl(“https://api.github.com/”)
 .build();
 GitHubService service = retrofit.create(GitHubService.class);
调用API方法,生成Call
Call<List> repos = service.listRepos(“octocat”);
Retrofit的创建
retrofit实例的创建,使用了builder模式,从下面的源码中可以看出
public static final class Builder {
 Builder(Platform platform) {
 this.platform = platform;
 converterFactories.add(new BuiltInConverters());
 }
 public Builder() {
 // Platform.get()方法可以用于判断当前的环境
 this(Platform.get());
 }
 public Builder baseUrl(String baseUrl) {
 checkNotNull(baseUrl, “baseUrl == null”);
 HttpUrl httpUrl = HttpUrl.parse(baseUrl);
 if (httpUrl == null) {
 throw new IllegalArgumentException("Illegal URL: " + baseUrl);
 }
 return baseUrl(httpUrl);
 }
public Retrofit build() {
 if (baseUrl == null) {
 throw new IllegalStateException(“Base URL required.”);
 }
 okhttp3.Call.Factory callFactory = this.callFactory;
 if (callFactory == null) {
 callFactory = new OkHttpClient();// 新建Client,留到之后newCall什么的
 }
 Executor callbackExecutor = this.callbackExecutor;
 if (callbackExecutor == null) {
 callbackExecutor = platform.defaultCallbackExecutor();
 }
 // Make a defensive copy of the adapters and add the default Call adapter.
 List<CallAdapter.Factory> adapterFactories = new ArrayList<>(this.adapterFactories);
 adapterFactories.add(platform.defaultCallAdapterFactory(callbackExecutor));
 // Make a defensive copy of the converters.
 List<Converter.Factory> converterFactories = new ArrayList<>(this.converterFactories);
 return new Retrofit(callFactory, baseUrl, converterFactories, adapterFactories,
 callbackExecutor, validateEagerly);
 }
 }
retrofit.create
好玩的地方开始了,我们先来看看这个方法
public T create(final Class service) {
 Utils.validateServiceInterface(service);
 if (validateEagerly) {
 eagerlyValidateMethods(service);
 }
 // 动态代理,啦啦啦
 return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service },
 new InvocationHandler() {
 // platform 可以分辨出你是在android,还是java8,又或者别的
 private final Platform platform = Platform.get();
 @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.
 // 这里的invoke,Object方法都走这里,比如equals、toString、hashCode什么的
 if (method.getDeclaringClass() == Object.class) {
 return method.invoke(this, args);
 }
 // java8默认方法,1.8的新特性
 if (platform.isDefaultMethod(method)) {
 return platform.invokeDefaultMethod(method, service, proxy, args);
 }
 // 这里是核心代码了
 ServiceMethod<Object, Object> serviceMethod =
 (ServiceMethod<Object, Object>) loadServiceMethod(method);
 OkHttpCall okHttpCall = new OkHttpCall<>(serviceMethod, args);
 return serviceMethod.callAdapter.adapt(okHttpCall);
 }
 });
 }
可以看出创建API使用了动态代理,根据接口动态生成的代理类,将接口的都转发给了负责连接代理类和委托类的InvocationHandler实例,接口方法也都通过其invoke方法来处理。
 在invoke方法中,首先会通过Platform.get()方法判断出当前代码的执行环境,之后会先把Object和Java8的默认方法进行一个处理,也是在进行后续处理之前进行去噪。其中的关键代码其实就是最后三句,这也是这篇文章将要分析的
创建ServiceMethod
erviceMethod<?, ?> loadServiceMethod(Method method) {
 // 从缓存里面取出,如果有的话,直接返回好了
 ServiceMethod<?, ?> result = serviceMethodCache.get(method);
 if (result != null) return result;
 synchronized (serviceMethodCache) {
 result = serviceMethodCache.get(method);
 if (result == null) {
 // 为null的话,解析方法的注解和返回类型、参数的注解he参数类型,新建一个ServiceMethod
 result = new ServiceMethod.Builder<>(this, method).build();// ->
 // 新建的ServiceMethod加到缓存列表里面
 serviceMethodCache.put(method, result);
 }
 }
 return result;
 }
注解的解析
CallAdapter和Converter等到后面再分析,这里先看看parseMethodAnnotation(annotation),功能和其名字一样,其对方法注解进行了解析
/**
- 解析方法注解,呜啦啦
- 通过判断注解类型来解析
- @param annotation
 */
 private void parseMethodAnnotation(Annotation annotation) {
 if (annotation instanceof DELETE) {
 parseHttpMethodAndPath(“DELETE”, ((DELETE) annotation).value(), false);
 } else if (annotation instanceof GET) {
 parseHttpMethodAndPath(“GET”, ((GET) annotation).value(), false);
 }
 // 其他的一些方法注解的解析
 …
 }
 private void parseHttpMethodAndPath(String httpMethod, String value, boolean hasBody) {
 if (this.httpMethod != null) {// 已经赋值过了
 throw methodError(“Only one HTTP method is allowed. Found: %s and %s.”,
 this.httpMethod, httpMethod);
 }
 this.httpMethod = httpMethod;
 this.hasBody = hasBody;
 // value为设置注解方法时候,设置的值,官方例子中的users/{user}/repos or user
 if (value.isEmpty()) {
 return;
 }
 // 查询条件的一些判断
 …
 this.relativeUrl = value;
 this.relativeUrlParamNames = parsePathParameters(value);
 }
 `
在解析注解时,先通过instanceof判断出注解的类型,之后调用parseHttpMethodAndPath方法解析注解参数值,并设置httpMethod、relativeUrl、relativeUrlParamNames等属性。
上面说了API中方法注解的解析,现在来看看方法参数注解的解析,这是通过调用parseParameterAnnotation方法生成ParameterHandler实例来实现的,代码比较多,这里挑选@Query来看看。
else if (annotation instanceof Query) {
 Query query = (Query) annotation;
 String name = query.value();
 boolean encoded = query.encoded();
 Class<?> rawParameterType = Utils.getRawType(type);// 返回基础的类
 gotQuery = true;
 // 可以迭代,Collection
 if (Iterable.class.isAssignableFrom(rawParameterType)) {
 if (!(type instanceof ParameterizedType)) {
 throw parameterError(p, rawParameterType.getSimpleName()
- " must include generic type (e.g., "
- rawParameterType.getSimpleName()
- “)”);
 }
 ParameterizedType parameterizedType = (ParameterizedType) type;
 Type iterableType = Utils.getParameterUpperBound(0, parameterizedType);// 返回基本类型
 Converter<?, String> converter =
 retrofit.stringConverter(iterableType, annotations);
 return new ParameterHandler.Query<>(name, converter, encoded).iterable();
 } else if (rawParameterType.isArray()) {// Array
 Class<?> arrayComponentType = boxIfPrimitive(rawParameterType.getComponentType());// 如果是基本类型,自动装箱
 Converter<?, String> converter =
 retrofit.stringConverter(arrayComponentType, annotations);
 return new ParameterHandler.Query<>(name, converter, encoded).array();
 } else {// Other
 Converter<?, String> converter =
 retrofit.stringConverter(type, annotations);
 return new ParameterHandler.Query<>(name, converter, encoded);
 }
在@Query中,将分成Collection、array、other三种情况处理参数,之后根据这些参数,调用ParameterHandler中的Query静态类,创建出一个ParameterHandler实例。这样循环直到解析了所有的参数注解,组合成为全局变量parameterHandlers,之后构建请求时会用到
OkHttpCall
ServiceMethod创建完成之后,我们来看看下一行代码中的OkHttpCall类,里面的包含了请求的执行和响应处理,我们来看看异步请求的做法
OkHttpCall(ServiceMethod<T, ?> serviceMethod, Object[] args) {
 this.serviceMethod = serviceMethod;
 this.args = args;
 }
 @Override public void enqueue(final Callback callback) {
 checkNotNull(callback, “callback == null”);
 okhttp3.Call call;
 Throwable failure;
 synchronized (this) {
 if (executed) throw new IllegalStateException(“Already executed.”);
 executed = true;
 call = rawCall;
 failure = creationFailure;
 if (call == null && failure == null) {
 try {
 call = rawCall = createRawCall();// 创建OkHttp3.Call
 } catch (Throwable t) {
 failure = creationFailure = t;
 }
 }
 }
 if (failure != null) {
 callback.onFailure(this, failure);
 return;
 }
 if (canceled) {
 call.cancel();
 }
 call.enqueue(new okhttp3.Callback() {
资源分享
- 最新大厂面试专题
这个题库内容是比较多的,除了一些流行的热门技术面试题,如Kotlin,数据库,Java虚拟机面试题,数组,Framework ,混合跨平台开发,等

- 对应导图的Android高级工程师进阶系统学习视频
 最近热门的,NDK,热修复,MVVM,源码等一系列系统学习视频都有!

些流行的热门技术面试题,如Kotlin,数据库,Java虚拟机面试题,数组,Framework ,混合跨平台开发,等
[外链图片转存中…(img-WajqqjLL-1644035867733)]
- 对应导图的Android高级工程师进阶系统学习视频
 最近热门的,NDK,热修复,MVVM,源码等一系列系统学习视频都有!
[外链图片转存中…(img-xYBDTmpP-1644035867734)]
下载方法:点赞+关注后 点击【Android高级工程师进阶学习】即可领取!









