Retrofit各个注解的含义及作用(项目经验和单个注解解释)

具体用法参照 Retrofit官网

简介

提示依赖 https://www.jianshu.com/p/6534f3ef58cc

  • 是一个基于okhttp的网络请求框架
  • 通过注解配置网络请求参数
  • 图片链接和图片上传
  • 支持同步和异步网络请求
  • 支持多种数据的解析,提供对Rxjava的支持
  • 可拓展性好,高度封装,简洁易用

步骤

  • 添加Retrofit库的依赖
  • 创建接收服务器返回数据的类
  • 创建用于描述网络请求的接口
  • 创建 Retrofit 实例
  • 创建 网络请求接口实例 并 配置网络请求参数
  • 发送网络请求(异步 / 同步)
  • 处理数据
image

1. GET

//简单的get请求(没有参数)
@GET("trades")
Call<TradesBean> getItem();

//简单的get请求(URL中带有参数)
@GET("News/{userId}")
Call<TradesBean> getItem(@Path("userId") String userId);

//简单的get请求(URL中带有两个参数)
@GET("News/{userId}")
Call<TradesBean> getItem(@Path("userId") String userId,@Path("type") String type);

//参数在url问号之后
@GET("trades")
Call<TradesBean> getItem(@Query("userId") String userId);

@GET("trades")
Call<TradesBean> getItem(@QueryMap Map<String, String> map);

@GET("trades")
Call<TradesBean> getItem(
@Query("userId") String userId,
@QueryMap Map<String, String> map);

@GET("{a}/json?")
Call<Bean> beanCall(@Path("a") int a, @Query("cid") String cid);

其他拼接方式

@GET
Call<ImageBean> getImageList3(@Url String url);

@GET
Call<ImageBean> getImageList4(@Url String url,@Query("uid") String uid);

2. POST

//需要补全URL,post的数据只有一条reason
@FormUrlEncoded
@POST("trades/{userId}")
Call<TradesBean> postResult(@Path("userId") String userId,
@Field("reason") String reason;

//需要补全URL,问号后需要加token,post的数据只有一条reason
@FormUrlEncoded
@POST("trades/{userId}")
Call<TradesBean> postResult(
@Path("userId") String userId,
@Query("token") String token,
@Field("reason") String reason;

//post一个对象
@POST("trades/{userId}")
Call<TradesBean> postResult(
@Path("userId") String userId,
@Query("token") String token,
@Body TradesBean bean;

//用不同注解post一个实体
@POST("trades/{userId}")
Call<TradesBean> postResult(
@Part("entity") TradesBean bean;

其他拼接方式

@POST("login")
@FormUrlEncoded
Call<LoginBean> login1(@Field("username") String username,@Field("password") String pw);

@POST
@FormUrlEncoded
Call<LoginBean> login3(@Url String url,@Field("username") String username,@Field("password") String password);

3. PUT

//put一个实体
@PUT("trade/carInfo/{pid}")
Call<TradesBean> putInfo
@Path("pid") Int pid,
@Body CarInfoBean carInfoBean;)

4. DELETE

http://192.168.43.173/api/trades/{userId}

//补全url
@DELETE("trades/{userId}")
Call<TradesBean> deleteInfo(
@Path("userId") String userId;

http://192.168.43.173/api/trades/{userId}?token={token}

//补全url并且后面还token
@DELETE("trades/{userId}")
Call<TradesBean> deleteInfo(
@Path("userId") String userId,
@Query("token") String token;)

5. 以Json格式提交数据 (模拟登录功能)两种方法

1、ApiService

 /**
 * 登录
 */
@Headers("Content-Type:application/json")
@POST("user/login")
Observable<ResponseBody> login(@Body RequestBody requestBody);

2、创建一个Bean类

public class UserBean {
private String nickname;
private String password;

public UserBean(String nickname, String password) {
    this.nickname = nickname;
    this.password = password;
}

public String getNickname() {
    return nickname;
}

public void setNickname(String nickname) {
    this.nickname = nickname;
}

public String getPassword() {
    return password;
}

public void setPassword(String password) {
    this.password = password;
}

}

3、将Bean类通过Gson转为json体

Gson gson = new Gson();
HashMap<String, String> map = new HashMap<>();
map.put("nickname", "123456");
map.put("password, "abcdefg");
String jsonBody = gson.toJson(map);

/ / 此时的jsonBody打印出来的结果
// {"password":"abcdefg","nickname":"123456"}

4、Retrofit请求头是否携带 如果请求带请求头 用方法一,如果不带请求头 用方法二,请求头需要添加的数据根据需求自行添加

1、不带请求头
private void useRetrofit(String url) {
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(url)
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .build();
    apiServices = retrofit.create(ApiService.class);
}
2、带请求头
   private void useHeaderRetrofit(String url) {
    OkHttpClient.Builder builder = new OkHttpClient.Builder();
    builder.addInterceptor(new Interceptor() {
        @Override
        public Response intercept(Chain chain) throws IOException {
            Request original = chain.request();
            Request request = original.newBuilder()
                    .addHeader("Content-Type", "application/json;charset=utf-8")
                    .build();

            return chain.proceed(request);
        }
    });
    OkHttpClient client = builder.build();
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(url)
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .client(client)
            .build();
    apiServices = retrofit.create(ApiService.class);
}

5、网络请求的代码

   RequestBody body = RequestBody.create(okhttp3.MediaType.parse("application/json; charset=utf-8"), requestBody);
    Observable<ResponseBody> responseBodyObservable = apiServices.login(body);
    responseBodyObservable.observeOn(AndroidSchedulers.mainThread())
            .subscribeOn(Schedulers.io())
            .subscribe(new Observer<ResponseBody>() {
                @Override
                public void onCompleted() {

                }

                @Override
                public void onError(Throwable e) {
                    Log.d("TAG", "e:" + e);
                }

                @Override
                public void onNext(ResponseBody responseBody) {

                    try {
                        String string = responseBody.string();
                        Log.d("TAG", string);
                        callBack.onSuccess((T) string);
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            });

第二种(简单方法)

String base_url = "https://www.firstgainfo.com/firstga/app/news/";

1. ApiService
//requestBody 获取下拉刷新数据
@Headers({"Content-Type:application/json"})
@POST("downListNews")
Call<ResponseBody> getDownList(@Body RequestBody body);
2 MainActivity中
private void getDownListData(){
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(MyServer.base_url)
            .addConverterFactory(GsonConverterFactory.create())
            .build();
    MyServer myServer = retrofit.create(MyServer.class);
    JSONObject param = new JSONObject();
    try {
        param.put("userId","d56ea66e7ee741f498ca51242c8c6394");
        param.put("newsId","ad30af17a2ad44fca00a2d3f7e16c004");
        RequestBody body = RequestBody.create(MediaType.parse("application/json"),param.toString());
        Call<ResponseBody> call = myServer.getDownList(body);
        call.enqueue(new Callback<ResponseBody>() {
            @Override
            public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
                try {
                    Log.i("response",response.body().string());
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }

            @Override
            public void onFailure(Call<ResponseBody> call, Throwable t) {
                Log.i("onFailure",t.getMessage());
            }
        });
    } catch (JSONException e) {
        e.printStackTrace();
    }

}

6 以form-data格式提交数据 (模拟登录功能)

1、ApiService
1、当数据量大的情况下用方法1 推荐此方法 处理数据时方便一些
/**
 * 登录
 */
@Multipart
@POST("users/login")
Observable<ResponseBody> phoneLogin(@PartMap Map<String, RequestBody> requestBodyMap);
2、当数据量小的情况下用方法2
/**
 * 登录
 */
@Multipart
@POST("users/login")
Call<JsonObject> phoneLogin(@Part("nickname") RequestBody username, @Part("password")  RequestBody password);
2、转为form-data
 /**
 * 转换为 form-data
 *
 * @param requestDataMap
 * @return
 */
public static Map<String, RequestBody> generateRequestBody(Map<String, String> requestDataMap) {
    Map<String, RequestBody> requestBodyMap = new HashMap<>();
    for (String key : requestDataMap.keySet()) {
        RequestBody requestBody = RequestBody.create(MediaType.parse("multipart/form-data"),
                requestDataMap.get(key) == null ? "" : requestDataMap.get(key));
        requestBodyMap.put(key, requestBody);
    }
    return requestBodyMap;
}
3、Retrofit请求头是否携带 如果请求带请求头 用方法一,如果不带请求头 用方法二
1、不带请求头
 private void useRetrofit(String url) {
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(url)
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .build();
    apiServices = retrofit.create(ApiService.class);
}
2、带请求头
  private void useHeaderRetrofit(String url) {
    OkHttpClient.Builder builder = new OkHttpClient.Builder();
    builder.addInterceptor(new Interceptor() {
        @Override
        public Response intercept(Chain chain) throws IOException {
            Request original = chain.request();
            Request request = original.newBuilder()
                    .addHeader("Content-Type", "application/json;charset=utf-8")
                    .build();

            return chain.proceed(request);
        }
    });
    OkHttpClient client = builder.build();
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(url)
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .client(client)
            .build();
    apiServices = retrofit.create(ApiService.class);
}

第二种 (简单)

1. ApiService

String base_url = "https://www.firstgainfo.com/firstga/app/news/";

//获取详情数据
@Headers({"Content-Type:application/x-www-form-urlencoded"})
@POST("info")
@FormUrlEncoded
Observable<InfoBean> getData(@Field("userId") String userId, @Field("newsId") String newsId);
2. MainActivity
private void getData(){
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(MyServer.base_url)
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
            .build();
    MyServer myServer = retrofit.create(MyServer.class);
    Observable<InfoBean> observable = myServer.getData("d56ea66e7ee741f498ca51242c8c6394","ad30af17a2ad44fca00a2d3f7e16c004");
    observable.subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(new Observer<InfoBean>() {
                @Override
                public void onSubscribe(Disposable d) {

                }

                @Override
                public void onNext(InfoBean body) {
                    showInfo(body.getData().getContent());
                }

                @Override
                public void onError(Throwable e) {

                }

                @Override
                public void onComplete() {

                }
            });

}

拼接经验

  • Path是网址中的参数,例如:trades/{userId}
  • Query是问号后面的参数,例如:trades/{userId}?token={token}
  • QueryMap 相当于多个@Query
  • Field用于Post请求,提交单个数据,然后要加@FormUrlEncoded
  • Body相当于多个@Field,以对象的方式提交
  • @Streaming:用于下载大文件
  • @Header,@Headers、加请求头
Gson解析报错的原因
  • 参数拼接错误
  • 看报错信息,一般封装出问题
  • 对象改为 requestBody给泛型
  • 采用原生解析

单个注解的含义及作用

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