11

この質問から続けます。

次の json 配列のデシリアライズに問題があります (サイズについて申し訳ありません)。

"geometry": { "type": "Polygon", "coordinates": [ [ [ 771230.894373, 4422896.962001 ],
 [ 804804.852796, 4451159.130080 ], [ 876828.563339, 4417873.954498 ], [ 959794.979827, 
4430944.287708 ], [ 910992.515063, 4372980.866944 ], [ 932488.308736, 4357684.778349 ], 
[ 918573.372386, 4115663.286966 ], [ 834059.614976, 4013708.358795 ], [ 929360.231044, 
3833522.241529 ], [ 1008029.715188, 3776446.653183 ], [ 1061663.445852, 3533717.758754      
], [ 1035703.740599, 3519308.069656 ], [ 1095348.723766, 3396028.487184 ], [ 
1108462.159782, 3230455.268230 ], [ 1083571.121640, 3163122.508021 ], [ 1103953.720405, 
3082716.041755 ], [ 1045722.494771, 3020215.642212 ], [ 1117367.719045, 2915275.458735 
], [ 1141268.013718, 2827405.304519 ], [ 1286729.192338, 2790314.754276 ], [ 
1334329.406601, 2695307.513404 ], [ 829417.592210, 2374337.277646 ], [ 647042.870444, 
2207530.090128 ], [ 370914.873531, 2152159.656850 ], [ 346669.488436, 2173360.227237 ], 
[ 359905.375891, 2251757.174668 ], [ 199905.871774, 2309591.361246 ], [ 129963.835709, 
2361036.252651 ], [ 130208.738589, 2404106.913263 ], [ -964785.432600, 3159802.671416 
], [ -964829.960396, 3338713.127631 ], [ -851005.781060, 3424742.002477 ], [ -
616522.405653, 3491025.523892 ], [ -547749.224241, 3569019.334331 ], [ -403724.067052, 
3628920.873754 ], [ -423973.082428, 3724062.779415 ], [ -333893.350478, 3741450.793542 
], [ -317696.364567, 3774909.265404 ], [ -131414.328674, 3777826.527844 ], [ -
112467.751341, 3830221.719769 ], [ -185682.580436, 3930014.456814 ], [ -194499.084106, 
4129581.855629 ], [ -245950.952751, 4175549.526399 ], [ -42303.076294, 4287174.981681 
], [ -11222.674464, 4271148.905617 ], [ 131633.628071, 4371332.547494 ], [ 
433220.392528, 4427574.250017 ], [ 593119.709103, 4389089.571176 ], [ 719645.442339, 
4451856.882422 ], [ 771230.894373, 4422896.962001 ] ] ] }

これを json-viewer に貼り付けると、次の構造が得られます。

[geometry]
...
[coordinates] => Array
    (
        [0] => Array
            (
                [0] => Array
                    (
                        [0] => 771230.894373
                        [1] => 4422896.962001
                    )
                [1] => Array
                    (
                        [0] => 804804.852796
                        [1] => 4451159.13008
                    )
                ...
                [n] => Array
        [n] => Array

これで、座標を持つ配列を含む配列のサイズが可変になりました。したがって、Java では、このオブジェクト全体が配列であり、配列のコレクションを含み、各配列にCollection<double[]>. 何かのようなものCollection<double[]>[][].

しかし、gson はこれを受け入れません。次のエラー メッセージが表示されます。

Exception in thread "main" com.google.gson.JsonParseException: Expecting object but     
found array: 2.963610

2.963610私には配列のように見えないので、奇妙に思えます。しかし、それは多かれ少なかれ、私が道に迷うところまで私を混乱させたかもしれません...

4

4 に答える 4

19

Gson API を読んで、あなたの問題がどこから来たのか知っていると思います:

シリアル化/逆シリアル化するオブジェクトが ParameterizedType (つまり、少なくとも 1 つの型パラメーターを含み、配列である可能性がある) の場合は、toJson(Object, Type) または fromJson(String, Type) メソッドを使用する必要があります。ParameterizedType をシリアル化および逆シリアル化する例を次に示します。

Type listType = new TypeToken<LinkedList>() {}.getType();
List target = new LinkedList();
target.add("blah");

Gson gson = new Gson();
String json = gson.toJson(target, listType);
List target2 = gson.fromJson(json, listType);

知っています

Type typeOfCollectionOfFoo = new TypeToken<Collection<Foo>>(){}.getType() 

お役に立てれば。

于 2010-08-11T14:20:36.900 に答える
4

JSONのcoordinatesは3次元マトリックスです。あなたCollection<double[]>[][]は一次元を行き過ぎています。それCollection自体はすでに1次元であるため、基本的に4次元行列を宣言しました。

エラーメッセージで、Gsonは基本的に、4次元のオブジェクトを期待していることを通知していますが、代わりにdoubleが発生しました。

以下は、Gsonによって完全に処理される必要がある有効な3次元行列を表しています。

  • private double[][][] coordinates;(おすすめされた)
  • private Collection<double[]>[] coordinates;
  • private Collection<double[][]> coordinates;
  • private Collection<Collection<double[]>> coordinates;
  • private Collection<Collection<Collection<Double>>> coordinates;

そうは言っても、私はこの特定のケースではList上記を好むでしょう。Collectionを使用するListと、挿入順序が満たされていることを保証でき、インデックスで要素を取得できるようになります。

于 2010-08-11T14:29:35.540 に答える
2

Gsonユーザーガイドによると:

ジェネリック型のシリアライズと
デシリアライズ toJson(obj) を呼び出すと、Gson が呼び出して、シリアライズobj.getClass()するフィールドに関する情報を取得します。同様に、通常はメソッドMyClass.classでオブジェクトを渡すことができます。fromJson(json, MyClass.class)オブジェクトが非ジェネリック型である限り、これは正常に機能します。ただし、オブジェクトがジェネリック型の場合、Java Type Erasureのためにジェネリック型情報が失われます。ポイントを説明する例を次に示します。

List<String> myStrings = new List<String>();
gson.toJson(myStrings); // Will cause a runtime exception

gson.fromJson(json, myStrings.getClass());

上記の呼び出しは、Gson がmyStrings.getClass()そのクラス情報を取得するために呼び出すため、実行時例外になりますが、このメソッドは未加工のクラスを返しますList.class。これは、Gson がこれが文字列のリストであり、単純なオブジェクトではないことを知る方法がないことを意味します。

この問題は、ジェネリック型に正しいパラメーター化された型を指定することで解決できます。TypeTokenクラスを使用してこれを行うことができます。

Type listType = new TypeToken<List<String>>() {}.getType();
gson.toJson(myStrings, listType);

gson.fromJson(json, listType);

get に使用されるイディオムは、完全にパラメーター化された型を返すlistTypeメソッドを含む匿名のローカル内部クラスを実際に定義します。getType()

お役に立てれば。

于 2011-02-11T05:42:25.610 に答える
1

逆シリアル化のためにあなたが書いたものなど、もっと詳細が必要だと思います。

于 2010-08-11T13:52:31.807 に答える