JSON只是一种字符串数据格式,使用它的不单单是Javascript。git
JSON能够表示三种类型的值:简单值、对象、数组github
如下是JSON能够辨识的简单值例子:web
5 "Hello World!" false null
*注意:编程
在Javascript中,咱们能够用对象字面量的方式建立一个对象:数组
var person = { name: "Sue", age: "18" }
但JSON要求给属性加双引号,上述对象转换成合法的JSON字符串格式:浏览器
{ "name": "Sue", "age": "18" }
*注意:服务器
在Javascript中,咱们能够建立一个以下数组:函数
var names = ["Sue", "Jane", "Ben"]
上述数组转换成合法的JSON字符串格式:this
["Sue", "Jane", "Ben"]
咱们能够将数组、简单值、对象嵌套使用,建立更加复杂的JSON字符串,好比:prototype
{ "teachers": [ { "name": "Sue", "age": 18, "students": [1, 45, 60]}, { "name": "Ben", "age": 25, "students": [2, 31, 40]} ], "students": [ { "id": 1, "name": "Jane"}, { "id": 2, "name": "Lee"} ] }
在JSON以前,web传输格式化数据都是使用XML,Javascript要想拿到XML中的数据,要先将其转换成DOM,然后从中提取数据,相比之下,JSON就显得很是简单。
早期解析JSON通常使用eval()
,好比:
eval({"name": "Sue"}) {name: "Sue"}
但这种方式存在风险,由于服务器返回的数据颇有可能存在恶意代码,只要被解析成合法的Javascript语法,就会被执行,好比:
eval("alert('Sue')")
所以不建议使用eval()
来解析JSON字符串。
早期浏览器中,可使用https://github.com/douglascro...。
在IE 8+、 Firefox 3.5+、 Safari 4+、 Chrome、Opera 10.5+中,ECMAScript定义了全局对象JSON。
用于把Javascript对象序列化为JSON字符串。
*注意:
undefined
的属性会被忽略举个例子:
function Person(name, secret) { this.name = name; this.secret = secret; this.makeFriends = function() { return this.secret; } } Person.prototype.sayHi = function() { return this.name; } var me = new Person("Sue", undefined) me.makeFriends() // undefined me.sayHi() // "Sue" var meJSON = JSON.stringify(me); // "{"name":"Sue"}"
上述例子中,咱们建立了一个Person
实例me
,其中secret
属性为undefined
,包含makeFriends
方法和一个原型成员sayHi
,这些都没有被添加到生成的JSON字符串中。
这个方法可使咱们没必要在意Javascript语法与JSON语法的差别,尽管建立合法的Javascript对象。
用于把JSON字符串序列化为Javascript值。
var meCopy = JSON.parse(meJSON) // {name: "Sue"}
*注意:me
与meCopy
是两个独立的对象,实际编程中,可使用stringify
parse
实现对象的拷贝。
JSON.stringify()
的第一个参数是要序列化的对象,后面还能够加两个参数,分别是过滤器和选项。
过滤器能够是数组,也能够是函数,若是是数组,JSON.stringify()
的结果中将保留数组中的属性,好比:
function Person(name, age, secret) { this.name = name; this.age = age; this.secret = secret; } var me = new Person("Sue", 18, "none"); var meJSON = JSON.stringify(me, ["name", "age"]) meJSON // "{"name":"Sue","age":18}"
若是过滤器是一个函数,会给这个函数传入两个参数,分别是属性名和属性值,根据须要返回要添加到JSON中的属性值,若是为undefined
,会移除该属性,好比:
var me = new Person("sue", 25, ["girl", "1024", "Beauty"]) function process(key, value) { switch(key) { case "name": return value.charAt(0).toUpperCase() + value.slice(1); case "age": return 18; case "secret": return undefined; default: return value; } } var meJSON = JSON.stringify(me, process) meJSON // "{"name":"Sue","age":18}"
第三个参数用于控制结果中的缩进和空白符,若是这个参数是一个数值,那么它表示每一个级别缩进的空格数,好比:
var meJSON = JSON.stringify(me, process, 4) undefined meJSON "{ "name": "Sue", "age": 18 }"
*注意:
若是第三个参数是一个字符串,那么将使用这个字符串做为缩进,这个字符串能够是制表符或其余任意字符,若是超过10位,则仅前10位有效,好比:
var meJSON = JSON.stringify(me, process, "\t") meJSON "{ "name": "Sue", "age": 18 }"
var meJSON = JSON.stringify(me, process, "****") meJSON "{ ****"name": "Sue", ****"age": 18 }"
当须要为某种对象添加自定义的序列化方法时,能够给对象定义toJSON()
方法,好比:
Person.prototype.toJSON = function() { return this.name; } var me = new Person("sue", 25, ["girl", "1024", "Beauty"]) JSON.stringify(me, process, "\t") // ""sue""
上述例子中,咱们在Person
的原型中添加了toJSON
方法,发现调用JSON.stringify
时,只序列化了name
属性。
序列化的内部顺序以下:
No. 1 若是存在toJSON()
,并且它能返回有效的值,则调用它,不然,返回对象自己
No. 2 若是提供了过滤器参数,则基于第一步返回的值调用过滤器
No. 3 序列化第二步的返回值
No. 4 若是提供了缩进,则格式化第三步的返回值
JSON.parse()
能够接收第二个参数,这个参数是一个还原函数,将在每一个键值对上调用,这个函数接收两个参数,分别是键和值,返回处理过的值,好比:
var birth = new Date(1993, 10, 24) JSON.stringify(birth) // ""1993-11-23T16:00:00.000Z"" var birthJSON = JSON.stringify(birth) var birthCopy = JSON.parse(birthJSON, function(key, value) { return new Date(value) }) birthCopy.getFullYear() // 1993