rk
2025-09-28 a8fb7ae2dbb61a94141ed5e73d3bb2632b7b84df
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
/*
 * Copyright (C) 2011 Google Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
 
package cn.emay.sdk.util.json.gson;
 
import java.io.IOException;
import java.io.Reader;
import java.io.StringReader;
import java.io.StringWriter;
import java.io.Writer;
 
import cn.emay.sdk.util.json.gson.internal.bind.JsonTreeReader;
import cn.emay.sdk.util.json.gson.internal.bind.JsonTreeWriter;
import cn.emay.sdk.util.json.gson.stream.JsonReader;
import cn.emay.sdk.util.json.gson.stream.JsonToken;
import cn.emay.sdk.util.json.gson.stream.JsonWriter;
 
/**
 * Converts Java objects to and from JSON.
 *
 * <h3>Defining a type's JSON form</h3> By default Gson converts application
 * classes to JSON using its built-in type adapters. If Gson's default JSON
 * conversion isn't appropriate for a type, extend this class to customize the
 * conversion. Here's an example of a type adapter for an (X,Y) coordinate
 * point:
 * 
 * <pre>
 * {
 *     &#64;code
 *
 *     public class PointAdapter extends TypeAdapter<Point> {
 *         public Point read(JsonReader reader) throws IOException {
 *             if (reader.peek() == JsonToken.NULL) {
 *                 reader.nextNull();
 *                 return null;
 *             }
 *             String xy = reader.nextString();
 *             String[] parts = xy.split(",");
 *             int x = Integer.parseInt(parts[0]);
 *             int y = Integer.parseInt(parts[1]);
 *             return new Point(x, y);
 *         }
 * 
 *         public void write(JsonWriter writer, Point value) throws IOException {
 *             if (value == null) {
 *                 writer.nullValue();
 *                 return;
 *             }
 *             String xy = value.getX() + "," + value.getY();
 *             writer.value(xy);
 *         }
 *     }
 * }
 * </pre>
 * 
 * With this type adapter installed, Gson will convert {@code Points} to JSON as
 * strings like {@code "5,8"} rather than objects like {@code {"x":5,"y":8}}. In
 * this case the type adapter binds a rich Java class to a compact JSON value.
 *
 * <p>
 * The {@link #read(JsonReader) read()} method must read exactly one value and
 * {@link #write(JsonWriter,Object) write()} must write exactly one value. For
 * primitive types this is means readers should make exactly one call to
 * {@code nextBoolean()}, {@code nextDouble()}, {@code nextInt()}, {@code
 * nextLong()}, {@code nextString()} or {@code nextNull()}. Writers should make
 * exactly one call to one of <code>value()</code> or <code>nullValue()</code>.
 * For arrays, type adapters should start with a call to {@code beginArray()},
 * convert all elements, and finish with a call to {@code endArray()}. For
 * objects, they should start with {@code beginObject()}, convert the object,
 * and finish with {@code endObject()}. Failing to convert a value or converting
 * too many values may cause the application to crash.
 *
 * <p>
 * Type adapters should be prepared to read null from the stream and write it to
 * the stream. Alternatively, they should use {@link #nullSafe()} method while
 * registering the type adapter with Gson. If your {@code Gson} instance has
 * been configured to {@link GsonBuilder#serializeNulls()}, these nulls will be
 * written to the final document. Otherwise the value (and the corresponding
 * name when writing to a JSON object) will be omitted automatically. In either
 * case your type adapter must handle null.
 *
 * <p>
 * To use a custom type adapter with Gson, you must <i>register</i> it with a
 * {@link GsonBuilder}:
 * 
 * <pre>
 *    {@code
 *
 *   GsonBuilder builder = new GsonBuilder();
 *   builder.registerTypeAdapter(Point.class, new PointAdapter());
 *   // if PointAdapter didn't check for nulls in its read/write methods, you should instead use
 *   // builder.registerTypeAdapter(Point.class, new PointAdapter().nullSafe());
 *   ...
 *   Gson gson = builder.create();
 * }
 * </pre>
 *
 * @since 2.1
 */
// non-Javadoc:
//
// <h3>JSON Conversion</h3>
// <p>A type adapter registered with Gson is automatically invoked while
// serializing
// or deserializing JSON. However, you can also use type adapters directly to
// serialize
// and deserialize JSON. Here is an example for deserialization: <pre> {@code
//
// String json = "{'origin':'0,0','points':['1,2','3,4']}";
// TypeAdapter<Graph> graphAdapter = gson.getAdapter(Graph.class);
// Graph graph = graphAdapter.fromJson(json);
// }</pre>
// And an example for serialization: <pre> {@code
//
// Graph graph = new Graph(...);
// TypeAdapter<Graph> graphAdapter = gson.getAdapter(Graph.class);
// String json = graphAdapter.toJson(graph);
// }</pre>
//
// <p>Type adapters are <strong>type-specific</strong>. For example, a {@code
// TypeAdapter<Date>} can convert {@code Date} instances to JSON and JSON to
// instances of {@code Date}, but cannot convert any other types.
//
public abstract class TypeAdapter<T> {
 
    /**
     * Writes one JSON value (an array, object, string, number, boolean or null) for
     * {@code value}.
     *
     * @param value
     *            the Java object to write. May be null.
     */
    public abstract void write(JsonWriter out, T value) throws IOException;
 
    /**
     * Converts {@code value} to a JSON document and writes it to {@code out}.
     * Unlike Gson's similar {@link Gson#toJson(JsonElement, Appendable) toJson}
     * method, this write is strict. Create a {@link JsonWriter#setLenient(boolean)
     * lenient} {@code JsonWriter} and call
     * {@link #write(com.google.gson.stream.JsonWriter, Object)} for lenient
     * writing.
     *
     * @param value
     *            the Java object to convert. May be null.
     * @since 2.2
     */
    public final void toJson(Writer out, T value) throws IOException {
        JsonWriter writer = new JsonWriter(out);
        write(writer, value);
    }
 
    /**
     * This wrapper method is used to make a type adapter null tolerant. In general,
     * a type adapter is required to handle nulls in write and read methods. Here is
     * how this is typically done:<br>
     * 
     * <pre>
     * {
     *     &#64;code
     *
     *     Gson gson = new GsonBuilder().registerTypeAdapter(Foo.class, new TypeAdapter<Foo>() {
     *         public Foo read(JsonReader in) throws IOException {
     *             if (in.peek() == JsonToken.NULL) {
     *                 in.nextNull();
     *                 return null;
     *             }
     *             // read a Foo from in and return it
     *         }
     * 
     *         public void write(JsonWriter out, Foo src) throws IOException {
     *             if (src == null) {
     *                 out.nullValue();
     *                 return;
     *             }
     *             // write src as JSON to out
     *         }
     *     }).create();
     * }
     * </pre>
     * 
     * You can avoid this boilerplate handling of nulls by wrapping your type
     * adapter with this method. Here is how we will rewrite the above example:
     * 
     * <pre>
     * {
     *     &#64;code
     *
     *     Gson gson = new GsonBuilder().registerTypeAdapter(Foo.class, new TypeAdapter<Foo>() {
     *         public Foo read(JsonReader in) throws IOException {
     *             // read a Foo from in and return it
     *         }
     * 
     *         public void write(JsonWriter out, Foo src) throws IOException {
     *             // write src as JSON to out
     *         }
     *     }.nullSafe()).create();
     * }
     * </pre>
     * 
     * Note that we didn't need to check for nulls in our type adapter after we used
     * nullSafe.
     */
    public final TypeAdapter<T> nullSafe() {
        return new TypeAdapter<T>() {
            @Override
            public void write(JsonWriter out, T value) throws IOException {
                if (value == null) {
                    out.nullValue();
                } else {
                    TypeAdapter.this.write(out, value);
                }
            }
 
            @Override
            public T read(JsonReader reader) throws IOException {
                if (reader.peek() == JsonToken.NULL) {
                    reader.nextNull();
                    return null;
                }
                return TypeAdapter.this.read(reader);
            }
        };
    }
 
    /**
     * Converts {@code value} to a JSON document. Unlike Gson's similar
     * {@link Gson#toJson(Object) toJson} method, this write is strict. Create a
     * {@link JsonWriter#setLenient(boolean) lenient} {@code JsonWriter} and call
     * {@link #write(com.google.gson.stream.JsonWriter, Object)} for lenient
     * writing.
     *
     * @param value
     *            the Java object to convert. May be null.
     * @since 2.2
     */
    public final String toJson(T value) {
        StringWriter stringWriter = new StringWriter();
        try {
            toJson(stringWriter, value);
        } catch (IOException e) {
            throw new AssertionError(e); // No I/O writing to a StringWriter.
        }
        return stringWriter.toString();
    }
 
    /**
     * Converts {@code value} to a JSON tree.
     *
     * @param value
     *            the Java object to convert. May be null.
     * @return the converted JSON tree. May be {@link JsonNull}.
     * @since 2.2
     */
    public final JsonElement toJsonTree(T value) {
        try {
            JsonTreeWriter jsonWriter = new JsonTreeWriter();
            write(jsonWriter, value);
            return jsonWriter.get();
        } catch (IOException e) {
            throw new JsonIOException(e);
        }
    }
 
    /**
     * Reads one JSON value (an array, object, string, number, boolean or null) and
     * converts it to a Java object. Returns the converted object.
     *
     * @return the converted Java object. May be null.
     */
    public abstract T read(JsonReader in) throws IOException;
 
    /**
     * Converts the JSON document in {@code in} to a Java object. Unlike Gson's
     * similar {@link Gson#fromJson(java.io.Reader, Class) fromJson} method, this
     * read is strict. Create a {@link JsonReader#setLenient(boolean) lenient}
     * {@code JsonReader} and call {@link #read(JsonReader)} for lenient reading.
     *
     * @return the converted Java object. May be null.
     * @since 2.2
     */
    public final T fromJson(Reader in) throws IOException {
        JsonReader reader = new JsonReader(in);
        return read(reader);
    }
 
    /**
     * Converts the JSON document in {@code json} to a Java object. Unlike Gson's
     * similar {@link Gson#fromJson(String, Class) fromJson} method, this read is
     * strict. Create a {@link JsonReader#setLenient(boolean) lenient} {@code
     * JsonReader} and call {@link #read(JsonReader)} for lenient reading.
     *
     * @return the converted Java object. May be null.
     * @since 2.2
     */
    public final T fromJson(String json) throws IOException {
        return fromJson(new StringReader(json));
    }
 
    /**
     * Converts {@code jsonTree} to a Java object.
     *
     * @param jsonTree
     *            the Java object to convert. May be {@link JsonNull}.
     * @since 2.2
     */
    public final T fromJsonTree(JsonElement jsonTree) {
        try {
            JsonReader jsonReader = new JsonTreeReader(jsonTree);
            return read(jsonReader);
        } catch (IOException e) {
            throw new JsonIOException(e);
        }
    }
}