Як перевірити, чи є об’єкт JavaScript JSON


87

У мене є вкладений об'єкт JSON, який мені потрібно прокрутити, і значенням кожного ключа може бути String, масив JSON або інший об'єкт JSON. Залежно від типу об’єкта мені потрібно виконувати різні операції. Чи можу я якось перевірити тип об’єкта, чи це String, JSON-об’єкт чи масив JSON?

Я спробував з допомогою typeofі , instanceofа й , схоже , не на роботу, а typeofбуде повертати об'єкт як для об'єкта JSON і масиву, і instanceofвидає помилку , коли я роблю obj instanceof JSON.

Якщо бути більш точним, то після синтаксичного аналізу JSON на об’єкт JS, чи можу я якось перевірити, чи це звичайний рядок, чи об’єкт із ключами та значеннями (від об’єкта JSON), чи масив (із масиву JSON )?

Наприклад:

JSON

var data = "{'hi':
             {'hello':
               ['hi1','hi2']
             },
            'hey':'words'
           }";

Зразок JavaScript

var jsonObj = JSON.parse(data);
var path = ["hi","hello"];

function check(jsonObj, path) {
    var parent = jsonObj;
    for (var i = 0; i < path.length-1; i++) {
        var key = path[i];
        if (parent != undefined) {
            parent = parent[key];
        }
    }
    if (parent != undefined) {
        var endLength = path.length - 1;
        var child = parent[path[endLength]];
        //if child is a string, add some text
        //if child is an object, edit the key/value
        //if child is an array, add a new element
        //if child does not exist, add a new key/value
    }
}

Як здійснити перевірку об’єкта, як показано вище?


3
JSON - це просто позначення, що зберігається у вигляді рядка . Ви впевнені, що не плутаєте терміни?
zerkms

Ні, я оновив запитання, щоб було зрозуміліше. Я думаю, моє головне питання полягає в тому, що відбувається після того, як ми робимо .parse()рядок JSON, і як це ідентифікувати?
Вей Хао

1
зміни не дали зрозуміти (принаймні для мене). Що, якщо навести приклад JSON, з яким ви маєте справу
zerkms

Оновлене запитання на прикладі. (:
Вей Хао

Справжнє запитання: чому вам все одно?
Ашера

Відповіді:


130

Я б перевірив атрибут конструктора.

напр

var stringConstructor = "test".constructor;
var arrayConstructor = [].constructor;
var objectConstructor = ({}).constructor;

function whatIsIt(object) {
    if (object === null) {
        return "null";
    }
    if (object === undefined) {
        return "undefined";
    }
    if (object.constructor === stringConstructor) {
        return "String";
    }
    if (object.constructor === arrayConstructor) {
        return "Array";
    }
    if (object.constructor === objectConstructor) {
        return "Object";
    }
    {
        return "don't know";
    }
}

var testSubjects = ["string", [1,2,3], {foo: "bar"}, 4];

for (var i=0, len = testSubjects.length; i < len; i++) {
    alert(whatIsIt(testSubjects[i]));
}

Редагувати: Додано нульову перевірку та невизначену перевірку.


9
else ifє непотрібним
McSonk

Хіба це не те саме, що використання instanceof? developer.mozilla.org/pt-BR/docs/Web/JavaScript/Reference/…
Перейра,

@Pereira: JavaScript має деякі незрозумілі зморшки. Спробуйте "surely_this_is_a_string" екземпляр рядка.
Хлопець із програмування,

{}.constructorзмушує мене потрапляти ERROR TypeError: Cannot read property 'constructor' of undefinedв мою кутову програму.
okay_programmer

@ Minyc510: Схоже, поведінка браузера змінилася. Обгортання в дужки, здається, це виправляє. Я відредагував відповідь, щоб відобразити це.
Хлопець із програмування

25

Ви можете використовувати Array.isArray для перевірки масивів. Потім typeof obj == 'рядок' , а typeof obj == 'об'єкт' .

var s = 'a string', a = [], o = {}, i = 5;
function getType(p) {
    if (Array.isArray(p)) return 'array';
    else if (typeof p == 'string') return 'string';
    else if (p != null && typeof p == 'object') return 'object';
    else return 'other';
}
console.log("'s' is " + getType(s));
console.log("'a' is " + getType(a));
console.log("'o' is " + getType(o));
console.log("'i' is " + getType(i));

's' є рядком
'a' є масивом
'o' є об'єктом
'i' є іншим


5
Не забудьте врахувати цеtypeof null === 'object'
hugomg

[{ "name":[ {"key": "any key" } ] }] це також дійсний json, але його повернутий масив за вашим кодом. перевірити це - скрипка
Sudhir K Gupta

15

Об'єкт JSON - це об'єкт. Щоб перевірити, чи тип є типом об’єкта, оцініть властивість конструктора.

function isObject(obj)
{
    return obj !== undefined && obj !== null && obj.constructor == Object;
}

Те саме стосується всіх інших типів:

function isArray(obj)
{
    return obj !== undefined && obj !== null && obj.constructor == Array;
}

function isBoolean(obj)
{
    return obj !== undefined && obj !== null && obj.constructor == Boolean;
}

function isFunction(obj)
{
    return obj !== undefined && obj !== null && obj.constructor == Function;
}

function isNumber(obj)
{
    return obj !== undefined && obj !== null && obj.constructor == Number;
}

function isString(obj)
{
    return obj !== undefined && obj !== null && obj.constructor == String;
}

function isInstanced(obj)
{
    if(obj === undefined || obj === null) { return false; }

    if(isArray(obj)) { return false; }
    if(isBoolean(obj)) { return false; }
    if(isFunction(obj)) { return false; }
    if(isNumber(obj)) { return false; }
    if(isObject(obj)) { return false; }
    if(isString(obj)) { return false; }

    return true;
}

2
Ресурс, кодований JSON, не є об'єктом. Це струна. Тільки після того, як ви його розшифруєте або в Javascript JSON.parse()ресурс JSON стає об'єктом. Тому, якщо ви тестуєте ресурс, що надходить із сервера, на наявність JSON, найкраще спочатку перевірити String, потім if - це не a, <empty string>а потім після аналізу, якщо це об’єкт.
Hmerman6006

8

Якщо ви намагаєтеся перевірити тип an objectпісля розбору JSONрядка, я пропоную перевірити атрибут конструктора:

obj.constructor == Array || obj.constructor == String || obj.constructor == Object

Це буде набагато швидша перевірка, ніж typeof або instanceof.

Якщо бібліотека JSON не повертає об'єкти, побудовані з цими функціями, я б дуже підозріло це використовувати.


Набагато більш прямий підхід. Дякую! = D
Едуардо Лусіо

Бажана відповідь. Звідки ви отримуєте інформацію про переваги продуктивності?
Даніель Ф

@DanielF це було загальноприйнятою мудрістю ще в 12-му році, зараз все інакше, тому я не знаю, чи так це
JoshRagem

5

Відповідь @PeterWilkinson для мене не спрацював, оскільки конструктор для "набраного" об'єкта налаштований на ім'я цього об'єкта. Мені довелося працювати з typeof

function isJson(obj) {
    var t = typeof obj;
    return ['boolean', 'number', 'string', 'symbol', 'function'].indexOf(t) == -1;
}

4

Ви можете створити власний конструктор для синтаксичного аналізу JSON:

var JSONObj = function(obj) { $.extend(this, JSON.parse(obj)); }
var test = new JSONObj('{"a": "apple"}');
//{a: "apple"}

Потім перевірте instanceof, щоб перевірити, чи потрібен був спочатку розбір

test instanceof JSONObj

4

Для написання цієї проблеми я написав модуль npm. Він доступний тут :

object-types: модуль для пошуку типів буквальних об'єктів

Встановити

  npm install --save object-types


Використання

const objectTypes = require('object-types');

objectTypes({});
//=> 'object'

objectTypes([]);
//=> 'array'

objectTypes(new Object(true));
//=> 'boolean'

Погляньте, це має вирішити саме вашу проблему. Повідомте мене, якщо у вас виникнуть запитання! https://github.com/dawsonbotsford/object-types


2

Ви також можете спробувати проаналізувати дані, а потім перевірити, чи отримали ви об’єкт:

var testIfJson = JSON.parse(data);
if (typeOf testIfJson == "object")
{
//Json
}
else
{
//Not Json
}

2

Я поєдную оператор typeof з перевіркою атрибута конструктора (за Пітером):

var typeOf = function(object) {
    var firstShot = typeof object;
    if (firstShot !== 'object') {
        return firstShot;
    } 
    else if (object.constructor === [].constructor) {
        return 'array';
    }
    else if (object.constructor === {}.constructor) {
        return 'object';
    }
    else if (object === null) {
        return 'null';
    }
    else {
        return 'don\'t know';
    } 
}

// Test
var testSubjects = [true, false, 1, 2.3, 'string', [4,5,6], {foo: 'bar'}, null, undefined];

console.log(['typeOf()', 'input parameter'].join('\t'))
console.log(new Array(28).join('-'));
testSubjects.map(function(testSubject){
    console.log([typeOf(testSubject), JSON.stringify(testSubject)].join('\t\t'));
});

Результат:

typeOf()    input parameter
---------------------------
boolean     true
boolean     false
number      1
number      2.3
string      "string"
array       [4,5,6]
object      {"foo":"bar"}
null        null
undefined       

2

Я знаю, що це дуже давнє запитання з хорошими відповідями. Однак, здається, все одно можна додати до нього свої 2 ¢.

Якщо припустити, що ви намагаєтеся протестувати не сам об'єкт JSON, а рядок, який відформатований як JSON (що, здається, має місце у вашому випадку var data), ви можете скористатися наступною функцією, яка повертає логічне значення (є чи ні JSON '):

function isJsonString( jsonString ) {

  // This function below ('printError') can be used to print details about the error, if any.
  // Please, refer to the original article (see the end of this post)
  // for more details. I suppressed details to keep the code clean.
  //
  let printError = function(error, explicit) {
  console.log(`[${explicit ? 'EXPLICIT' : 'INEXPLICIT'}] ${error.name}: ${error.message}`);
  }


  try {
      JSON.parse( jsonString );
      return true; // It's a valid JSON format
  } catch (e) {
      return false; // It's not a valid JSON format
  }

}

Ось кілька прикладів використання функції вище:

console.log('\n1 -----------------');
let j = "abc";
console.log( j, isJsonString(j) );

console.log('\n2 -----------------');
j = `{"abc": "def"}`;
console.log( j, isJsonString(j) );

console.log('\n3 -----------------');
j = '{"abc": "def}';
console.log( j, isJsonString(j) );

console.log('\n4 -----------------');
j = '{}';
console.log( j, isJsonString(j) );

console.log('\n5 -----------------');
j = '[{}]';
console.log( j, isJsonString(j) );

console.log('\n6 -----------------');
j = '[{},]';
console.log( j, isJsonString(j) );

console.log('\n7 -----------------');
j = '[{"a":1, "b":   2}, {"c":3}]';
console.log( j, isJsonString(j) );

Запустивши наведений вище код, ви отримаєте такі результати:

1 -----------------
abc false

2 -----------------
{"abc": "def"} true

3 -----------------
{"abc": "def} false

4 -----------------
{} true

5 -----------------
[{}] true

6 -----------------
[{},] false

7 -----------------
[{"a":1, "b":   2}, {"c":3}] true

Будь ласка, спробуйте фрагмент нижче і повідомте нам, чи це вам підходить. :)

ВАЖЛИВО: функція, представлена ​​в цьому дописі, була адаптована з https://airbrake.io/blog/javascript-error-handling/syntaxerror-json-parse-bad-parsing, де ви можете знайти більше та цікаві подробиці про JSON.parse ( ) функція.

function isJsonString( jsonString ) {

  let printError = function(error, explicit) {
  console.log(`[${explicit ? 'EXPLICIT' : 'INEXPLICIT'}] ${error.name}: ${error.message}`);
  }


  try {
      JSON.parse( jsonString );
      return true; // It's a valid JSON format
  } catch (e) {
      return false; // It's not a valid JSON format
  }

}


console.log('\n1 -----------------');
let j = "abc";
console.log( j, isJsonString(j) );

console.log('\n2 -----------------');
j = `{"abc": "def"}`;
console.log( j, isJsonString(j) );

console.log('\n3 -----------------');
j = '{"abc": "def}';
console.log( j, isJsonString(j) );

console.log('\n4 -----------------');
j = '{}';
console.log( j, isJsonString(j) );

console.log('\n5 -----------------');
j = '[{}]';
console.log( j, isJsonString(j) );

console.log('\n6 -----------------');
j = '[{},]';
console.log( j, isJsonString(j) );

console.log('\n7 -----------------');
j = '[{"a":1, "b":   2}, {"c":3}]';
console.log( j, isJsonString(j) );


1

Спробуйте це

if ( typeof is_json != "function" )
function is_json( _obj )
{
    var _has_keys = 0 ;
    for( var _pr in _obj )
    {
        if ( _obj.hasOwnProperty( _pr ) && !( /^\d+$/.test( _pr ) ) )
        {
           _has_keys = 1 ;
           break ;
        }
    }

    return ( _has_keys && _obj.constructor == Object && _obj.constructor != Array ) ? 1 : 0 ;
}

Це працює для прикладу нижче

var _a = { "name" : "me",
       "surname" : "I",
       "nickname" : {
                      "first" : "wow",
                      "second" : "super",
                      "morelevel" : {
                                      "3level1" : 1,
                                      "3level2" : 2,
                                      "3level3" : 3
                                    }
                    }
     } ;

var _b = [ "name", "surname", "nickname" ] ;
var _c = "abcdefg" ;

console.log( is_json( _a ) );
console.log( is_json( _b ) );
console.log( is_json( _c ) );

1

Чому б не перевірити Number - трохи коротший і працює в IE / Chrome / FF / node.js

function whatIsIt(object) {
    if (object === null) {
        return "null";
    }
    else if (object === undefined) {
        return "undefined";
    }
    if (object.constructor.name) {
            return object.constructor.name;
    }
    else { // last chance 4 IE: "\nfunction Number() {\n    [native code]\n}\n" / node.js: "function String() { [native code] }"
        var name = object.constructor.toString().split(' ');
        if (name && name.length > 1) {
            name = name[1];
            return name.substr(0, name.indexOf('('));
        }
        else { // unreachable now(?)
            return "don't know";
        }
    }
}

var testSubjects = ["string", [1,2,3], {foo: "bar"}, 4];
// Test all options
console.log(whatIsIt(null));
console.log(whatIsIt());
for (var i=0, len = testSubjects.length; i < len; i++) {
    console.log(whatIsIt(testSubjects[i]));
}


0

Відповідь Петра з додатковою перевіркою! Звичайно, не на 100% гарантовано!

var isJson = false;
outPutValue = ""
var objectConstructor = {}.constructor;
if(jsonToCheck.constructor === objectConstructor){
    outPutValue = JSON.stringify(jsonToCheck);
    try{
            JSON.parse(outPutValue);
            isJson = true;
    }catch(err){
            isJson = false;
    }
}

if(isJson){
    alert("Is json |" + JSON.stringify(jsonToCheck) + "|");
}else{
    alert("Is other!");
}

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.