
我们所面临的JSON数据结构如下所示:
[
[
"S#",
"Name of Minister",
"Portfolio",
"Contact #",
"PRO Name",
"PRO Contact",
"PRO Contact #"
],
[
"1",
"Mr. Mohammad Ali Saif",
"Information and PRs",
"9212894",
"Mr. Rizwan Malik",
"0345-",
""
],
[
"2",
"Mr. Abdul Karim",
"Industries",
"9213859",
"Mr. Khan Sarwar",
"0333-",
"abdulkarim.png"
]
]这个JSON本质上是一个List<List<String>>。其特殊之处在于:
由于这种结构并非典型的键值对对象数组,jsonschema2pojo等工具通常无法自动识别并生成带有有意义字段名的POJO。因此,我们需要采用自定义反序列化的方式来解决这个问题。
为了更好地在Java代码中操作这些数据,我们将为每一行数据设计一个POJO(Plain Old Java Object)。根据JSON的首行标题,我们可以创建一个Minister类,包含相应的字段:
import com.google.gson.annotations.SerializedName;
public class Minister {
@SerializedName("S#") // 使用SerializedName注解映射JSON中的特殊字符字段
private String serialNumber;
@SerializedName("Name of Minister")
private String name;
private String portfolio;
@SerializedName("Contact #")
private String contactNumber;
@SerializedName("PRO Name")
private String proName;
@SerializedName("PRO Contact")
private String proContact;
@SerializedName("PRO Contact #")
private String proContactNumber;
// 构造函数
public Minister(String serialNumber, String name, String portfolio, String contactNumber,
String proName, String proContact, String proContactNumber) {
this.serialNumber = serialNumber;
this.name = name;
this.portfolio = portfolio;
this.contactNumber = contactNumber;
this.proName = proName;
this.proContact = proContact;
this.proContactNumber = proContactNumber;
}
// Getter和Setter方法 (此处省略,实际项目中应添加)
// 例如:
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Minister{" +
"serialNumber='" + serialNumber + '\'' +
", name='" + name + '\'' +
", portfolio='" + portfolio + '\'' +
", contactNumber='" + contactNumber + '\'' +
", proName='" + proName + '\'' +
", proContact='" + proContact + '\'' +
", proContactNumber='" + proContactNumber + '\'' +
'}';
}
}注意事项:
由于Retrofit2通常与Gson、Jackson或Moshi等JSON处理库配合使用,我们将以Gson为例,实现一个自定义的反序列化器来解析上述特殊JSON结构。
首先,确保你的项目中已添加Gson和Retrofit的依赖:
// build.gradle (Module: app)
dependencies {
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter-gson:2.9.0'
implementation 'com.google.code.gson:gson:2.10.1'
}接下来,创建MinisterListDeserializer类,它将负责将整个JSON数组反序列化为List<Minister>:
import com.google.gson.*;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;
public class MinisterListDeserializer implements JsonDeserializer<List<Minister>> {
@Override
public List<Minister> deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
List<Minister> ministers = new ArrayList<>();
// 确保传入的是一个JSON数组
if (!json.isJsonArray()) {
throw new JsonParseException("Expected a JSON array but got: " + json.getClass().getName());
}
JsonArray jsonArray = json.getAsJsonArray();
// 至少需要有标题行和一行数据
if (jsonArray.size() < 2) {
// 如果只有标题行或更少,则返回空列表或抛出异常
return ministers;
}
// 提取标题行
JsonArray headerRow = jsonArray.get(0).getAsJsonArray();
List<String> headers = new ArrayList<>();
for (JsonElement headerElement : headerRow) {
headers.add(headerElement.getAsString());
}
// 遍历数据行 (从索引1开始)
for (int i = 1; i < jsonArray.size(); i++) {
JsonArray dataRow = jsonArray.get(i).getAsJsonArray();
// 确保数据行与标题行长度匹配,或者至少不越界
if (dataRow.size() != headers.size()) {
// 可以选择跳过不匹配的行,或者抛出异常
System.err.println("Warning: Data row at index " + i + " has " + dataRow.size() +
" elements, but expected " + headers.size() + " elements based on headers. Skipping this row.");
continue;
}
// 根据标题和数据创建Minister对象
String serialNumber = null;
String name = null;
String portfolio = null;
String contactNumber = null;
String proName = null;
String proContact = null;
String proContactNumber = null;
for (int j = 0; j < headers.size(); j++) {
String header = headers.get(j);
String value = dataRow.get(j).getAsString();
switch (header) {
case "S#":
serialNumber = value;
break;
case "Name of Minister":
name = value;
break;
case "Portfolio":
portfolio = value;
break;
case "Contact #":
contactNumber = value;
break;
case "PRO Name":
proName = value;
break;
case "PRO Contact":
proContact = value;
break;
case "PRO Contact #":
proContactNumber = value;
break;
// 如果有其他字段,可以在这里添加case
default:
// 忽略未知字段或记录警告
break;
}
}
ministers.add(new Minister(serialNumber, name, portfolio, contactNumber, proName, proContact, proContactNumber));
}
return ministers;
}
}反序列化逻辑详解:
现在,我们将这个自定义反序列化器集成到Retrofit2中。
配置GsonBuilder: 创建一个Gson实例,并通过GsonBuilder注册MinisterListDeserializer。
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
import java.util.List;
public class RetrofitClient {
private static final String BASE_URL = "http://your.api.base.url/"; // 替换为你的API基地址
private static Retrofit retrofit = null;
public static Retrofit getClient() {
if (retrofit == null) {
// 注册自定义反序列化器
Gson gson = new GsonBuilder()
.registerTypeAdapter(new com.google.common.reflect.TypeToken<List<Minister>>(){}.getType(), new MinisterListDeserializer())
.create();
retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.addConverterFactory(GsonConverterFactory.create(gson)) // 使用配置好的Gson
.build();
}
return retrofit;
}
}注意: new com.google.common.reflect.TypeToken<List<Minister>>(){}.getType() 用于获取List<Minister>的泛型类型,这在Java的泛型擦除机制下是必要的。你需要添加Guava库的依赖:implementation 'com.google.guava:guava:32.1.3-android'。如果不想引入Guava,也可以直接使用Type type = new TypeToken<List<Minister>>() {}.getType();,但需要确保TypeToken是com.google.gson.reflect.TypeToken。
定义API服务接口: 创建一个Retrofit服务接口,定义获取数据的方法。
import retrofit2.Call;
import retrofit2.http.GET;
import java.util.List;
public interface ApiService {
@GET("your_endpoint_here") // 替换为你的API端点
Call<List<Minister>> getMinisters();
}现在,你可以在你的应用程序中调用这个API服务来获取并使用Minister对象的列表了:
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import java.util.List;
public class MainActivity { // 示例,实际可能在Activity或ViewModel中
public void fetchData() {
ApiService apiService = RetrofitClient.getClient().create(ApiService.class);
Call<List<Minister>> call = apiService.getMinisters();
call.enqueue(new Callback<List<Minister>>() {
@Override
public void onResponse(Call<List<Minister>> call, Response<List<Minister>> response) {
if (response.isSuccessful() && response.body() != null) {
List<Minister> ministers = response.body();
for (Minister minister : ministers) {
System.out.println(minister.toString());
}
// 在这里处理获取到的Minister列表数据
} else {
System.err.println("API call failed: " + response.code() + " " + response.message());
}
}
@Override
public void onFailure(Call<List<Minister>> call, Throwable t) {
System.err.println("Network error: " + t.getMessage());
t.printStackTrace();
}
});
}
public static void main(String[] args) {
// 假设在非Android环境运行,RetrofitClient需要配置好MockServer或真实URL
// 在Android中,通常在Activity或Fragment的生命周期方法中调用fetchData()
new MainActivity().fetchData();
}
}处理非标准JSON结构是API集成中常见的挑战之一。通过本教程介绍的自定义反序列化器方法,我们能够灵活、精确地将任意复杂或非标准的JSON数据映射到我们定义的Java POJO模型中。这不仅确保了
以上就是处理非标准JSON数组与Retrofit2:定制化POJO模型生成与反序列化的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号