我有一个需要循环的嵌套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
}
}
如何进行如上所示的对象检查?
答案 0 :(得分:101)
我检查构造函数属性。
e.g。
var stringConstructor = "test".constructor;
var arrayConstructor = [].constructor;
var objectConstructor = {}.constructor;
function whatIsIt(object) {
if (object === null) {
return "null";
}
else if (object === undefined) {
return "undefined";
}
else if (object.constructor === stringConstructor) {
return "String";
}
else if (object.constructor === arrayConstructor) {
return "Array";
}
else if (object.constructor === objectConstructor) {
return "Object";
}
else {
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]));
}
编辑:添加了空检查和未定义的检查。
答案 1 :(得分:15)
您可以使用 Array.isArray 来检查数组。然后 typeof obj =='string', typeof obj =='object'。
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'是对象
'我'是其他
答案 2 :(得分:6)
如果在解析object
字符串后尝试检查JSON
的类型,我建议检查构造函数属性:
obj.constructor == Array || obj.constructor == String || obj.constructor == Object
这将是比typeof或instanceof更快的检查。
如果 JSON库没有返回使用这些函数构造的对象,我会非常怀疑它。
答案 3 :(得分:5)
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;
}
答案 4 :(得分:4)
您可以为JSON解析创建自己的构造函数:
var JSONObj = function(obj) { $.extend(this, JSON.parse(obj)); }
var test = new JSONObj('{"a": "apple"}');
//{a: "apple"}
然后检查instanceof以查看它是否需要原始解析
test instanceof JSONObj
答案 5 :(得分:4)
@PeterWilkinson的答案对我不起作用,因为“类型”对象的构造函数是根据该对象的名称定制的。我必须使用typeof
function isJson(obj) {
var t = typeof obj;
return ['boolean', 'number', 'string', 'symbol', 'function'].indexOf(t) == -1;
}
答案 6 :(得分:4)
我写了一个npm模块来解决这个问题。它可用here:
self.toolbar.setOrientation(QtCore.Qt.Vertical)
:一个模块,用于查找底层对象的文字类型
object-types
npm install --save object-types
看一下,它应该可以解决您的确切问题。如果您有任何疑问,请告诉我! https://github.com/dawsonbotsford/object-types
答案 7 :(得分:2)
你也可以尝试解析数据,然后检查你是否有对象:
var testIfJson = JSON.parse(data);
if (typeOf testIfJson == "object")
{
//Json
}
else
{
//Not Json
}
答案 8 :(得分: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) );
答案 9 :(得分: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 ) );
答案 10 :(得分:1)
我将typeof运算符与构造函数属性的检查结合起来(由Peter完成):
1. REGISTER '/home/training/pig_xml.jar'
2. xml_input_data = load '/home/training/project/StatewiseDistrictwisePhysicalProgress.xml' using pig.XML.newloader('row') as (x:chararray);
`3. data_from_xml_ip = foreach xml_input_data GENERATE FLATTEN(REGEX_EXTRACT_ALL(x,'<row>\\s*<State_Name>(.*)</State_Name>\\s*<District_Name>(.*)</District_Name>\\s*<Project_Objectives_IHHL_BPL>(.*)</Project_Objectives_IHHL_BPL>\\s*<Project_Objectives_IHHL_APL>(.*)</Project_Objectives_IHHL_APL>\\s*<Project_Objectives_IHHL_TOTAL>(.*)</Project_Objectives_IHHL_TOTAL>\\s*<Project_Objectives_SCW>(.*)</Project_Objectives_SCW>\\s*<Project_Objectives_School_Toilets>(.*)</Project_Objectives_School_Toilets>\\s*<Project_Objectives_Anganwadi_Toilets>(.*)</Project_Objectives_Anganwadi_Toilets>\\s*<Project_Objectives_RSM>(.*)</Project_Objectives_RSM>\\s*<Project_Objectives_PC>(.*)</Project_Objectives_PC>\\s*<Project_Performance-IHHL_BPL>(.*)</Project_Performance-IIHL_BPL>\\s*<Project_Performance-IHHL_APL>(.*)</Project_Performance-IHHL_APL>\\s*<Project_Performance-IHHL_TOTAL>(.*)</Project_Performance-IHHL_TOTAL>//s*<Project_Performance-SCW>(.*)</Project_Performance-SCW>\\s*<Project_performance-School_Toilets>(.*)</Project_Performance-School_Toilets>\\s*<Project_Performance-Anganwadi_Toilets>(.*)</Project_Performance-Anganwadi_Toilets>\\s*<Project_Performance-RSM>(.*)</Project_Performance-RSM>\\s*<Project_Performance-PC>(.*)</Project_Performance-PC>\\s*</row>'));`"
结果:
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'));
});
答案 11 :(得分:0)
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!");
}
答案 12 :(得分:0)
为什么不检查数字-短一点并且可以在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]));
}
答案 13 :(得分:0)
基于@Martin Wantke的答案,但有一些建议的改进/调整...
select t2.*
from table2 t2
where not exists (select 1
from table1 t1
where t2.room = t1.room
);
注意::我发现这种方法很有说服力,所以我提交了这个答案。
答案 14 :(得分:0)
对此我有一个非常懒惰的答案,如果您尝试解析字符串/其他值,它不会抛出错误。
const checkForJson = (value) => {
if (typeof value !== "string") return false;
return value[0] === "{" && value[value.length - 1] === "}";
}
你可以用它来检查你的键的值,同时你做一些递归函数;抱歉,如果这不能完全回答问题
Ofc 这不是最优雅的解决方案,当字符串实际以“{”开头并以“}”结尾时会失败,尽管这些用例很少见,如果你真的想要,你可以检查是否存在引用或其他废话...无论如何,请自行决定使用。
TLDR:它不是万无一失的,但它很简单,适用于绝大多数用例。
答案 15 :(得分:-3)
尝试这种肮脏的方式
('' + obj).includes('{')