G相关文章

Android网络编程(一)HTTP协议原理

Android网络编程(二)HttpClient与HttpURLConnection

Android网络编程(三)Volley用法全解析

Android网络编程(四)从源码解析volley

Android网络编程(五)OkHttp2.x用法全解析

Android网络编程(六)OkHttp3用法全解析

Android网络编程(七)源码解析OkHttp前篇[请求网络]

Android网络编程(八)源码解析OkHttp后篇[复用连接池]

Android网络编程(九)Retrofit2前篇[基本使用]

前言

在上一篇Android网络编程(九)Retrofit2前篇[基本使用]中我们了解了Retrofit的最基本的GET方式访问网络的写法以及请求参数的简单介绍。这一篇我们来详细的了解Retrofit的注解。

1.GET请求访问网络

动态配置URL地址:@Path

Retrofit提供了很多的请求参数注解,使得请求网路时更加便捷。在这里我们仍旧访问淘宝ip库。其中,@Path用来动态的配置URL地址。请求网络接口代码如下所示。

  1. public interface IpServiceForPath {
  2. @GET("{path}/getIpInfo.php?ip=59.108.54.37")
  3. Call<IpModel> getIpMsg(@Path("path") String path);
  4. }

在GET注解中包含了{path},它对应着@Path注解中的”path”,而用来替换{path}的正是需要传入的 “String path”的值。接下来请求网络的代码如下所示。

  1. String url = "http://ip.taobao.com/";
  2. Retrofit retrofit = new Retrofit.Builder()
  3. .baseUrl(url)
  4. .addConverterFactory(GsonConverterFactory.create())
  5. .build();
  6. IpServiceForPath ipService = retrofit.create(IpServiceForPath.class);
  7. Call<IpModel>call=ipService.getIpMsg("service");//1
  8. call.enqueue(new Callback<IpModel>() {
  9. @Override
  10. public void onResponse(Call<IpModel> call, Response<IpModel> response) {
  11. String country= response.body().getData().getCountry();
  12. Toast.makeText(getApplicationContext(),country,Toast.LENGTH_SHORT).show();
  13. }
  14. @Override
  15. public void onFailure(Call<IpModel> call, Throwable t) {
  16. }
  17. });

在注释1处,传入”service”来替换 @GET注解中的{path}的值。

动态指定查询条件:@Query与@QueryMap

在上一篇中我们用@Query来动态的替换ip地址为了能更方便的得到该ip所对应的地理信息:

  1. public interface IpServiceForQuery{
  2. @GET("getIpInfo.php")
  3. Call<IpModel> getIpMsg(@Query("ip")String ip);
  4. }

但是在网络请求中一般为了更精确的查找到我们所需要的数据,需要传入很多的查询参数,如果用@Query会比较麻烦,这时我们可以采用@QueryMap,将所有的参数集成在一个Map统一传递:

  1. public interface IpServiceForQueryMap {
  2. @GET("getIpInfo.php")
  3. Call<IpModel> getIpMsg(@QueryMap Map<String, String> options);
  4. }

2.POST请求访问网络

传输数据类型为键值对:@Field

传输数据类型为键值对,这是我们最常用的POST请求数据类型,淘宝ip库支持数据类型为键值对的POST请求:

  1. public interface IpServiceForPost {
  2. @FormUrlEncoded
  3. @POST("getIpInfo.php")
  4. Call<IpModel> getIpMsg(@Field("ip") String first);
  5. }

首先用到@FormUrlEncoded注解来标明这是一个表单请求,然后在getIpMsg方法中使用@Field注解来标示所对应的String类型数据的键,从而组成一组键值对进行传递。接下来请求网络的代码如下所示。

  1. String url = "http://ip.taobao.com/service/";
  2. Retrofit retrofit = new Retrofit.Builder()
  3. .baseUrl(url)
  4. .addConverterFactory(GsonConverterFactory.create())
  5. .build();
  6. IpServiceForPost ipService = retrofit.create(IpServiceForPost.class);
  7. Call<IpModel>call=ipService.getIpMsg("59.108.54.37");
  8. call.enqueue(new Callback<IpModel>() {
  9. @Override
  10. public void onResponse(Call<IpModel> call, Response<IpModel> response) {
  11. String country= response.body().getData().getCountry();
  12. Toast.makeText(getApplicationContext(),country,Toast.LENGTH_SHORT).show();
  13. }
  14. @Override
  15. public void onFailure(Call<IpModel> call, Throwable t) {
  16. }
  17. });

传输数据类型Json字符串:@Body

我们也可以用POST方式将Json字符串作为请求体发送到服务器,请求网络接口代码为:

  1. public interface IpServiceForPostBody {
  2. @POST("getIpInfo.php")
  3. Call<IpModel> getIpMsg(@Body Ip ip);
  4. }

用@Body这个注解标识参数对象即可,retrofit会将Ip对象转换为字符串。

  1. public class Ip {
  2. private String ip;
  3. public Ip(String ip) {
  4. this.ip = ip;
  5. }
  6. }

请求网络的代码基本上都是一致的:

  1. String url = "http://ip.taobao.com/service/";
  2. Retrofit retrofit = new Retrofit.Builder()
  3. .baseUrl(url)
  4. .addConverterFactory(GsonConverterFactory.create())
  5. .build();
  6. IpServiceForPostBody ipService = retrofit.create(IpServiceForPostBody.class);
  7. Call<IpModel>call=ipService.getIpMsg(new Ip(ip));
  8. call.enqueue(new Callback<IpModel>() {
  9. @Override
  10. public void onResponse(Call<IpModel> call, Response<IpModel> response) {
  11. String country= response.body().getData().getCountry();
  12. Log.i("wangshu","country"+country);
  13. Toast.makeText(getApplicationContext(),country,Toast.LENGTH_SHORT).show();
  14. }
  15. @Override
  16. public void onFailure(Call<IpModel> call, Throwable t) {
  17. }
  18. });
  19. }

运行程序用Fiddler抓包,如下图所示。

可以看到请求数据是一个Json字符串,因为淘宝ip库并不支持此类型所以不会返回我们需要的地理信息数据。

单个文件上传:@Part

  1. public interface UploadFileForPart {
  2. @Multipart
  3. @POST("user/photo")
  4. Call<User> updateUser(@Part MultipartBody.Part photo, @Part("description") RequestBody description);
  5. }

Multipart注解表示允许多个@Part,updateUser方法第一个参数是准备上传的图片文件,使用了MultipartBody.Part类型,另一个参数是RequestBody类型,它用来传递简单的键值对。请求网络代码如下所示。

  1. ...
  2. File file = new File(Environment.getExternalStorageDirectory(), "wangshu.png");
  3. RequestBody photoRequestBody = RequestBody.create(MediaType.parse("image/png"), file);
  4. MultipartBody.Part photo = MultipartBody.Part.createFormData("photos", "wangshu.png", photoRequestBody);
  5. UploadFileForPart uploadFile = retrofit.create(UploadFileForPart.class);
  6. Call<User> call = uploadFile.updateUser(photo, RequestBody.create(null, "wangshu"));
  7. ...

多个文件上传:@PartMap

  1. @Multipart
  2. @POST("user/photo")
  3. Call<User> updateUser(@PartMap Map<String, RequestBody> photos, @Part("description") RequestBody description);

和单文件上传是类似的,只是使用Map封装了上传的文件,并用@PartMap注解来标示起来。其他的都一样,这里就不赘述了。

3.消息报头Header

Http请求中,为了防止攻击或是过滤掉不安全的访问或是添加特殊加密的访问等等,用来减轻服务器的压力和保证请求的安全,通常都会在消息报头中携带一些特殊的消息头处理。Retrofit也提供了@Header来添加消息报头。添加消息报头有两种方式,一种是静态的,另一种是动态的,先来看静态的方式,如下所示。

  1. interface SomeService {
  2. @GET("some/endpoint")
  3. @Headers("Accept-Encoding: application/json")
  4. Call<ResponseBody> getCarType();
  5. }

使用@Headers注解添加消息报头,如果想要添加多个消息报头,则可以使用{}包含起来:

  1. interface SomeService {
  2. @GET("some/endpoint")
  3. @Headers({
  4. "Accept-Encoding: application/json",
  5. "User-Agent: MoonRetrofit"
  6. })
  7. Call<ResponseBody> getCarType();
  8. }

动态的方式添加消息报头如下所示。

  1. interface SomeService {
  2. @GET("some/endpoint")
  3. Call<ResponseBody> getCarType(
  4. @Header("Location") String location);
  5. }

使用@Header注解,可以通过调用getCarType方法来动态的添加消息报头。

github源码下载


欢迎关注我的微信公众号,第一时间获得博客更新提醒,以及更多成体系的Android相关原创技术干货。

扫一扫下方二维码或者长按识别二维码,即可关注。

Android网络编程(十)Retrofit2后篇[注解]的更多相关文章

  1. Android网络编程(一)HTTP协议原理

    相关文章 Android网络编程(一)HTTP协议原理 Android网络编程(二)HttpClient与HttpURLConnection Android网络编程(三)Volley使用方法全解析 A ...

  2. 1.Android网络编程-HTML介绍

    1.HTML介绍 超文本标记语言(HyperText Markup Language,简称:HTML)是一种用于创建网页的标准标记语言. 在Eclipse下则可以使用自带的浏览器浏览html: 2.H ...

  3. Android网络编程系列 一 Socket抽象层

     在<Android网络编程>系列文章中,前面已经将Java的通信底层大致的描述了,在我们了解了TCP/IP通信族架构及其原理,接下来我们就开始来了解基于tcp/ip协议层的Socket抽 ...

  4. Android网络编程系列 一 TCP/IP协议族

    在学习和使用Android网路编程时,我们接触的仅仅是上层协议和接口如Apache的httpclient或者Android自带的httpURlconnection等等.对于这些接口的底层实现我们也有必 ...

  5. Android网络编程要学的东西与Http协议学习

    本节引言: 本节开始我们来学习Android网络编程相关的一些东西:Android端网络编程是要干嘛?http协议的学习,使用自带扣脚Json解析类解析Json,XML解析常用的几种方式,HttpUr ...

  6. Android网络编程随想录(1)

    本系列文章对整个Android网络编程进行了总结,包括基本的TCP/IP协议,HTTP协议,HTTPS协议,HttpClient,UrlConnection,一些网络通信的库到棉花糖新加入的OKHTT ...

  7. Android网络编程只局域网传输文件

    Android网络编程之局域网传输文件: 首先创建一个socket管理类,该类是传输文件的核心类,主要用来发送文件和接收文件 具体代码如下: package com.jiao.filesend; im ...

  8. Android网络编程基础

    Android网络编程只TCP通信 TCP 服务器端工作的主要步骤如下.步骤1 调用ServerSocket(int port)创建一个ServerSocket,并绑定到指定端口上.步骤2 调用acc ...

  9. Android 网络编程 Socket

    1.服务端开发 创建一个Java程序 public class MyServer { // 定义保存所有的Socket,与客户端建立连接得到一个Socket public static List< ...

随机推荐

  1. ubuntu下创建第一个rails应用程序

    一.创建一个新的应用程序 在控制台输入 > rails new  demo create create README.rdoc create Rakefile create config.ru ...

  2. nginx源代码分析--进程间通信机制 &amp; 同步机制

    Nginx源代码分析-进程间通信机制 从nginx的进程模型能够知道.master进程和worker进程须要通信,nginx中通信的方式有套接字.共享内存.信号.对于master进程,从外部接受信号, ...

  3. POJ1151 Atlantis 扫描线算法

    题目大意 给出几个矩形对角端点坐标,求这些矩形整体覆盖的面积. 扫描线算法 整个平面被每个矩形的水平边所在直线(以后简称“水平线”)分成了几个部分,而整体覆盖面积则为每相邻的两个水平线间夹的长度(以后 ...

  4. 【Codeforces 258E】 Devu and Flowers

    [题目链接] http://codeforces.com/contest/451/problem/E [算法] 容斥原理 [代码] #include<bits/stdc++.h> usin ...

  5. [JavaEE] DWR入门教程

    DWR(Direct Web Remoting)是一个WEB远程调用框架.利用这个框架可以让AJAX开发变得很简单.利用DWR可以在客户端利用JavaScript直接调用服务端的Java方法并返回值给 ...

  6. testNG中方法的调用顺序

    今天在执行selnium的test case时,总是遇到空指针错误.但是以前也有run成功过,然后换了各种方法定位元素,都失败了,所以怀疑应该不是元素定位不到的问题,所以可能是method之间有依赖, ...

  7. 本地文件SVN和 vs svn 插件的使用!!

    比如:客服端是用的TortoiseSVN-1.7.7.22907-x64-svn-1.7.5.msi 里面svn 版本是1.7.5 vs里的插件 也需要svn 版本是1.7.5 对应的AnkhSvn- ...

  8. Redux入门

    Redux入门 本文转载自:众成翻译 译者:miaoYu 链接:http://www.zcfy.cc/article/4728 原文:https://bumbu.github.io/simple-re ...

  9. [ Linux ] [ OS ] [ memory ] Linux 如何查看系統硬體的記憶體(RAM)資訊

    cat /proc/meminfo https://blog.longwin.com.tw/2013/05/linux-ram-memory-info-2013/

  10. 百度map API

    1.做demo用的 http://developer.baidu.com/map/jsdemo.htm demo代码(外部使用的话需要提供密钥): <!DOCTYPE html> < ...