仿照Retrofit写出自己的网络请求框架

Retrofit

RetrofitSquare出品的Android Http请求框架,官网给出的介绍是Type-safe HTTP client for Android and Java by Square, Inc.,简单来说其实是对okhttp扩展,更加方便的构造REST风格的HTTP客户端,我们再使用时可以看出其主要特点是接口类和注解类的创建和使用。

API定义:

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);
Call<List<Repo>> repos = service.listRepos("octocat");

简析

Retrofit可以这么简洁的原因是使用了java动态代理技术
动态代理技术是整个java技术中最重要的一个技术,它是学习java框架的基础。
详细介绍可以参考:https://www.cnblogs.com/xdp-gacl/p/3971367.html

开始实现

首先,我们确定最终实现的效果:

public interface User{
    /**
     * 企业基本信息更新接口
     *
     * @param member_id
     * @param company_id
     * @param context_data
     * @return
     */
    @ACT("UCD0507")
    Call<ResponseSimple> updateCompanyData(
            @KEY("user_id") @NonNull String member_id,
            @KEY("company_id") @NonNull String company_id,
            @KEY(value = "context_data", type = KeyType.JSONOArray) @NonNull JSONArray context_data
    );
}
API.create(User.class).updateCompanyData("user_id","company_id",context_data).enqueue(new ResponseCallback<ResponseSimple>(this) {
                        @Override
                        public void onResponseSuccess(Response<ResponseSimple> response) {
                            if (response.isSuccessful()){
                                CustomToast.showShort("提交成功");
                            }else {
                                CustomToast.showShort(response.message());
                            }
                        }

                        @Override
                        public void onResponseError(Response<ResponseSimple> response, String error) {
                            CustomToast.showShort(error);
                        }
                    });

context_data是一個jsonArray的對象,大家也可以自定義其他的對象與輸出格式,这里就不一一实现了。这也是自定义Retrofit的好处之一,可以随心所欲的控制api的协议与调用。
@ACT代表的是接口的协议编号,这里可能不太符合rest风格,限于接口协议,我们仅参考Retrofit的使用与实现,并不完全按照Retrofit来。@KEY为参数的键。

关键代码

@ACT的实现

@Documented
@Target(METHOD)
@Retention(RUNTIME)
public @interface ACT {
    public String value() default "";
}

@KEY的实现

@Documented
@Target(PARAMETER)
@Retention(RUNTIME)
public @interface KEY {
    public String value() default "";
    public KeyType type() default KeyType.String;
}

//此处代表参数类型,可自定义为需要输入值的类型,方便按对应的规则解析
public enum KeyType {
    JSONObject,
    JSONOArray,
    JSONObjectString,
    JSONArrayString,
    Object,
    ObjectList,
    String
}

开始实现User接口,并解析@ACT@KEY注解


public final class API<T>{

    /**
     * 实现接口的定义
     * @param interfaceClass
     * @param <T>
     * @return 
     */
    public static <T> T create(Class<T> interfaceClass) {
        return (T) Proxy.newProxyInstance(interfaceClass.getClassLoader(), new Class[]{interfaceClass}, (proxy, method, args) -> {
            final String act = method.getAnnotation(ACT.class).value();//获取方法所包含的注解项
            final Type returnType = method.getGenericReturnType();//获取返回值类型
            ParameterizedType parameterizedType = (ParameterizedType) returnType;
            final Type type = parameterizedType.getActualTypeArguments()[0];
            return new API(act, method, args, type);
        });
    }

    private final OkHttpClient okHttpClient=new OkHttpClient();
    private final String uri="接口地址";
    /**
     * api协议编号
     */
    private final String act;
    /**
     * 【反射调用】协议的方法
     */
    private final Method method;
    /**
     * 【反射调用】协议的参数
     */
    private final Object[] args;
    private final Type type;
    private static final MediaType REQUEST_MEDIA_TYPE = MediaType.parse("application/x-www-form-urlencoded");

    private API(String act,Method method, Object[] args, Type type) {
        this.act = act;
        this.method = method;
        this.args = args;
        this.type = type;
    }

    /**
     * 解析请求参数
     * @return
     */
    private okhttp3.Call createCall() {
        JSONObject rootJson = new JSONObject();
        JSONObject dataJson = new JSONObject();
        Annotation[][] annotationsList = method.getParameterAnnotations();
        try {
            for (int i = 0; i < args.length; i++) {
                Annotation[] annotations = annotationsList[i];//获取参数注解
                for (Annotation annotation : annotations) {
                    if (annotation instanceof KEY) {//找到注解KEY并进行处理
                        KEY key = (KEY) annotation;
                        if (args[i] == null) {
                        } else {//含有类型值,判断后找到相应的处理方案进行加工
                            if (args[i] instanceof JSONObject
                                    || args[i] instanceof JSONArray) {
                                dataJson.put(key.value(), args[i]);
                            } else if (key.type() == KeyType.JSONObject) {
                                dataJson.put(key.value(), args[i]);
                            } else if (key.type() == KeyType.JSONOArray) {
                                dataJson.put(key.value(), args[i]);
                            } else if (key.type() == KeyType.JSONArrayString) {
                                dataJson.put(key.value(), new JSONArray(args[i] + ""));
                            } else if (key.type() == KeyType.JSONObjectString) {
                                dataJson.put(key.value(), new JSONObject(args[i] + ""));
                            } else if (key.type() == KeyType.String) {
                                dataJson.put(key.value(), args[i] + "");
                            } else if (key.type() == KeyType.Object) {
                                dataJson.put(key.value(), new JSONObject(new Gson().toJson(args[i])));
                            } else if (key.type() == KeyType.ObjectList) {
                                dataJson.put(key.value(), new JSONArray(new Gson().toJson(args[i])));
                            }
                        }
                        break;
                    }
                }
            }
            String signString = HEXUtil.str2HexStr(Base64Util.encode(SecretUtil.encrypt("[自定义的协议加密]".getBytes())));
            rootJson.put("act", act)
                    .put("sign", signString)
                    .put("data", dataJson);
        } catch (JSONException e) {
            e.printStackTrace();
        }
        return okHttpClient.newCall(new Request.Builder()
                .url(uri)
                .post(RequestBody.create(REQUEST_MEDIA_TYPE, rootJson.toString()))
                .build());
    }


    @Override
    public void enqueue(Callback<T> callback) {
        createCall().enqueue(new okhttp3.Callback() {
            @Override
            public void onFailure(okhttp3.Call call, IOException e) {
                if (callback != null) {
                    callback.onFailure(e);
                }
            }

            @Override
            public void onResponse(okhttp3.Call call, okhttp3.Response response) throws IOException {
                if (callback != null) {
                    callback.onResponse(convert(response));
                }
            }
        });
    }

    /**
     * 解析返回值
     * @param response
     * @return
     */
    private T convert(okhttp3.Response response){
        return null;
    }

}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容