MrShi
2025-02-08 c7be7788de8798c510edd4bdbfedbfa71f417b7e
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
/*
 * 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 cn.emay.sdk.util.json.gson.reflect.TypeToken;
 
/**
 * Creates type adapters for set of related types. Type adapter factories are
 * most useful when several types share similar structure in their JSON form.
 *
 * <h3>Example: Converting enums to lowercase</h3> In this example, we implement
 * a factory that creates type adapters for all enums. The type adapters will
 * write enums in lowercase, despite the fact that they're defined in
 * {@code CONSTANT_CASE} in the corresponding Java model:
 * 
 * <pre>
 * {
 *     &#64;code
 *
 *     public class LowercaseEnumTypeAdapterFactory implements TypeAdapterFactory {
 *         public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
 *             Class<T> rawType = (Class<T>) type.getRawType();
 *             if (!rawType.isEnum()) {
 *                 return null;
 *             }
 *
 *             final Map<String, T> lowercaseToConstant = new HashMap<String, T>();
 *             for (T constant : rawType.getEnumConstants()) {
 *                 lowercaseToConstant.put(toLowercase(constant), constant);
 *             }
 *
 *             return new TypeAdapter<T>() {
 *                 public void write(JsonWriter out, T value) throws IOException {
 *                     if (value == null) {
 *                         out.nullValue();
 *                     } else {
 *                         out.value(toLowercase(value));
 *                     }
 *                 }
 *
 *                 public T read(JsonReader reader) throws IOException {
 *                     if (reader.peek() == JsonToken.NULL) {
 *                         reader.nextNull();
 *                         return null;
 *                     } else {
 *                         return lowercaseToConstant.get(reader.nextString());
 *                     }
 *                 }
 *             };
 *         }
 *
 *         private String toLowercase(Object o) {
 *             return o.toString().toLowerCase(Locale.US);
 *         }
 *     }
 * }
 * </pre>
 *
 * <p>
 * Type adapter factories select which types they provide type adapters for. If
 * a factory cannot support a given type, it must return null when that type is
 * passed to {@link #create}. Factories should expect {@code
 * create()} to be called on them for many types and should return null for most
 * of those types. In the above example the factory returns null for calls to
 * {@code create()} where {@code type} is not an enum.
 *
 * <p>
 * A factory is typically called once per type, but the returned type adapter
 * may be used many times. It is most efficient to do expensive work like
 * reflection in {@code create()} so that the type adapter's {@code
 * read()} and {@code write()} methods can be very fast. In this example the
 * mapping from lowercase name to enum value is computed eagerly.
 *
 * <p>
 * As with type adapters, factories must be <i>registered</i> with a
 * {@link com.google.gson.GsonBuilder} for them to take effect:
 * 
 * <pre>
 *    {@code
 *
 *  GsonBuilder builder = new GsonBuilder();
 *  builder.registerTypeAdapterFactory(new LowercaseEnumTypeAdapterFactory());
 *  ...
 *  Gson gson = builder.create();
 * }
 * </pre>
 * 
 * If multiple factories support the same type, the factory registered earlier
 * takes precedence.
 *
 * <h3>Example: composing other type adapters</h3> In this example we implement
 * a factory for Guava's {@code Multiset} collection type. The factory can be
 * used to create type adapters for multisets of any element type: the type
 * adapter for {@code
 * Multiset<String>} is different from the type adapter for {@code
 * Multiset<URL>}.
 *
 * <p>
 * The type adapter <i>delegates</i> to another type adapter for the multiset
 * elements. It figures out the element type by reflecting on the multiset's
 * type token. A {@code Gson} is passed in to {@code create} for just this
 * purpose:
 * 
 * <pre>
 * {
 *     &#64;code
 *
 *     public class MultisetTypeAdapterFactory implements TypeAdapterFactory {
 *         public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> typeToken) {
 *             Type type = typeToken.getType();
 *             if (typeToken.getRawType() != Multiset.class || !(type instanceof ParameterizedType)) {
 *                 return null;
 *             }
 *
 *             Type elementType = ((ParameterizedType) type).getActualTypeArguments()[0];
 *             TypeAdapter<?> elementAdapter = gson.getAdapter(TypeToken.get(elementType));
 *             return (TypeAdapter<T>) newMultisetAdapter(elementAdapter);
 *         }
 *
 *         private <E> TypeAdapter<Multiset<E>> newMultisetAdapter(final TypeAdapter<E> elementAdapter) {
 *             return new TypeAdapter<Multiset<E>>() {
 *                 public void write(JsonWriter out, Multiset<E> value) throws IOException {
 *                     if (value == null) {
 *                         out.nullValue();
 *                         return;
 *                     }
 *
 *                     out.beginArray();
 *                     for (Multiset.Entry<E> entry : value.entrySet()) {
 *                         out.value(entry.getCount());
 *                         elementAdapter.write(out, entry.getElement());
 *                     }
 *                     out.endArray();
 *                 }
 *
 *                 public Multiset<E> read(JsonReader in) throws IOException {
 *                     if (in.peek() == JsonToken.NULL) {
 *                         in.nextNull();
 *                         return null;
 *                     }
 *
 *                     Multiset<E> result = LinkedHashMultiset.create();
 *                     in.beginArray();
 *                     while (in.hasNext()) {
 *                         int count = in.nextInt();
 *                         E element = elementAdapter.read(in);
 *                         result.add(element, count);
 *                     }
 *                     in.endArray();
 *                     return result;
 *                 }
 *             };
 *         }
 *     }
 * }
 * </pre>
 * 
 * Delegating from one type adapter to another is extremely powerful; it's the
 * foundation of how Gson converts Java objects and collections. Whenever
 * possible your factory should retrieve its delegate type adapter in the
 * {@code create()} method; this ensures potentially-expensive type adapter
 * creation happens only once.
 *
 * @since 2.1
 */
public interface TypeAdapterFactory {
 
    /**
     * Returns a type adapter for {@code type}, or null if this factory doesn't
     * support {@code type}.
     */
    <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type);
}