Retrofit 的基本使用

基于okhttp,并支持ESTful API 设计风格。网络请求工作实际由OkHttp完成,Retrofit主要负责接口的封装。可以通过注解配置请求:包括请求方法,请求参数,请求头,返回值等。对RxJava支持。

1.准备工作
1.1 导入依赖

implementation 'com.squareup.retrofit2:converter-gson:2.9.0'
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter-scalars:2.9.0'

1.2 开启网络权限

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>

在 res 目录下新建 xml 文件夹,新建 network-config.xml

<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
    <base-config cleartextTrafficPermitted="true" />
</network-security-config>

在 manifest application 下引入

android:networkSecurityConfig="@xml/network_config"

2.创建网络请求
2.1 创建网络请求的接口

public interface CreditService {
    @POST("/findStudentById")
    Call<String> getStudentById(@Query("id") String id);
}

2.2 创建 Retrofit 实例

OkHttpClient okHttpClient1 = new OkHttpClient.Builder()
        .addInterceptor(new MineIntercept())
        .build();
Retrofit retrofit1 = new Retrofit.Builder()
    .baseUrl("http://172.16.49.173:8080")
    .addConverterFactory(ScalarsConverterFactory.create())
    .client(okHttpClient1)
    .build();
CreditService creditService = retrofit1.create(CreditService.class);

.addInterceptor: 传入参数为拦截器,可省略.以下是我自定义拦截器,用于拦截并打印日志

public class MineIntercept implements Interceptor {
    @Override
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request();
        Response response = chain.proceed(request);
        Log.i("XXX", "intercept request ->  " + request);
        Log.i("XXX", "intercept response ->  " + response);
        return response;
    }
}

.baseUrl: 访问的是 基地址(必须以 / 结尾),在请求接口的GET注解中是我真正要访问的地址,因为 @Query 中的id,所以最终的访问地址如下
http://172.16.49.173:8080/findStudentById?id=传入的id值
.addConverterFactory: 对服务器的数据进行解析
eg: addConverterFactory(GsonConverterFactory.create()) 发可以将返回的json数据直接解析为一个实体类
2.3 创建网络请求接口的实例

CreditService creditService = retrofit1.create(CreditService.class);

2.4 发送网络请求

creditService.getStudentById(id).enqueue(new retrofit2.Callback<String>() {
    @Override
    public void onResponse(retrofit2.Call<String> call, retrofit2.Response<String> response) {
        String res = response.body();
        Log.i(TAG, "res -> " + res);
        tvCredit.setText(res);
    }

    @Override
    public void onFailure(retrofit2.Call<String> call, Throwable t) {

    }
});

3 Get请求注解
3.1 @GET
上诉例子即为 GET 实例,其最终的访问地址为 :
http://172.16.49.173:8080/findStudentById?id=传入的id值
@Query 该注解就是再getUserInfo 后面添加? 并以 id=传来的参数userId 的形式拼接 id
3.2 @QueryMap
定义请求方法,返回值为 Activity 的list集合

@GET("/findAllActivity")
Call<List<Activity>> getArticalInfo(@QueryMap Map<String, String> params);

异步请求获取数据:

OkHttpClient okHttpClient = new OkHttpClient.Builder().build();
Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("http://172.16.49.173:8080")
        .addConverterFactory(GsonConverterFactory.create())
        .client(okHttpClient).build();
CreditService creditService1 = retrofit.create(CreditService.class);
creditService1.findAllActivity(list).enqueue(new retrofit2.Callback<List<Activity>>() {
    @Override
    public void onResponse(retrofit2.Call<List<Activity>> call, retrofit2.Response<List<Activity>> response) {
        List<Activity> activityList = response.body();

        for (Activity i : activityList){
            Log.i(TAG, i.toString());
        }
    }
    public void onFailure(retrofit2.Call<List<Activity>> call, Throwable t) {

    }
});

注:从此以下,demo语言为kotlin
3.3 @Path
当访问的参数由 某个参数 动态凭借而成,
当不知道返回的类型是啥玩意,可以返回 ResponseBody 类型

@GET("/{id}/pathDemo")
fun pathDemo(@Path("id") id : Int) : ResponseBody

为使用方便,用 kotlin 将 Retrofit 进行了简单的封装,需传入 baseUrl

fun getCreditServiceInstance(baseUrl : String) : CreditService? {

    var creditService : CreditService? = null

    if (creditService == null){
        val okHttpClient = OkHttpClient.Builder().build();
        val retrofit = Retrofit.Builder()
            .baseUrl(baseUrl)
            .addConverterFactory(GsonConverterFactory.create())
            .client(okHttpClient)
            .build();
        creditService = retrofit.create(CreditService::class.java)
    }

    return creditService;
}

3.4 @Url
当访问的地址不只是在基地址中动态的改变几个参数什么的,而是整个地址都需要发生变化,甚至基类地址也需变化,则需用到 @Url

@GET
fun getOtherUrl(@Url url: String ,@Query("id") id : String?) : Call<String>
getCreditServiceInstance(baseUrl)?.getOtherUrl(baseUrl + "/demo","1234")?.enqueue(object : Callback<String>{
    override fun onResponse(call: Call<String>, response: Response<String>) {
        Log.i("XXX", "res is ->> " + response.body())
        tvOtherUrl?.text = response.body().toString()
    }

    override fun onFailure(call: Call<String>, t: Throwable) {
    }

})

注:虽然说最终访问的地址与原先的baseUrl无关,但是baseUrl还是要以 http:// https://,并且后面至少要跟一个字母或者其他东西,不然就会报错。
3.5 @Header
该请求是用于配置HTTP请求的 Header。
4.POST请求注解
对比GET中的请求
@Fileld == @Query
@FieldMap == @QueryMap
@Body == @Url

4.1 @FormUrlEncoded
@FormUrlEncoded 用于修饰 @Field ,@FieldMap 注解,将自动将请求参数的类型调整为 :
application/x-www-form-urlencoded
4.2 @Field
用法同 @Query
区别在于,get请求参数在 url 路径后面,post 在请求的主体中

注:记得加上 @FormUrlEncoded ,否则会报错
4.3 @FieldMap
用法同 @QueryMap
4.4 @Body
@FieldMap 注解适合多个相同类型参数的传递

@POST("/demo/find")
fun getNextStudent(@Body student: Student) : Call<Student>

注: @FormUrlEncoded 和 @Body 不可以同时使用,否则会报错:
IllegalArgumentException:@Body parameters cannot be used with form or multi-part encoding
因为 @FormUrlEncoded , @Body , @Maltiipart 都是用来指定数据发送的方式,故不能同时使用
reason:Content-Type 位于Http 协议的消息头,用于表示具体请求的媒体类型。
application/x-www-form-urlencoded: 请求以键值对的形式加到 url 的后面
**application/json:**标识消息主体是序列化的json字符串
**multipart/form-data :**多用于文件的上传

使用 @Body 必须在创建 Retrofit 实例时,必须添加 ,.addConverterFactory(GsonConverterFactory.create()) 用于序列化实体类

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值