Java REST API:高效获取JSON数组的教程
知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个文章开发实战,手把手教大家学习《Java REST API:高效获取JSON数组的教程 》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!

本教程详细介绍了如何在Java应用程序中从REST API获取JSON数组。文章探讨了两种主要方法:使用低级别的HttpURLConnection进行直接HTTP请求,以及利用更现代、功能强大的Retrofit和RxJava库。教程提供了完整的代码示例,并强调了JSON解析、POJO定义以及错误处理等关键实践,旨在帮助开发者构建健壮的API客户端。
在现代应用开发中,从RESTful API获取数据是常见的任务。很多时候,API会返回一个对象列表,即JSON数组,例如搜索结果、商品列表或用户列表。正确地解析这些JSON数组并将其映射到Java对象是构建健壮API客户端的关键。本文将深入探讨两种主流的Java实现方式:使用标准库HttpURLConnection和更高级的Retrofit结合RxJava。
1. 理解从REST API获取JSON数组的需求
当API返回的数据结构是[...]而不是{...}时,意味着我们期望接收一个JSON对象数组。例如,一个查询化妆品列表的API可能会返回如下格式:
[
{
"_id":"6353e8fe5d63726919402cec",
"code":"0000016615656",
"name":"Lipstick",
"brand":"BrandX"
},
{
"_id":"6353e8fe5d63726919402ced",
"code":"0000016615657",
"name":"Mascara",
"brand":"BrandY"
}
]我们的Java应用程序需要能够识别并解析这种数组结构,将其转换为List
2. 准备工作:定义数据模型(POJO)
无论是哪种API客户端,首先都需要定义一个Java Plain Old Java Object (POJO) 来映射JSON数据结构。这个POJO应该包含JSON对象中的所有字段。为了更好地与JSON字段匹配,特别是当Java字段名与JSON字段名不一致时,可以使用Gson库的@SerializedName注解。
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
import java.io.Serializable;
public class Cosmetic implements Serializable {
@SerializedName("_id")
@Expose
private String id;
@SerializedName("code")
@Expose
private String code;
@SerializedName("name")
@Expose
private String name;
@SerializedName("brand")
@Expose
private String brand;
// 构造函数
public Cosmetic(String id, String code, String name, String brand) {
this.id = id;
this.code = code;
this.name = name;
this.brand = brand;
}
// Getters and Setters
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getBrand() {
return brand;
}
public void setBrand(String brand) {
this.brand = brand;
}
@Override
public String toString() {
return "Cosmetic{" +
"id='" + id + '\'' +
", code='" + code + '\'' +
", name='" + name + '\'' +
", brand='" + brand + '\'' +
'}';
}
}3. 方法一:使用 HttpURLConnection 获取JSON数组
HttpURLConnection是Java标准库提供的一个低级别HTTP客户端,可以直接用于发送HTTP请求和接收响应。这种方法适用于简单的请求或不希望引入额外依赖的项目。
3.1 原理概述
使用HttpURLConnection获取JSON数组的基本流程包括:
- 构建URL对象。
- 打开HttpURLConnection连接。
- 设置请求方法(GET)。
- 获取输入流并读取服务器响应。
- 将响应内容(JSON字符串)解析为Java对象列表。
3.2 示例代码
为了解析JSON字符串,我们将使用Gson库。请确保在pom.xml或build.gradle中添加Gson依赖:
Maven:
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
<version>2.10.1</version>
</dependency>Gradle:
implementation 'com.google.code.gson:gson:2.10.1'
以下是使用HttpURLConnection获取JSON数组的示例:
import com.google.gson.Gson;
import com.google.gson.reflect.TypeToken;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.lang.reflect.Type;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.Collections;
import java.util.List;
public class HttpUrlConnectionApiExample {
private static final String API_URL = "http://your-api-base-url/prod/beauty/search"; // 替换为你的API地址
public static List<Cosmetic> getCosmeticsArray(String queryField, String queryValue) {
HttpURLConnection connection = null;
BufferedReader reader = null;
try {
// 1. 构建URL对象,添加查询参数
URL url = new URL(API_URL + "?" + queryField + "=" + queryValue);
connection = (HttpURLConnection) url.openConnection();
// 2. 设置请求方法
connection.setRequestMethod("GET");
connection.setRequestProperty("Accept", "application/json"); // 告知服务器我们期望JSON响应
connection.setConnectTimeout(5000); // 连接超时
connection.setReadTimeout(5000); // 读取超时
// 3. 获取响应码
int responseCode = connection.getResponseCode();
if (responseCode == HttpURLConnection.HTTP_OK) { // 200 OK
// 4. 读取响应流
reader = new BufferedReader(new InputStreamReader(connection.getInputStream()));
StringBuilder response = new StringBuilder();
String line;
while ((line = reader.readLine()) != null) {
response.append(line);
}
// 5. 使用Gson解析JSON数组
Gson gson = new Gson();
// 对于泛型类型(如List<Cosmetic>),需要使用TypeToken来获取正确的Type
Type cosmeticListType = new TypeToken<List<Cosmetic>>() {}.getType();
return gson.fromJson(response.toString(), cosmeticListType);
} else {
System.err.println("API请求失败,响应码: " + responseCode);
// 可以读取错误流并打印
if (connection.getErrorStream() != null) {
reader = new BufferedReader(new InputStreamReader(connection.getErrorStream()));
StringBuilder errorResponse = new StringBuilder();
String line;
while ((line = reader.readLine()) != null) {
errorResponse.append(line);
}
System.err.println("错误详情: " + errorResponse.toString());
}
}
} catch (IOException e) {
System.err.println("网络连接或读取数据时发生错误: " + e.getMessage());
e.printStackTrace();
} finally {
// 确保关闭资源
if (reader != null) {
try {
reader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (connection != null) {
connection.disconnect();
}
}
return Collections.emptyList(); // 返回空列表表示失败或无结果
}
public static void main(String[] args) {
// 假设API查询参数是 "name" 和 "Lipstick"
List<Cosmetic> cosmetics = getCosmeticsArray("name", "Lipstick");
if (!cosmetics.isEmpty()) {
System.out.println("成功获取到化妆品列表:");
for (Cosmetic cosmetic : cosmetics) {
System.out.println(cosmetic);
}
} else {
System.out.println("未获取到化妆品或请求失败。");
}
}
}3.3 注意事项
- 手动资源管理: 需要手动管理连接的打开、关闭以及输入/输出流的关闭。
- 错误处理: 必须显式检查HTTP响应码并处理各种IOException。
- 同步阻塞: HttpURLConnection默认是同步阻塞的,在主线程中直接调用会阻塞UI。在实际应用中,应将其放在单独的线程或使用ExecutorService执行。
- 安全性: HttpURLConnection本身是安全的,但如果处理不当(例如不验证SSL证书),可能会引入安全风险。这里的"安全"更多指的是代码的健壮性和对复杂场景的处理能力。
4. 方法二:利用 Retrofit 和 RxJava 获取JSON数组
Retrofit是一个类型安全的HTTP客户端,它将REST API转换为Java接口。结合RxJava,可以实现强大的异步和响应式编程模型,极大地简化API调用和结果处理。这是现代Java和Android应用中更推荐的方式。
4.1 原理概述
使用Retrofit和RxJava获取JSON数组的流程:
- 定义Retrofit服务接口: 声明一个方法,其返回类型为Single
- >或Call
- >。
- 构建Retrofit实例: 配置baseUrl、JSON转换器(如GsonConverterFactory)和Call Adapter(如RxJava2CallAdapterFactory)。
- 创建服务实例: 通过Retrofit实例创建服务接口的实现。
- 发起API调用: 调用服务方法并订阅其返回的Single或执行Call。
4.2 依赖配置
请确保在pom.xml或build.gradle中添加以下依赖:
Maven:
<dependency>
<groupId>com.squareup.retrofit2</groupId>
<artifactId>retrofit</artifactId>
<version>2.9.0</version>
</dependency>
<dependency>
<groupId>com.squareup.retrofit2</groupId>
<artifactId>converter-gson</artifactId>
<version>2.9.0</version>
</dependency>
<dependency>
<groupId>com.squareup.retrofit2</groupId>
<artifactId>adapter-rxjava2</artifactId>
<version>2.9.0</version>
</dependency>
<dependency>
<groupId>io.reactivex.rxjava2</groupId>
<artifactId>rxjava</artifactId>
<version>2.2.21</version>
</dependency>
<dependency>
<groupId>io.reactivex.rxjava2</groupId>
<artifactId>rxandroid</artifactId>
<version>2.1.1</version>
</dependency>Gradle:
implementation 'com.squareup.retrofit2:retrofit:2.9.0' implementation 'com.squareup.retrofit2:converter-gson:2.9.0' implementation 'com.squareup.retrofit2:adapter-rxjava2:2.9.0' implementation 'io.reactivex.rxjava2:rxjava:2.2.21' implementation 'io.reactivex.rxjava2:rxandroid:2.1.1' // 如果是Android项目
4.3 示例代码
首先,定义Retrofit服务接口:
import io.reactivex.Single;
import java.util.List;
import retrofit2.http.GET;
import retrofit2.http.Query;
public interface CosmeticsService {
// 获取单个Cosmetic对象的方法 (如果API支持)
@GET("/prod/beauty/{field}")
Single<Cosmetic> getByName(@Query("name") String name);
// 获取Cosmetic对象数组的方法
// 注意返回类型是 List<Cosmetic>
@GET("/prod/beauty/search") // 假设这是获取列表的API路径
Single<List<Cosmetic>> searchCosmetics(@Query("queryField") String queryField, @Query("queryValue") String queryValue);
}然后,进行API调用:
import io.reactivex.Single;
import io.reactivex.SingleObserver;
import io.reactivex.android.schedulers.AndroidSchedulers; // 仅限Android
import io.reactivex.disposables.CompositeDisposable;
import io.reactivex.disposables.Disposable;
import io.reactivex.schedulers.Schedulers;
import retrofit2.Retrofit;
import retrofit2.adapter.rxjava2.RxJava2CallAdapterFactory;
import retrofit2.converter.gson.GsonConverterFactory;
import java.util.List;
import java.util.concurrent.CountDownLatch; // 用于桌面应用模拟异步等待
public class RetrofitRxJavaApiExample {
private static final String BASE_URL = "http://your-api-base-url/"; // 替换为你的API基础URL
private Retrofit retrofit;
private CosmeticsService service;
private CompositeDisposable compositeDisposable = new CompositeDisposable();
public RetrofitRxJavaApiExample() {
retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.addConverterFactory(GsonConverterFactory.create())
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
.build();
service = retrofit.create(CosmeticsService.class);
}
public void fetchCosmeticsList(String queryField, String queryValue) {
// 调用获取Cosmetic列表的方法
Single<List<Cosmetic>> callSync = service.searchCosmetics(queryField, queryValue);
callSync.subscribeOn(Schedulers.io()) // 在IO线程执行网络请求
// .observeOn(AndroidSchedulers.mainThread()) // 如果是Android应用,切换到主线程更新UI
.observeOn(Schedulers.computation()) // 对于非Android应用,可以切换到其他线程处理结果
.onErrorReturn(throwable -> {
System.err.println("API请求发生错误: " + throwable.getMessage());
throwable.printStackTrace();
return Collections.emptyList(); // 错误时返回空列表
})
.subscribe(new SingleObserver<List<Cosmetic>>() {
@Override
public void onSubscribe(Disposable d) {
compositeDisposable.add(d); // 管理Disposable,防止内存泄漏
}
@Override
public void onSuccess(List<Cosmetic> cosmetics) {
if (!cosmetics.isEmpty()) {
System.out.println("成功获取到化妆品列表:");
for (Cosmetic cosmetic : cosmetics) {
System.out.println(cosmetic);
}
} else {
System.out.println("未获取到化妆品或结果为空。");
}
// 如果是桌面应用,需要某种机制来通知主线程操作完成
// 例如使用CountDownLatch
// latch.countDown();
}
@Override
public void onError(Throwable e) {
System.err.println("订阅处理中发生错误: " + e.getMessage());
e.printStackTrace();
// latch.countDown(); // 同样通知完成
}
});
}
// 在应用程序退出或组件销毁时调用,以清理资源
public void dispose() {
compositeDisposable.clear();
}
public static void main(String[] args) throws InterruptedException {
RetrofitRxJavaApiExample apiExample = new RetrofitRxJavaApiExample();
CountDownLatch latch = new CountDownLatch(1); // 用于模拟异步等待,在桌面应用中
// 假设API查询参数是 "name" 和 "Lipstick"
apiExample.fetchCosmeticsList("name", "Lipstick");
// 在桌面应用中,主线程可能需要等待异步操作完成
// 在Android中,通常不需要显式等待,因为UI会在回调中更新
latch.await(10, java.util.concurrent.TimeUnit.SECONDS); // 等待最多10秒
apiExample.dispose(); // 清理资源
}
}4.4 注意事项
- 类型安全: Retrofit通过接口定义提供了强大的类型安全,减少了运行时错误。
- 异步处理: RxJava提供了强大的异步编程模型,通过subscribeOn和observeOn可以轻松管理线程切换。
- 错误处理: RxJava操作符(如onErrorReturn)使得错误处理更加优雅和链式化。
- 资源管理: 使用CompositeDisposable管理Disposable对象,以防止内存泄漏,尤其是在Android应用中。
- 可测试性: 接口化的设计使得API服务更易于进行单元测试和集成测试。
5. JSON解析库的选择与使用
本文示例中均使用了Google的Gson库进行JSON解析。Gson是一个功能强大且易于使用的Java库,可以将Java对象序列化为JSON,反之亦然。
- 单个对象解析: gson.fromJson(jsonString, Cosmetic.class)
- 对象列表解析: gson.fromJson(jsonString, new TypeToken
- >() {}.getType())
TypeToken是处理Java泛型类型擦除的关键。由于Java在运行时会擦除泛型信息,List
6. 错误处理与健壮性
无论是哪种方法,健壮的API客户端都必须妥善处理各种错误情况:
- 网络连接错误: `IOException
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。
php文件怎么发布_php项目打包发布与部署流程
- 上一篇
- php文件怎么发布_php项目打包发布与部署流程
- 下一篇
- PHPSession劫持怎么防范_PHP会话安全防护措施与实践
-
- 文章 · java教程 | 1小时前 |
- Java代码风格统一技巧分享
- 107浏览 收藏
-
- 文章 · java教程 | 2小时前 | java 格式化输出 字节流 PrintStream System.out
- JavaPrintStream字节输出方法解析
- 362浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- ThreadLocalRandom提升并发效率的原理与实践
- 281浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- 身份证扫描及信息提取教程(安卓)
- 166浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- JavaCopyOnWriteArrayList与Set使用解析
- 287浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- Java线程安全用法:CopyOnWriteArrayList详解
- 136浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- Java流收集后处理:Collectors.collectingAndThen用法解析
- 249浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- staticfinal变量初始化与赋值规则解析
- 495浏览 收藏
-
- 文章 · java教程 | 4小时前 |
- 判断两个Map键是否一致的技巧
- 175浏览 收藏
-
- 文章 · java教程 | 4小时前 | java 空指针异常 空值判断 requireNonNull Objects类
- JavaObjects空值判断实用技巧
- 466浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3190次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3402次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3433次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4540次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3811次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览

