Как перебрать JSONObject?
Я использую библиотеку JSON под названием JSONObject
(Я не против переключения, если мне нужно).
Я знаю как перебирать JSONArrays
, но когда я анализирую данные JSON из Facebook, я не получаю массив, только JSONObject
, но мне нужно иметь возможность получить доступ к элементу через его индекс, такой как JSONObject[0]
чтобы получить первый, и я не могу понять, как это сделать.
{
"http://http://url.com/": {
"id": "http://http://url.com//"
},
"http://url2.co/": {
"id": "http://url2.com//",
"shares": 16
}
,
"http://url3.com/": {
"id": "http://url3.com//",
"shares": 16
}
}
18 ответов
Может быть, это поможет:
jsonObject = new JSONObject(contents.trim());
Iterator<String> keys = jsonObject.keys();
while(keys.hasNext()) {
String key = keys.next();
if (jObject.get(key) instanceof JSONObject) {
}
}
Для моего случая я нашел итерацию names()
работает хорошо
for(int i = 0; i<jobject.names().length(); i++){
Log.v(TAG, "key = " + jobject.names().getString(i) + " value = " + jobject.get(jobject.names().getString(i)));
}
Я буду избегать итераторов, поскольку они могут добавлять / удалять объекты во время итерации, а также для чистого кода использовать цикл for. меньше строки, простой код.
import org.json.simple.JSONObject;
public static void printJsonObject(JSONObject jsonObj) {
for (Object key : jsonObj.keySet()) {
//based on you key types
String keyStr = (String)key;
Object keyvalue = jsonObj.get(keyStr);
//Print key and value
System.out.println("key: "+ keyStr + " value: " + keyvalue);
//for nested objects iteration if required
if (keyvalue instanceof JSONObject)
printJsonObject((JSONObject)keyvalue);
}
}
Не верьте, что нет более простого и безопасного решения, чем использование итератора...
JSONObject names ()
Метод возвращает JSONArray ключей JSONObject, поэтому вы можете просто пройти по нему в цикле:
JSONObject object = new JSONObject ();
JSONArray keys = object.names ();
for (int i = 0; i < keys.length (); ++i) {
String key = keys.getString (i); // Here's your key
String value = object.getString (key); // Here's your value
}
Iterator<JSONObject> iterator = jsonObject.values().iterator();
while (iterator.hasNext()) {
jsonChildObject = iterator.next();
// Do whatever you want with jsonChildObject
String id = (String) jsonChildObject.get("id");
}
org.json.JSONObject теперь имеет метод keySet(), который возвращает Set<String>
и может быть легко зацикливаться на каждом.
for(String key : jsonObject.keySet())
Большинство ответов здесь для плоских структур JSON, если у вас есть JSON, который может иметь вложенные массивы JSONArrays или вложенные объекты JSONObject, возникает реальная сложность. Следующий фрагмент кода удовлетворяет такое бизнес-требование. Он принимает хеш-карту и иерархический JSON с вложенными JSONArrays и JSONObjects и обновляет JSON с данными в хеш-карте.
public void updateData(JSONObject fullResponse, HashMap<String, String> mapToUpdate) {
fullResponse.keySet().forEach(keyStr -> {
Object keyvalue = fullResponse.get(keyStr);
if (keyvalue instanceof JSONArray) {
updateData(((JSONArray) keyvalue).getJSONObject(0), mapToUpdate);
} else if (keyvalue instanceof JSONObject) {
updateData((JSONObject) keyvalue, mapToUpdate);
} else {
// System.out.println("key: " + keyStr + " value: " + keyvalue);
if (mapToUpdate.containsKey(keyStr)) {
fullResponse.put(keyStr, mapToUpdate.get(keyStr));
}
}
});
}
Здесь вы должны заметить, что возвращаемый тип this недействителен, но объекты sice передаются в качестве ссылки, это изменение отражается вызывающей стороне.
Сначала поместите это куда-нибудь:
private <T> Iterable<T> iteratorToIterable(final Iterator<T> iterator) {
return new Iterable<T>() {
@Override
public Iterator<T> iterator() {
return iterator;
}
};
}
Или, если у вас есть доступ к Java8, просто так:
private <T> Iterable<T> iteratorToIterable(Iterator<T> iterator) {
return () -> iterator;
}
Затем просто переберите ключи и значения объекта:
for (String key : iteratorToIterable(object.keys())) {
JSONObject entry = object.getJSONObject(key);
// ...
Я сделал небольшую рекурсивную функцию, которая проходит через весь объект json и сохраняет путь к ключу и его значение.
// My stored keys and values from the json object
HashMap<String,String> myKeyValues = new HashMap<String,String>();
// Used for constructing the path to the key in the json object
Stack<String> key_path = new Stack<String>();
// Recursive function that goes through a json object and stores
// its key and values in the hashmap
private void loadJson(JSONObject json){
Iterator<?> json_keys = json.keys();
while( json_keys.hasNext() ){
String json_key = (String)json_keys.next();
try{
key_path.push(json_key);
loadJson(json.getJSONObject(json_key));
}catch (JSONException e){
// Build the path to the key
String key = "";
for(String sub_key: key_path){
key += sub_key+".";
}
key = key.substring(0,key.length()-1);
System.out.println(key+": "+json.getString(json_key));
key_path.pop();
myKeyValues.put(key, json.getString(json_key));
}
}
if(key_path.size() > 0){
key_path.pop();
}
}
Однажды у меня был json, у которого были идентификаторы, которые нужно было увеличить на единицу, так как они были проиндексированы на 0, и это нарушало автоинкремент Mysql.
Таким образом, для каждого объекта, который я написал этот код - может быть полезным для кого-то:
public static void incrementValue(JSONObject obj, List<String> keysToIncrementValue) {
Set<String> keys = obj.keySet();
for (String key : keys) {
Object ob = obj.get(key);
if (keysToIncrementValue.contains(key)) {
obj.put(key, (Integer)obj.get(key) + 1);
}
if (ob instanceof JSONObject) {
incrementValue((JSONObject) ob, keysToIncrementValue);
}
else if (ob instanceof JSONArray) {
JSONArray arr = (JSONArray) ob;
for (int i=0; i < arr.length(); i++) {
Object arrObj = arr.get(0);
if (arrObj instanceof JSONObject) {
incrementValue((JSONObject) arrObj, keysToIncrementValue);
}
}
}
}
}
использование:
JSONObject object = ....
incrementValue(object, Arrays.asList("id", "product_id", "category_id", "customer_id"));
это может быть преобразовано для работы для JSONArray как родительский объект
Мы использовали ниже набор кода для итерации JSONObject
поля
Iterator iterator = jsonObject.entrySet().iterator();
while (iterator.hasNext()) {
Entry<String, JsonElement> entry = (Entry<String, JsonElement>) iterator.next();
processedJsonObject.add(entry.getKey(), entry.getValue());
}
С Java 8 и лямбда, чище:
JSONObject jObject = new JSONObject(contents.trim());
jObject.keys().forEachRemaining(k ->
{
});
https://docs.oracle.com/javase/8/docs/api/java/util/Iterator.html
Я создал свой небольшой метод для регистрации полей JsonObject и получения некоторых проблем. Посмотрим, может ли это быть полезным.
object JsonParser {
val TAG = "JsonParser"
/**
* parse json object
* @param objJson
* @return Map<String, String>
* @throws JSONException
*/
@Throws(JSONException::class)
fun parseJson(objJson: Any?): Map<String, String> {
val map = HashMap<String, String>()
// If obj is a json array
if (objJson is JSONArray) {
for (i in 0 until objJson.length()) {
parseJson(objJson[i])
}
} else if (objJson is JSONObject) {
val it: Iterator<*> = objJson.keys()
while (it.hasNext()) {
val key = it.next().toString()
// If you get an array
when (val jobject = objJson[key]) {
is JSONArray -> {
Log.e(TAG, " JSONArray: $jobject")
parseJson(jobject)
}
is JSONObject -> {
Log.e(TAG, " JSONObject: $jobject")
parseJson(jobject)
}
else -> {
Log.e(TAG, " adding to map: $key $jobject")
map[key] = jobject.toString()
}
}
}
}
return map
}
}
Более простой подход (только что найденный на W3Schools):
let data = {.....}; // JSON Object
for(let d in data){
console.log(d); // It gives you property name
console.log(data[d]); // And this gives you its value
}
ОБНОВИТЬ
Этот подход работает нормально, пока вы не разберетесь с вложенным объектом, поэтому этот подход будет работать.
const iterateJSON = (jsonObject, output = {}) => {
for (let d in jsonObject) {
if (typeof jsonObject[d] === "string") {
output[d] = jsonObject[d];
}
if (typeof jsonObject[d] === "object") {
output[d] = iterateJSON(jsonObject[d]);
}
}
return output;
}
И используйте такой метод
let output = iterateJSON(your_json_object);
Похоже, что предоставленные вами данные JSON представляют собой объект, где каждый ключ является URL-адресом, а соответствующее значение — другим объектом. Если вы хотите получить доступ к элементам по индексу, вы можете преобразовать внешний объект в массив. Однако имейте в виду, что в JavaScript объекты не имеют определенного порядка, поэтому преобразование их в массив может не сохранить порядок.
Предполагая, что вы работаете в среде JavaScript, вы можете преобразовать внешний объект в массив, а затем получать доступ к элементам по индексу. Вот пример:
const jsonData = {
"http://url.com/": {
"id": "http://url.com//"
},
"http://url2.co/": {
"id": "http://url2.com//",
"shares": 16
},
"http://url3.com/": {
"id": "http://url3.com//",
"shares": 16
}
};
// Convert the object to an array
const jsonArray = Object.entries(jsonData).map(([key, value]) => ({ key, value }));
// Access the first item
const firstItem = jsonArray[0];
// Accessing specific properties of the first item
const firstItemId = firstItem.value.id;
const firstItemShares = firstItem.value.shares;
console.log(firstItemId); // Output: http://url.com//
console.log(firstItemShares); // Output: undefined (shares property is not present in the first item)
В этом примере Object.entries используется для преобразования объекта в массив пар ключ-значение, а карта используется для преобразования его в массив объектов. Теперь вы можете получить доступ к элементам по индексу в jsonArray. Имейте в виду, что если порядок элементов важен, вы можете пересмотреть структуру ваших данных JSON или в первую очередь использовать массив, если это возможно.
Самый простой способ — использовать цикл for-each для набора ключей. Учитывая, что у вас есть строковое значение JSON, называемое «текст»:
JSONObject jsonObject = new JSONObject(text);
for (String key : jsonObject.keySet()) {
// Print out each value to verify this code
System.out.println(jsonObject.get(key));
}
Ниже код работал нормально для меня. Пожалуйста, помогите мне, если настройка может быть сделана. Это получает все ключи даже от вложенных объектов JSON.
public static void main(String args[]) {
String s = ""; // Sample JSON to be parsed
JSONParser parser = new JSONParser();
JSONObject obj = null;
try {
obj = (JSONObject) parser.parse(s);
@SuppressWarnings("unchecked")
List<String> parameterKeys = new ArrayList<String>(obj.keySet());
List<String> result = null;
List<String> keys = new ArrayList<>();
for (String str : parameterKeys) {
keys.add(str);
result = this.addNestedKeys(obj, keys, str);
}
System.out.println(result.toString());
} catch (ParseException e) {
e.printStackTrace();
}
}
public static List<String> addNestedKeys(JSONObject obj, List<String> keys, String key) {
if (isNestedJsonAnArray(obj.get(key))) {
JSONArray array = (JSONArray) obj.get(key);
for (int i = 0; i < array.length(); i++) {
try {
JSONObject arrayObj = (JSONObject) array.get(i);
List<String> list = new ArrayList<>(arrayObj.keySet());
for (String s : list) {
putNestedKeysToList(keys, key, s);
addNestedKeys(arrayObj, keys, s);
}
} catch (JSONException e) {
LOG.error("", e);
}
}
} else if (isNestedJsonAnObject(obj.get(key))) {
JSONObject arrayObj = (JSONObject) obj.get(key);
List<String> nestedKeys = new ArrayList<>(arrayObj.keySet());
for (String s : nestedKeys) {
putNestedKeysToList(keys, key, s);
addNestedKeys(arrayObj, keys, s);
}
}
return keys;
}
private static void putNestedKeysToList(List<String> keys, String key, String s) {
if (!keys.contains(key + Constants.JSON_KEY_SPLITTER + s)) {
keys.add(key + Constants.JSON_KEY_SPLITTER + s);
}
}
private static boolean isNestedJsonAnObject(Object object) {
boolean bool = false;
if (object instanceof JSONObject) {
bool = true;
}
return bool;
}
private static boolean isNestedJsonAnArray(Object object) {
boolean bool = false;
if (object instanceof JSONArray) {
bool = true;
}
return bool;
}
Это еще одно рабочее решение проблемы:
public void test (){
Map<String, String> keyValueStore = new HasMap<>();
Stack<String> keyPath = new Stack();
JSONObject json = new JSONObject("thisYourJsonObject");
keyValueStore = getAllXpathAndValueFromJsonObject(json, keyValueStore, keyPath);
for(Map.Entry<String, String> map : keyValueStore.entrySet()) {
System.out.println(map.getKey() + ":" + map.getValue());
}
}
public Map<String, String> getAllXpathAndValueFromJsonObject(JSONObject json, Map<String, String> keyValueStore, Stack<String> keyPath) {
Set<String> jsonKeys = json.keySet();
for (Object keyO : jsonKeys) {
String key = (String) keyO;
keyPath.push(key);
Object object = json.get(key);
if (object instanceof JSONObject) {
getAllXpathAndValueFromJsonObject((JSONObject) object, keyValueStore, keyPath);
}
if (object instanceof JSONArray) {
doJsonArray((JSONArray) object, keyPath, keyValueStore, json, key);
}
if (object instanceof String || object instanceof Boolean || object.equals(null)) {
String keyStr = "";
for (String keySub : keyPath) {
keyStr += keySub + ".";
}
keyStr = keyStr.substring(0, keyStr.length() - 1);
keyPath.pop();
keyValueStore.put(keyStr, json.get(key).toString());
}
}
if (keyPath.size() > 0) {
keyPath.pop();
}
return keyValueStore;
}
public void doJsonArray(JSONArray object, Stack<String> keyPath, Map<String, String> keyValueStore, JSONObject json,
String key) {
JSONArray arr = (JSONArray) object;
for (int i = 0; i < arr.length(); i++) {
keyPath.push(Integer.toString(i));
Object obj = arr.get(i);
if (obj instanceof JSONObject) {
getAllXpathAndValueFromJsonObject((JSONObject) obj, keyValueStore, keyPath);
}
if (obj instanceof JSONArray) {
doJsonArray((JSONArray) obj, keyPath, keyValueStore, json, key);
}
if (obj instanceof String || obj instanceof Boolean || obj.equals(null)) {
String keyStr = "";
for (String keySub : keyPath) {
keyStr += keySub + ".";
}
keyStr = keyStr.substring(0, keyStr.length() - 1);
keyPath.pop();
keyValueStore.put(keyStr , json.get(key).toString());
}
}
if (keyPath.size() > 0) {
keyPath.pop();
}
}