1 package org
.gaixie
.jibu
.json
;
4 Copyright (c) 2002 JSON.org
6 Permission is hereby granted, free of charge, to any person obtaining a copy
7 of this software and associated documentation files (the "Software"), to deal
8 in the Software without restriction, including without limitation the rights
9 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 copies of the Software, and to permit persons to whom the Software is
11 furnished to do so, subject to the following conditions:
13 The above copyright notice and this permission notice shall be included in all
14 copies or substantial portions of the Software.
16 The Software shall be used for Good, not Evil.
18 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
27 import java
.io
.IOException
;
28 import java
.io
.Writer
;
29 import java
.lang
.reflect
.Field
;
30 import java
.lang
.reflect
.Modifier
;
31 import java
.lang
.reflect
.Method
;
32 import java
.util
.Collection
;
33 import java
.util
.HashMap
;
34 import java
.util
.Iterator
;
36 import java
.util
.TreeSet
;
39 * A JSONObject is an unordered collection of name/value pairs. Its
40 * external form is a string wrapped in curly braces with colons between the
41 * names and values, and commas between the values and names. The internal form
42 * is an object having <code>get</code> and <code>opt</code> methods for
43 * accessing the values by name, and <code>put</code> methods for adding or
44 * replacing values by name. The values can be any of these types:
45 * <code>Boolean</code>, <code>JSONArray</code>, <code>JSONObject</code>,
46 * <code>Number</code>, <code>String</code>, or the <code>JSONObject.NULL</code>
47 * object. A JSONObject constructor can be used to convert an external form
48 * JSON text into an internal form whose values can be retrieved with the
49 * <code>get</code> and <code>opt</code> methods, or to convert values into a
50 * JSON text using the <code>put</code> and <code>toString</code> methods.
51 * A <code>get</code> method returns a value if one can be found, and throws an
52 * exception if one cannot be found. An <code>opt</code> method returns a
53 * default value instead of throwing an exception, and so is useful for
54 * obtaining optional values.
56 * The generic <code>get()</code> and <code>opt()</code> methods return an
57 * object, which you can cast or query for type. There are also typed
58 * <code>get</code> and <code>opt</code> methods that do type checking and type
61 * The <code>put</code> methods adds values to an object. For example, <pre>
62 * myString = new JSONObject().put("JSON", "Hello, World!").toString();</pre>
63 * produces the string <code>{"JSON": "Hello, World"}</code>.
65 * The texts produced by the <code>toString</code> methods strictly conform to
66 * the JSON syntax rules.
67 * The constructors are more forgiving in the texts they will accept:
69 * <li>An extra <code>,</code> <small>(comma)</small> may appear just
70 * before the closing brace.</li>
71 * <li>Strings may be quoted with <code>'</code> <small>(single
72 * quote)</small>.</li>
73 * <li>Strings do not need to be quoted at all if they do not begin with a quote
74 * or single quote, and if they do not contain leading or trailing spaces,
75 * and if they do not contain any of these characters:
76 * <code>{ } [ ] / \ : , = ; #</code> and if they do not look like numbers
77 * and if they are not the reserved words <code>true</code>,
78 * <code>false</code>, or <code>null</code>.</li>
79 * <li>Keys can be followed by <code>=</code> or <code>=></code> as well as
80 * by <code>:</code>.</li>
81 * <li>Values can be followed by <code>;</code> <small>(semicolon)</small> as
82 * well as by <code>,</code> <small>(comma)</small>.</li>
83 * <li>Numbers may have the <code>0-</code> <small>(octal)</small> or
84 * <code>0x-</code> <small>(hex)</small> prefix.</li>
89 public class JSONObject
{
92 * JSONObject.NULL is equivalent to the value that JavaScript calls null,
93 * whilst Java's null is equivalent to the value that JavaScript calls
96 private static final class Null
{
99 * There is only intended to be a single instance of the NULL object,
100 * so the clone method returns itself.
103 protected final Object
clone() {
109 * A Null object is equal to the null value and to itself.
110 * @param object An object to test for nullness.
111 * @return true if the object parameter is the JSONObject.NULL object
114 public boolean equals(Object object
) {
115 return object
== null || object
== this;
120 * Get the "null" string value.
121 * @return The string "null".
123 public String
toString() {
130 * The map where the JSONObject's properties are kept.
136 * It is sometimes more convenient and less ambiguous to have a
137 * <code>NULL</code> object than to use Java's <code>null</code> value.
138 * <code>JSONObject.NULL.equals(null)</code> returns <code>true</code>.
139 * <code>JSONObject.NULL.toString()</code> returns <code>"null"</code>.
141 public static final Object NULL
= new Null();
145 * Construct an empty JSONObject.
147 public JSONObject() {
148 this.map
= new HashMap();
153 * Construct a JSONObject from a subset of another JSONObject.
154 * An array of strings is used to identify the keys that should be copied.
155 * Missing keys are ignored.
156 * @param jo A JSONObject.
157 * @param names An array of strings.
158 * @exception JSONException If a value is a non-finite number or if a name is duplicated.
160 public JSONObject(JSONObject jo
, String
[] names
) throws JSONException
{
162 for (int i
= 0; i
< names
.length
; i
+= 1) {
163 putOnce(names
[i
], jo
.opt(names
[i
]));
169 * Construct a JSONObject from a JSONTokener.
170 * @param x A JSONTokener object containing the source string.
171 * @throws JSONException If there is a syntax error in the source string
172 * or a duplicated key.
174 public JSONObject(JSONTokener x
) throws JSONException
{
179 if (x
.nextClean() != '{') {
180 throw x
.syntaxError("A JSONObject text must begin with '{'");
186 throw x
.syntaxError("A JSONObject text must end with '}'");
191 key
= x
.nextValue().toString();
195 * The key is followed by ':'. We will also tolerate '=' or '=>'.
200 if (x
.next() != '>') {
203 } else if (c
!= ':') {
204 throw x
.syntaxError("Expected a ':' after a key");
206 putOnce(key
, x
.nextValue());
209 * Pairs are separated by ','. We will also tolerate ';'.
212 switch (x
.nextClean()) {
215 if (x
.nextClean() == '}') {
223 throw x
.syntaxError("Expected a ',' or '}'");
230 * Construct a JSONObject from a Map.
232 * @param map A map object that can be used to initialize the contents of
235 public JSONObject(Map map
) {
236 this.map
= (map
== null) ?
new HashMap() : map
;
241 * Construct a JSONObject from a Map.
243 * Note: Use this constructor when the map contains <key,bean>.
245 * @param map - A map with Key-Bean data.
246 * @param includeSuperClass - Tell whether to include the super class properties.
248 public JSONObject(Map map
, boolean includeSuperClass
) {
249 this.map
= new HashMap();
251 Iterator i
= map
.entrySet().iterator();
252 while (i
.hasNext()) {
253 Map
.Entry e
= (Map
.Entry
)i
.next();
254 if (isStandardProperty(e
.getValue().getClass())) {
255 this.map
.put(e
.getKey(), e
.getValue());
257 this.map
.put(e
.getKey(), new JSONObject(e
.getValue(),
266 * Construct a JSONObject from an Object using bean getters.
267 * It reflects on all of the public methods of the object.
268 * For each of the methods with no parameters and a name starting
269 * with <code>"get"</code> or <code>"is"</code> followed by an uppercase letter,
270 * the method is invoked, and a key and the value returned from the getter method
271 * are put into the new JSONObject.
273 * The key is formed by removing the <code>"get"</code> or <code>"is"</code> prefix.
274 * If the second remaining character is not upper case, then the first
275 * character is converted to lower case.
277 * For example, if an object has a method named <code>"getName"</code>, and
278 * if the result of calling <code>object.getName()</code> is <code>"Larry Fine"</code>,
279 * then the JSONObject will contain <code>"name": "Larry Fine"</code>.
281 * @param bean An object that has getter methods that should be used
282 * to make a JSONObject.
284 public JSONObject(Object bean
) {
286 populateInternalMap(bean
, false);
291 * Construct a JSONObject from an Object using bean getters.
292 * It reflects on all of the public methods of the object.
293 * For each of the methods with no parameters and a name starting
294 * with <code>"get"</code> or <code>"is"</code> followed by an uppercase letter,
295 * the method is invoked, and a key and the value returned from the getter method
296 * are put into the new JSONObject.
298 * The key is formed by removing the <code>"get"</code> or <code>"is"</code> prefix.
299 * If the second remaining character is not upper case, then the first
300 * character is converted to lower case.
302 * @param bean An object that has getter methods that should be used
303 * to make a JSONObject.
304 * @param includeSuperClass If true, include the super class properties.
306 public JSONObject(Object bean
, boolean includeSuperClass
) {
308 populateInternalMap(bean
, includeSuperClass
);
311 private void populateInternalMap(Object bean
, boolean includeSuperClass
){
312 Class klass
= bean
.getClass();
314 /* If klass.getSuperClass is System class then force includeSuperClass to false. */
316 if (klass
.getClassLoader() == null) {
317 includeSuperClass
= false;
320 Method
[] methods
= (includeSuperClass
) ?
321 klass
.getMethods() : klass
.getDeclaredMethods();
322 for (int i
= 0; i
< methods
.length
; i
+= 1) {
324 Method method
= methods
[i
];
325 if (Modifier
.isPublic(method
.getModifiers())) {
326 String name
= method
.getName();
328 if (name
.startsWith("get")) {
329 key
= name
.substring(3);
330 } else if (name
.startsWith("is")) {
331 key
= name
.substring(2);
333 if (key
.length() > 0 &&
334 Character
.isUpperCase(key
.charAt(0)) &&
335 method
.getParameterTypes().length
== 0) {
336 if (key
.length() == 1) {
337 key
= key
.toLowerCase();
338 } else if (!Character
.isUpperCase(key
.charAt(1))) {
339 key
= key
.substring(0, 1).toLowerCase() +
343 Object result
= method
.invoke(bean
, (Object
[])null);
344 if (result
== null) {
346 } else if (result
.getClass().isArray()) {
347 map
.put(key
, new JSONArray(result
, includeSuperClass
));
348 } else if (result
instanceof Collection
) { // List or Set
349 map
.put(key
, new JSONArray((Collection
)result
, includeSuperClass
));
350 } else if (result
instanceof Map
) {
351 map
.put(key
, new JSONObject((Map
)result
, includeSuperClass
));
352 } else if (isStandardProperty(result
.getClass())) { // Primitives, String and Wrapper
353 map
.put(key
, result
);
355 if (result
.getClass().getPackage().getName().startsWith("java") ||
356 result
.getClass().getClassLoader() == null) {
357 map
.put(key
, result
.toString());
358 } else { // User defined Objects
359 map
.put(key
, new JSONObject(result
, includeSuperClass
));
364 } catch (Exception e
) {
365 throw new RuntimeException(e
);
371 static boolean isStandardProperty(Class clazz
) {
372 return clazz
.isPrimitive() ||
373 clazz
.isAssignableFrom(Byte
.class) ||
374 clazz
.isAssignableFrom(Short
.class) ||
375 clazz
.isAssignableFrom(Integer
.class) ||
376 clazz
.isAssignableFrom(Long
.class) ||
377 clazz
.isAssignableFrom(Float
.class) ||
378 clazz
.isAssignableFrom(Double
.class) ||
379 clazz
.isAssignableFrom(Character
.class) ||
380 clazz
.isAssignableFrom(String
.class) ||
381 clazz
.isAssignableFrom(Boolean
.class);
386 * Construct a JSONObject from an Object, using reflection to find the
387 * public members. The resulting JSONObject's keys will be the strings
388 * from the names array, and the values will be the field values associated
389 * with those keys in the object. If a key is not found or not visible,
390 * then it will not be copied into the new JSONObject.
391 * @param object An object that has fields that should be used to make a
393 * @param names An array of strings, the names of the fields to be obtained
396 public JSONObject(Object object
, String names
[]) {
398 Class c
= object
.getClass();
399 for (int i
= 0; i
< names
.length
; i
+= 1) {
400 String name
= names
[i
];
402 putOpt(name
, c
.getField(name
).get(object
));
403 } catch (Exception e
) {
404 /* forget about it */
411 * Construct a JSONObject from a source JSON text string.
412 * This is the most commonly used JSONObject constructor.
413 * @param source A string beginning
414 * with <code>{</code> <small>(left brace)</small> and ending
415 * with <code>}</code> <small>(right brace)</small>.
416 * @exception JSONException If there is a syntax error in the source
417 * string or a duplicated key.
419 public JSONObject(String source
) throws JSONException
{
420 this(new JSONTokener(source
));
425 * Accumulate values under a key. It is similar to the put method except
426 * that if there is already an object stored under the key then a
427 * JSONArray is stored under the key to hold all of the accumulated values.
428 * If there is already a JSONArray, then the new value is appended to it.
429 * In contrast, the put method replaces the previous value.
430 * @param key A key string.
431 * @param value An object to be accumulated under the key.
433 * @throws JSONException If the value is an invalid number
434 * or if the key is null.
436 public JSONObject
accumulate(String key
, Object value
)
437 throws JSONException
{
441 put(key
, value
instanceof JSONArray ?
442 new JSONArray().put(value
) :
444 } else if (o
instanceof JSONArray
) {
445 ((JSONArray
)o
).put(value
);
447 put(key
, new JSONArray().put(o
).put(value
));
454 * Append values to the array under a key. If the key does not exist in the
455 * JSONObject, then the key is put in the JSONObject with its value being a
456 * JSONArray containing the value parameter. If the key was already
457 * associated with a JSONArray, then the value parameter is appended to it.
458 * @param key A key string.
459 * @param value An object to be accumulated under the key.
461 * @throws JSONException If the key is null or if the current value
462 * associated with the key is not a JSONArray.
464 public JSONObject
append(String key
, Object value
)
465 throws JSONException
{
469 put(key
, new JSONArray().put(value
));
470 } else if (o
instanceof JSONArray
) {
471 put(key
, ((JSONArray
)o
).put(value
));
473 throw new JSONException("JSONObject[" + key
+
474 "] is not a JSONArray.");
481 * Produce a string from a double. The string "null" will be returned if
482 * the number is not finite.
486 static public String
doubleToString(double d
) {
487 if (Double
.isInfinite(d
) || Double
.isNaN(d
)) {
491 // Shave off trailing zeros and decimal point, if possible.
493 String s
= Double
.toString(d
);
494 if (s
.indexOf('.') > 0 && s
.indexOf('e') < 0 && s
.indexOf('E') < 0) {
495 while (s
.endsWith("0")) {
496 s
= s
.substring(0, s
.length() - 1);
498 if (s
.endsWith(".")) {
499 s
= s
.substring(0, s
.length() - 1);
507 * Get the value object associated with a key.
509 * @param key A key string.
510 * @return The object associated with the key.
511 * @throws JSONException if the key is not found.
513 public Object
get(String key
) throws JSONException
{
516 throw new JSONException("JSONObject[" + quote(key
) +
524 * Get the boolean value associated with a key.
526 * @param key A key string.
528 * @throws JSONException
529 * if the value is not a Boolean or the String "true" or "false".
531 public boolean getBoolean(String key
) throws JSONException
{
533 if (o
.equals(Boolean
.FALSE
) ||
534 (o
instanceof String
&&
535 ((String
)o
).equalsIgnoreCase("false"))) {
537 } else if (o
.equals(Boolean
.TRUE
) ||
538 (o
instanceof String
&&
539 ((String
)o
).equalsIgnoreCase("true"))) {
542 throw new JSONException("JSONObject[" + quote(key
) +
543 "] is not a Boolean.");
548 * Get the double value associated with a key.
549 * @param key A key string.
550 * @return The numeric value.
551 * @throws JSONException if the key is not found or
552 * if the value is not a Number object and cannot be converted to a number.
554 public double getDouble(String key
) throws JSONException
{
557 return o
instanceof Number ?
558 ((Number
)o
).doubleValue() :
559 Double
.valueOf((String
)o
).doubleValue();
560 } catch (Exception e
) {
561 throw new JSONException("JSONObject[" + quote(key
) +
562 "] is not a number.");
568 * Get the int value associated with a key. If the number value is too
569 * large for an int, it will be clipped.
571 * @param key A key string.
572 * @return The integer value.
573 * @throws JSONException if the key is not found or if the value cannot
574 * be converted to an integer.
576 public int getInt(String key
) throws JSONException
{
578 return o
instanceof Number ?
579 ((Number
)o
).intValue() : (int)getDouble(key
);
584 * Get the JSONArray value associated with a key.
586 * @param key A key string.
587 * @return A JSONArray which is the value.
588 * @throws JSONException if the key is not found or
589 * if the value is not a JSONArray.
591 public JSONArray
getJSONArray(String key
) throws JSONException
{
593 if (o
instanceof JSONArray
) {
596 throw new JSONException("JSONObject[" + quote(key
) +
597 "] is not a JSONArray.");
602 * Get the JSONObject value associated with a key.
604 * @param key A key string.
605 * @return A JSONObject which is the value.
606 * @throws JSONException if the key is not found or
607 * if the value is not a JSONObject.
609 public JSONObject
getJSONObject(String key
) throws JSONException
{
611 if (o
instanceof JSONObject
) {
612 return (JSONObject
)o
;
614 throw new JSONException("JSONObject[" + quote(key
) +
615 "] is not a JSONObject.");
620 * Get the long value associated with a key. If the number value is too
621 * long for a long, it will be clipped.
623 * @param key A key string.
624 * @return The long value.
625 * @throws JSONException if the key is not found or if the value cannot
626 * be converted to a long.
628 public long getLong(String key
) throws JSONException
{
630 return o
instanceof Number ?
631 ((Number
)o
).longValue() : (long)getDouble(key
);
636 * Get an array of field names from a JSONObject.
638 * @return An array of field names, or null if there are no names.
640 public static String
[] getNames(JSONObject jo
) {
641 int length
= jo
.length();
645 Iterator i
= jo
.keys();
646 String
[] names
= new String
[length
];
648 while (i
.hasNext()) {
649 names
[j
] = (String
)i
.next();
657 * Get an array of field names from an Object.
659 * @return An array of field names, or null if there are no names.
661 public static String
[] getNames(Object object
) {
662 if (object
== null) {
665 Class klass
= object
.getClass();
666 Field
[] fields
= klass
.getFields();
667 int length
= fields
.length
;
671 String
[] names
= new String
[length
];
672 for (int i
= 0; i
< length
; i
+= 1) {
673 names
[i
] = fields
[i
].getName();
680 * Get the string associated with a key.
682 * @param key A key string.
683 * @return A string which is the value.
684 * @throws JSONException if the key is not found.
686 public String
getString(String key
) throws JSONException
{
687 return get(key
).toString();
692 * Determine if the JSONObject contains a specific key.
693 * @param key A key string.
694 * @return true if the key exists in the JSONObject.
696 public boolean has(String key
) {
697 return this.map
.containsKey(key
);
702 * Determine if the value associated with the key is null or if there is
704 * @param key A key string.
705 * @return true if there is no value associated with the key or if
706 * the value is the JSONObject.NULL object.
708 public boolean isNull(String key
) {
709 return JSONObject
.NULL
.equals(opt(key
));
714 * Get an enumeration of the keys of the JSONObject.
716 * @return An iterator of the keys.
718 public Iterator
keys() {
719 return this.map
.keySet().iterator();
724 * Get the number of keys stored in the JSONObject.
726 * @return The number of keys in the JSONObject.
728 public int length() {
729 return this.map
.size();
734 * Produce a JSONArray containing the names of the elements of this
736 * @return A JSONArray containing the key strings, or null if the JSONObject
739 public JSONArray
names() {
740 JSONArray ja
= new JSONArray();
741 Iterator keys
= keys();
742 while (keys
.hasNext()) {
745 return ja
.length() == 0 ?
null : ja
;
749 * Produce a string from a Number.
752 * @throws JSONException If n is a non-finite number.
754 static public String
numberToString(Number n
)
755 throws JSONException
{
757 throw new JSONException("Null pointer");
761 // Shave off trailing zeros and decimal point, if possible.
763 String s
= n
.toString();
764 if (s
.indexOf('.') > 0 && s
.indexOf('e') < 0 && s
.indexOf('E') < 0) {
765 while (s
.endsWith("0")) {
766 s
= s
.substring(0, s
.length() - 1);
768 if (s
.endsWith(".")) {
769 s
= s
.substring(0, s
.length() - 1);
777 * Get an optional value associated with a key.
778 * @param key A key string.
779 * @return An object which is the value, or null if there is no value.
781 public Object
opt(String key
) {
782 return key
== null ?
null : this.map
.get(key
);
787 * Get an optional boolean associated with a key.
788 * It returns false if there is no such key, or if the value is not
789 * Boolean.TRUE or the String "true".
791 * @param key A key string.
794 public boolean optBoolean(String key
) {
795 return optBoolean(key
, false);
800 * Get an optional boolean associated with a key.
801 * It returns the defaultValue if there is no such key, or if it is not
802 * a Boolean or the String "true" or "false" (case insensitive).
804 * @param key A key string.
805 * @param defaultValue The default.
808 public boolean optBoolean(String key
, boolean defaultValue
) {
810 return getBoolean(key
);
811 } catch (Exception e
) {
818 * Put a key/value pair in the JSONObject, where the value will be a
819 * JSONArray which is produced from a Collection.
820 * @param key A key string.
821 * @param value A Collection value.
823 * @throws JSONException
825 public JSONObject
put(String key
, Collection value
) throws JSONException
{
826 put(key
, new JSONArray(value
));
832 * Get an optional double associated with a key,
833 * or NaN if there is no such key or if its value is not a number.
834 * If the value is a string, an attempt will be made to evaluate it as
837 * @param key A string which is the key.
838 * @return An object which is the value.
840 public double optDouble(String key
) {
841 return optDouble(key
, Double
.NaN
);
846 * Get an optional double associated with a key, or the
847 * defaultValue if there is no such key or if its value is not a number.
848 * If the value is a string, an attempt will be made to evaluate it as
851 * @param key A key string.
852 * @param defaultValue The default.
853 * @return An object which is the value.
855 public double optDouble(String key
, double defaultValue
) {
858 return o
instanceof Number ?
((Number
)o
).doubleValue() :
859 new Double((String
)o
).doubleValue();
860 } catch (Exception e
) {
867 * Get an optional int value associated with a key,
868 * or zero if there is no such key or if the value is not a number.
869 * If the value is a string, an attempt will be made to evaluate it as
872 * @param key A key string.
873 * @return An object which is the value.
875 public int optInt(String key
) {
876 return optInt(key
, 0);
881 * Get an optional int value associated with a key,
882 * or the default if there is no such key or if the value is not a number.
883 * If the value is a string, an attempt will be made to evaluate it as
886 * @param key A key string.
887 * @param defaultValue The default.
888 * @return An object which is the value.
890 public int optInt(String key
, int defaultValue
) {
893 } catch (Exception e
) {
900 * Get an optional JSONArray associated with a key.
901 * It returns null if there is no such key, or if its value is not a
904 * @param key A key string.
905 * @return A JSONArray which is the value.
907 public JSONArray
optJSONArray(String key
) {
909 return o
instanceof JSONArray ?
(JSONArray
)o
: null;
914 * Get an optional JSONObject associated with a key.
915 * It returns null if there is no such key, or if its value is not a
918 * @param key A key string.
919 * @return A JSONObject which is the value.
921 public JSONObject
optJSONObject(String key
) {
923 return o
instanceof JSONObject ?
(JSONObject
)o
: null;
928 * Get an optional long value associated with a key,
929 * or zero if there is no such key or if the value is not a number.
930 * If the value is a string, an attempt will be made to evaluate it as
933 * @param key A key string.
934 * @return An object which is the value.
936 public long optLong(String key
) {
937 return optLong(key
, 0);
942 * Get an optional long value associated with a key,
943 * or the default if there is no such key or if the value is not a number.
944 * If the value is a string, an attempt will be made to evaluate it as
947 * @param key A key string.
948 * @param defaultValue The default.
949 * @return An object which is the value.
951 public long optLong(String key
, long defaultValue
) {
954 } catch (Exception e
) {
961 * Get an optional string associated with a key.
962 * It returns an empty string if there is no such key. If the value is not
963 * a string and is not null, then it is coverted to a string.
965 * @param key A key string.
966 * @return A string which is the value.
968 public String
optString(String key
) {
969 return optString(key
, "");
974 * Get an optional string associated with a key.
975 * It returns the defaultValue if there is no such key.
977 * @param key A key string.
978 * @param defaultValue The default.
979 * @return A string which is the value.
981 public String
optString(String key
, String defaultValue
) {
983 return o
!= null ? o
.toString() : defaultValue
;
988 * Put a key/boolean pair in the JSONObject.
990 * @param key A key string.
991 * @param value A boolean which is the value.
993 * @throws JSONException If the key is null.
995 public JSONObject
put(String key
, boolean value
) throws JSONException
{
996 put(key
, value ? Boolean
.TRUE
: Boolean
.FALSE
);
1002 * Put a key/double pair in the JSONObject.
1004 * @param key A key string.
1005 * @param value A double which is the value.
1007 * @throws JSONException If the key is null or if the number is invalid.
1009 public JSONObject
put(String key
, double value
) throws JSONException
{
1010 put(key
, new Double(value
));
1016 * Put a key/int pair in the JSONObject.
1018 * @param key A key string.
1019 * @param value An int which is the value.
1021 * @throws JSONException If the key is null.
1023 public JSONObject
put(String key
, int value
) throws JSONException
{
1024 put(key
, new Integer(value
));
1030 * Put a key/long pair in the JSONObject.
1032 * @param key A key string.
1033 * @param value A long which is the value.
1035 * @throws JSONException If the key is null.
1037 public JSONObject
put(String key
, long value
) throws JSONException
{
1038 put(key
, new Long(value
));
1044 * Put a key/value pair in the JSONObject, where the value will be a
1045 * JSONObject which is produced from a Map.
1046 * @param key A key string.
1047 * @param value A Map value.
1049 * @throws JSONException
1051 public JSONObject
put(String key
, Map value
) throws JSONException
{
1052 put(key
, new JSONObject(value
));
1058 * Put a key/value pair in the JSONObject. If the value is null,
1059 * then the key will be removed from the JSONObject if it is present.
1060 * @param key A key string.
1061 * @param value An object which is the value. It should be of one of these
1062 * types: Boolean, Double, Integer, JSONArray, JSONObject, Long, String,
1063 * or the JSONObject.NULL object.
1065 * @throws JSONException If the value is non-finite number
1066 * or if the key is null.
1068 public JSONObject
put(String key
, Object value
) throws JSONException
{
1070 throw new JSONException("Null key.");
1072 if (value
!= null) {
1073 testValidity(value
);
1074 this.map
.put(key
, value
);
1083 * Put a key/value pair in the JSONObject, but only if the key and the
1084 * value are both non-null, and only if there is not already a member
1089 * @throws JSONException if the key is a duplicate
1091 public JSONObject
putOnce(String key
, Object value
) throws JSONException
{
1092 if (key
!= null && value
!= null) {
1093 if (opt(key
) != null) {
1094 throw new JSONException("Duplicate key \"" + key
+ "\"");
1103 * Put a key/value pair in the JSONObject, but only if the
1104 * key and the value are both non-null.
1105 * @param key A key string.
1106 * @param value An object which is the value. It should be of one of these
1107 * types: Boolean, Double, Integer, JSONArray, JSONObject, Long, String,
1108 * or the JSONObject.NULL object.
1110 * @throws JSONException If the value is a non-finite number.
1112 public JSONObject
putOpt(String key
, Object value
) throws JSONException
{
1113 if (key
!= null && value
!= null) {
1121 * Produce a string in double quotes with backslash sequences in all the
1122 * right places. A backslash will be inserted within </, allowing JSON
1123 * text to be delivered in HTML. In JSON text, a string cannot contain a
1124 * control character or an unescaped quote or backslash.
1125 * @param string A String
1126 * @return A String correctly formatted for insertion in a JSON text.
1128 public static String
quote(String string
) {
1129 if (string
== null || string
.length() == 0) {
1136 int len
= string
.length();
1137 StringBuffer sb
= new StringBuffer(len
+ 4);
1141 for (i
= 0; i
< len
; i
+= 1) {
1143 c
= string
.charAt(i
);
1172 if (c
< ' ' || (c
>= '\u0080' && c
< '\u00a0') ||
1173 (c
>= '\u2000' && c
< '\u2100')) {
1174 t
= "000" + Integer
.toHexString(c
);
1175 sb
.append("\\u" + t
.substring(t
.length() - 4));
1182 return sb
.toString();
1186 * Remove a name and its value, if present.
1187 * @param key The name to be removed.
1188 * @return The value that was associated with the name,
1189 * or null if there was no value.
1191 public Object
remove(String key
) {
1192 return this.map
.remove(key
);
1196 * Get an enumeration of the keys of the JSONObject.
1197 * The keys will be sorted alphabetically.
1199 * @return An iterator of the keys.
1201 public Iterator
sortedKeys() {
1202 return new TreeSet(this.map
.keySet()).iterator();
1206 * Try to convert a string into a number, boolean, or null. If the string
1207 * can't be converted, return the string.
1208 * @param s A String.
1209 * @return A simple JSON value.
1211 static public Object
stringToValue(String s
) {
1215 if (s
.equalsIgnoreCase("true")) {
1216 return Boolean
.TRUE
;
1218 if (s
.equalsIgnoreCase("false")) {
1219 return Boolean
.FALSE
;
1221 if (s
.equalsIgnoreCase("null")) {
1222 return JSONObject
.NULL
;
1226 * If it might be a number, try converting it. We support the 0- and 0x-
1227 * conventions. If a number cannot be produced, then the value will just
1228 * be a string. Note that the 0-, 0x-, plus, and implied string
1229 * conventions are non-standard. A JSON parser is free to accept
1230 * non-JSON forms as long as it accepts all correct JSON forms.
1233 char b
= s
.charAt(0);
1234 if ((b
>= '0' && b
<= '9') || b
== '.' || b
== '-' || b
== '+') {
1236 if (s
.length() > 2 &&
1237 (s
.charAt(1) == 'x' || s
.charAt(1) == 'X')) {
1239 return new Integer(Integer
.parseInt(s
.substring(2),
1241 } catch (Exception e
) {
1242 /* Ignore the error */
1246 return new Integer(Integer
.parseInt(s
, 8));
1247 } catch (Exception e
) {
1248 /* Ignore the error */
1253 if (s
.indexOf('.') > -1 || s
.indexOf('e') > -1 || s
.indexOf('E') > -1) {
1254 return Double
.valueOf(s
);
1256 Long myLong
= new Long(s
);
1257 if (myLong
.longValue() == myLong
.intValue()) {
1258 return new Integer(myLong
.intValue());
1263 } catch (Exception f
) {
1264 /* Ignore the error */
1272 * Throw an exception if the object is an NaN or infinite number.
1273 * @param o The object to test.
1274 * @throws JSONException If o is a non-finite number.
1276 static void testValidity(Object o
) throws JSONException
{
1278 if (o
instanceof Double
) {
1279 if (((Double
)o
).isInfinite() || ((Double
)o
).isNaN()) {
1280 throw new JSONException(
1281 "JSON does not allow non-finite numbers.");
1283 } else if (o
instanceof Float
) {
1284 if (((Float
)o
).isInfinite() || ((Float
)o
).isNaN()) {
1285 throw new JSONException(
1286 "JSON does not allow non-finite numbers.");
1294 * Produce a JSONArray containing the values of the members of this
1296 * @param names A JSONArray containing a list of key strings. This
1297 * determines the sequence of the values in the result.
1298 * @return A JSONArray of values.
1299 * @throws JSONException If any of the values are non-finite numbers.
1301 public JSONArray
toJSONArray(JSONArray names
) throws JSONException
{
1302 if (names
== null || names
.length() == 0) {
1305 JSONArray ja
= new JSONArray();
1306 for (int i
= 0; i
< names
.length(); i
+= 1) {
1307 ja
.put(this.opt(names
.getString(i
)));
1313 * Make a JSON text of this JSONObject. For compactness, no whitespace
1314 * is added. If this would not result in a syntactically correct JSON text,
1315 * then null will be returned instead.
1317 * Warning: This method assumes that the data structure is acyclical.
1319 * @return a printable, displayable, portable, transmittable
1320 * representation of the object, beginning
1321 * with <code>{</code> <small>(left brace)</small> and ending
1322 * with <code>}</code> <small>(right brace)</small>.
1324 public String
toString() {
1326 Iterator keys
= keys();
1327 StringBuffer sb
= new StringBuffer("{");
1329 while (keys
.hasNext()) {
1330 if (sb
.length() > 1) {
1333 Object o
= keys
.next();
1334 sb
.append(quote(o
.toString()));
1336 sb
.append(valueToString(this.map
.get(o
)));
1339 return sb
.toString();
1340 } catch (Exception e
) {
1347 * Make a prettyprinted JSON text of this JSONObject.
1349 * Warning: This method assumes that the data structure is acyclical.
1350 * @param indentFactor The number of spaces to add to each level of
1352 * @return a printable, displayable, portable, transmittable
1353 * representation of the object, beginning
1354 * with <code>{</code> <small>(left brace)</small> and ending
1355 * with <code>}</code> <small>(right brace)</small>.
1356 * @throws JSONException If the object contains an invalid number.
1358 public String
toString(int indentFactor
) throws JSONException
{
1359 return toString(indentFactor
, 0);
1364 * Make a prettyprinted JSON text of this JSONObject.
1366 * Warning: This method assumes that the data structure is acyclical.
1367 * @param indentFactor The number of spaces to add to each level of
1369 * @param indent The indentation of the top level.
1370 * @return a printable, displayable, transmittable
1371 * representation of the object, beginning
1372 * with <code>{</code> <small>(left brace)</small> and ending
1373 * with <code>}</code> <small>(right brace)</small>.
1374 * @throws JSONException If the object contains an invalid number.
1376 String
toString(int indentFactor
, int indent
) throws JSONException
{
1382 Iterator keys
= sortedKeys();
1383 StringBuffer sb
= new StringBuffer("{");
1384 int newindent
= indent
+ indentFactor
;
1388 sb
.append(quote(o
.toString()));
1390 sb
.append(valueToString(this.map
.get(o
), indentFactor
,
1393 while (keys
.hasNext()) {
1395 if (sb
.length() > 1) {
1400 for (j
= 0; j
< newindent
; j
+= 1) {
1403 sb
.append(quote(o
.toString()));
1405 sb
.append(valueToString(this.map
.get(o
), indentFactor
,
1408 if (sb
.length() > 1) {
1410 for (j
= 0; j
< indent
; j
+= 1) {
1416 return sb
.toString();
1421 * Make a JSON text of an Object value. If the object has an
1422 * value.toJSONString() method, then that method will be used to produce
1423 * the JSON text. The method is required to produce a strictly
1424 * conforming text. If the object does not contain a toJSONString
1425 * method (which is the most common case), then a text will be
1426 * produced by other means. If the value is an array or Collection,
1427 * then a JSONArray will be made from it and its toJSONString method
1428 * will be called. If the value is a MAP, then a JSONObject will be made
1429 * from it and its toJSONString method will be called. Otherwise, the
1430 * value's toString method will be called, and the result will be quoted.
1433 * Warning: This method assumes that the data structure is acyclical.
1434 * @param value The value to be serialized.
1435 * @return a printable, displayable, transmittable
1436 * representation of the object, beginning
1437 * with <code>{</code> <small>(left brace)</small> and ending
1438 * with <code>}</code> <small>(right brace)</small>.
1439 * @throws JSONException If the value is or contains an invalid number.
1441 static String
valueToString(Object value
) throws JSONException
{
1442 if (value
== null || value
.equals(null)) {
1445 if (value
instanceof JSONString
) {
1448 o
= ((JSONString
)value
).toJSONString();
1449 } catch (Exception e
) {
1450 throw new JSONException(e
);
1452 if (o
instanceof String
) {
1455 throw new JSONException("Bad value from toJSONString: " + o
);
1457 if (value
instanceof Number
) {
1458 return numberToString((Number
) value
);
1460 if (value
instanceof Boolean
|| value
instanceof JSONObject
||
1461 value
instanceof JSONArray
) {
1462 return value
.toString();
1464 if (value
instanceof Map
) {
1465 return new JSONObject((Map
)value
).toString();
1467 if (value
instanceof Collection
) {
1468 return new JSONArray((Collection
)value
).toString();
1470 if (value
.getClass().isArray()) {
1471 return new JSONArray(value
).toString();
1473 return quote(value
.toString());
1478 * Make a prettyprinted JSON text of an object value.
1480 * Warning: This method assumes that the data structure is acyclical.
1481 * @param value The value to be serialized.
1482 * @param indentFactor The number of spaces to add to each level of
1484 * @param indent The indentation of the top level.
1485 * @return a printable, displayable, transmittable
1486 * representation of the object, beginning
1487 * with <code>{</code> <small>(left brace)</small> and ending
1488 * with <code>}</code> <small>(right brace)</small>.
1489 * @throws JSONException If the object contains an invalid number.
1491 static String
valueToString(Object value
, int indentFactor
, int indent
)
1492 throws JSONException
{
1493 if (value
== null || value
.equals(null)) {
1497 if (value
instanceof JSONString
) {
1498 Object o
= ((JSONString
)value
).toJSONString();
1499 if (o
instanceof String
) {
1503 } catch (Exception e
) {
1504 /* forget about it */
1506 if (value
instanceof Number
) {
1507 return numberToString((Number
) value
);
1509 if (value
instanceof Boolean
) {
1510 return value
.toString();
1512 if (value
instanceof JSONObject
) {
1513 return ((JSONObject
)value
).toString(indentFactor
, indent
);
1515 if (value
instanceof JSONArray
) {
1516 return ((JSONArray
)value
).toString(indentFactor
, indent
);
1518 if (value
instanceof Map
) {
1519 return new JSONObject((Map
)value
).toString(indentFactor
, indent
);
1521 if (value
instanceof Collection
) {
1522 return new JSONArray((Collection
)value
).toString(indentFactor
, indent
);
1524 if (value
.getClass().isArray()) {
1525 return new JSONArray(value
).toString(indentFactor
, indent
);
1527 return quote(value
.toString());
1532 * Write the contents of the JSONObject as JSON text to a writer.
1533 * For compactness, no whitespace is added.
1535 * Warning: This method assumes that the data structure is acyclical.
1537 * @return The writer.
1538 * @throws JSONException
1540 public Writer
write(Writer writer
) throws JSONException
{
1543 Iterator keys
= keys();
1546 while (keys
.hasNext()) {
1550 Object k
= keys
.next();
1551 writer
.write(quote(k
.toString()));
1553 Object v
= this.map
.get(k
);
1554 if (v
instanceof JSONObject
) {
1555 ((JSONObject
)v
).write(writer
);
1556 } else if (v
instanceof JSONArray
) {
1557 ((JSONArray
)v
).write(writer
);
1559 writer
.write(valueToString(v
));
1565 } catch (IOException e
) {
1566 throw new JSONException(e
);