0

我的问题是,PHP 对数组进行不同的编码,这取决于它们是否是连续的并且是否以零索引开头。例子:

$arr = array(array(12), array(13));
===> [[12], [13]]

$arr = array("0" => array(12), "1" => array(13));
===> [[12], [13]]

$arr = array("0" => array(12), "2" => array(13));
===> {"0": [12], "2": [13]}

为什么第三个有这么大的不同?

第一个示例生成一个列表列表,第三个示例生成一个带有列表的对象。我需要将所有这些转换为 Java 的Map<Integer, List<Double>>. 这是我能在 Java 中为这些 PHP 对象找到的最通用的数据类型。我正在使用谷歌的 Gson。但是,由于这些示例产生了不同类型的对象,我不能只将其读入地图。我必须先检查它是否有索引,然后一个一个地添加到自定义地图中。请查看“必须有更好的方法来完成这部分”的行。这是我的代码:

import java.lang.StringBuilder;
import com.google.gson.*;
import com.google.gson.reflect.*;
import java.util.Map;
import java.util.HashMap;
import java.util.List;

public class Saving {

    public static void main(String[] args) {

        String json = "[[12], [13]]";
        json = json.trim();
        Map<Integer, List<Double>> fuelSavings = null;

        // such a cluster****
        if(json.startsWith("[[")) { // THERE HAS TO BE A BETTER WAY TO DO THIS PART
        // ANY WAY I CAN AVOID THIS ENTIRE IF CONDITION

            //implicit keys
            fuelSavings = new HashMap<Integer, List<Double>>();
            List<List<Double>> temporaryList = new Gson().fromJson(json, new TypeToken<List<List<Double>>>(){}.getType());
            int index = 0;
            for(List<Double> temporaryListMember: temporaryList) {
                fuelSavings.put(index, temporaryListMember);
                index++;
            }

        } else {

            // explicit keys
            // THIS PART IS PERFECT
            fuelSavings = new Gson().fromJson(json, new TypeToken<Map<Integer, List<Double>>>(){}.getType());

        }

        System.out.println(fuelSavings);

    }

}

任何帮助表示赞赏,谢谢!

4

2 回答 2

0

很有意思!

据我了解,PHP 服务器将其数据作为编码为 JSON 的有序映射提供服务,但有时运气不好,它的键是连续整数,这使得将其json_encode()理解为一个简单的数组并为输出的 JSON 使用不同的格式。

据我了解,所有 PHP 数组都是有序映射。但是当一个数组有连续的整数键时,会json_encode()生成一个没有键的数组,而不是“JSON 对象”(映射)。这破坏了不期望数组而不是“对象”的客户端代码。

JsonArray令人难以置信的是,我发现没有将 a 转换为 a 的现有 Java 函数,JsonObject如果我们尝试将数组作为对象,有时甚至可能会遇到一些异常!天哪,aJsonArray只是JsonObject带有隐式连续整数键的 a,它应该自动转换!:p

我没有尝试过,但也许JsonObject::entrySet()可以解决它。它返回 a ,如果 Integer 自动解析 String 并且可以使用适当的 转换Set<Map.Entry<String,JsonElement>>,它可能会转换为。Map<Integer, AnyObject>JsonElementTypeToken

于 2012-12-26T22:41:00.297 回答
0

您在 PHP 中所做的事情是两件不同的事情。PHP 中的数组用作数组、映射、列表。

array(array(12), array(13));即使在 PHP中array("0" => array(12), "2" => array(13));也不相同。

在 Java 中使用 Gson 需要做的是将 json 解析为 String 并检查主对象是 aMap还是 aList然后给出建议。如果您加载了一个列表,那么您知道它是一个,array(array(12), array(13));否则它是另一个。

你不能从 PHP 端做点什么来生成它总是一样的吗?

编辑:

如果你不能从 PHP 端做任何事情,那么你的检查将与实例化JsonParser解析然后检查是否JsonObject.isJsonArray()返回 true 相同,除了它最终会更快但你不会依赖 gson 库。

Map<Integer, List<Double>> fuelSavings = null;
JsonElement jElement = new JsonParser().parse("[[12], [13]]");
JsonObject jObject = jelement.getAsJsonObject();
if (jObject.isJsonArray()) {
    fuelSavings = new HashMap<Integer, List<Double>>();
    List<List<Double>> temporaryList = new Gson().fromJson(jElement, new TypeToken<List<List<Double>>>(){}.getType());
    int index = 0;
    for(List<Double> temporaryListMember: temporaryList) {
        fuelSavings.put(index++, temporaryListMember);
    }
} else /* this is a map */ {
    fuelSavings = new Gson().fromJson(jElement, new TypeToken<Map<Integer, List<Double>>>(){}.getType());
}
于 2012-09-11T19:12:51.550 回答