进阶篇:JSON & 对象 (7-1)

饥人谷学习进阶第 7 天

对象

一种无序的数据集合,由若干个“键值对”(key-value)构成。

var obj = {
    key: 'value'
}

如上述代码,大括号定义了一个对象,被赋值给变量obj。对象内部包含一个键值对(又称为成员)。键名和键值之间可以用冒号分隔。如果对象内部包含多个键值对,每个键值对之间用逗号分隔,最后一个键值对后面的逗号可写可不写。

{key: value} 是JS对象字面量写法
类似:[] 是数组字面量的写法

基本使用

//声明赋值一个对象:
var person = {
    name: '小明',
    age: 3,
    sayHello: function() {
        console.log('hello world')
    }
}

//获取对象属性值的方法:
person.name 
person['name']
对象名.属性名 / 对象名['属性名']
person.sayHello() //调用对象函数成员

//赋值方法&通过赋值直接添加对象成员
person.addr = '广州市'
person['business'] = '前端课程'

//for-in遍历
for(var key in person) {
    console.log(key)
    console.log(person[key])
}

详细介绍

  • 键名
    对象的所有键名都是字符串,所以不加引号也可以
    如果键名是数值,会自动转为字符串
var num = {
    1: 'a',
    2.2: 'b',
    3e3: 'c',
    4e-1: 'd',
    .567: true,
    0xFF: false
}

如果键名不符合标识名的条件(比如第一个字符为数字或者含有空格或运算符),也不是数字,则必须加上引号,否则会报错

var f = {
    '1a': "1a",
    'm v': "MV",
    'c+d': "C+D"
}//上面三个键名都不符合标识名的条件,必须加上引号

注意:JS的保留字for class等可以不加引号当作键名

  • 属性
    对象的每一个键名又称为属性,其键值可以是任何数据类型。如果一个属性的值为函数,通常把这个属性称为“方法”,它可以像函数那样调用。如:前面提到的person.sayHello()
    属性可以动态创建,不必在对象声明时就指定
var obj = {};
obj.foo = 123;
obj.foo //123

上面也提到了,直接对obj对象的foo属性赋值,结果会在运行时创建foo属性

  • 对象的引用
    如果不同的变量名指向同一个对象,那么它们都是这个对象的引用,也就是说指向同一个内存地址。修改其中一个变量,会影响到其他所有变量。
var a = {};
var b = a;
a.name = 'A';
b.name //'A'

a b 指向同一个对象,为其中任何一个变量添加属性,另一个变量都可以读写该属性。此时如果取消某一个变量对于原对象的引用,不会影响到另一个变量。
但是,这种引用只局限于对象,对于原始类型的数据则是传值引用,也就是说,都是值的拷贝。

var x = 1;
var y = x;
var x = 2;
y     //1
  • 表达式or语句
{ foo: 123 }

JavaScript引擎读到上面这行代码,会发现可能有两种含义。第一种可能是,这是一个表达式,表示一个包含foo属性的对象;第二种可能是,这是一个语句,表示一个代码区块,里面有一个标签foo,指向表达式123。
为了避免这种歧义,JavaScript规定,如果行首是大括号,一律解释为语句(即代码块)。如果要解释为表达式(即对象),必须在大括号前加上圆括号。

({ foo: 123 })

eval语句中反映明显差异

eval('{foo: 123}') // 123
eval('({foo: 123})') // {foo: 123}

上面代码中,如果没有圆括号,eval将其理解为一个代码块;加上圆括号以后,就理解成一个对象。

属性操作

  • 读取属性
    读取对象的属性,有两种方法,一种是使用点运算符,还有一种是使用方括号运算符。
    前面有提到
    注意,如果使用方括号运算符,键名必须放在引号里面,否则会被当作变量处理。
    但是,数字键可以不加引号,因为会被当作字符串处理。
    o['0.7'] & o[0.7] 都是可以的
    方括号运算符内部可以使用表达式。
    o['na'+'me'] & o[3+3]
    数值键名不能使用点运算符(因为会被当成小数点),只能使用方括号运算符。
    obj[0xFF] //true

  • 检查变量是否声明
    如果读取一个不存在的键,会返回undefined,而不是报错。可以利用这一点,来检查一个全局变量是否被声明。

// 检查a变量是否被声明
if (a) {...} // 报错
if (window.a) {...} // 不报错
if (window['a']) {...} // 不报错

上面的后二种写法之所以不报错,是因为在浏览器环境,所有全局变量都是window对象的属性。window.a的含义就是读取window对象的a属性,如果该属性不存在,就返回undefined,并不会报错。
需要注意的是,后二种写法有漏洞,如果a属性是一个空字符串(或其他对应的布尔值为false的情况),则无法起到检查变量是否声明的作用。正确的做法是可以采用下面的写法

if ('a' in window) {
  // 变量 a 声明过
} else {
  // 变量 a 未声明
}
  • 属性的赋值
    点运算符和方括号运算符,不仅可以用来读取值,还可以用来赋值。
o.p = 'abc';
o['p'] = 'abc';

JavaScript允许属性的“后绑定”,也就是说,你可以在任意时刻新增属性,没必要在定义对象的时候,就定义好属性。

var o = { p: 1 };
// 等价于
var o = {};
o.p = 1;
  • 查看所有属性
    查看一个对象本身的所有属性,可以使用Object.keys方法。
var o = {
  key1: 1,
  key2: 2
};
Object.keys(o);
// ['key1', 'key2']
  • delete命令
    delete命令用于删除对象的属性,删除成功后返回true
var o = {p: 1};
Object.keys(o) // ["p"]
delete o.p // true
o.p // undefined
Object.keys(o) // []

注意,删除一个不存在的属性,delete不报错,而且返回true

var o = {};
delete o.p // true

因此,不能根据delete命令的结果,认定某个属性是存在的,只能保证读取这个属性肯定得到undefined
delete命令不能删除var命令声明的变量,只能用来删除属性。因为var声明的全局变量都是顶层对象的属性,而且默认不得删除。

var p = 1;
delete p // false
delete window.p // false
  • for-in循环遍历
    遍历一个对象的全部属性
for(var key in obj) {
    console.log(obj[key]);
} 

JSON

JSON 格式

JSON 格式(JavaScript Object Notation 的缩写)是一种用于数据交换的文本格式,2001年由 Douglas Crockford 提出,目的是取代繁琐笨重的 XML 格式。

JSON 对值的类型和格式有严格的规定。

  1. 复合类型的值只能是数组或对象,不能是函数、正则表达式对象、日期对象。
  2. 简单类型的值只有四种:字符串、数值(必须以十进制表示)、布尔值和null(不能使用NaN, Infinity, -Infinityundefined)。
  3. 字符串必须使用双引号表示,不能使用单引号。
  4. 对象的键名必须放在双引号里面。
  5. 数组或对象最后一个成员的后面,不能加逗号。

需要注意的是,空数组和空对象都是合格的 JSON 值,null本身也是一个合格的 JSON 值。

ES5 新增了JSON对象,用来处理 JSON 格式数据。它有两个方法:JSON.stringify()JSON.parse()

JSON.stringify()

基本用法

JSON.stringify方法用于将一个值转为字符串。该字符串符合 JSON 格式,并且可以被JSON.parse方法还原。

需要注意的是,对于原始类型的字符串,转换结果会带双引号。如下:将来还原的时候,双引号可以让 JavaScript 引擎知道,foo是一个字符串,而不是一个变量名。

JSON.stringify('foo') === ""foo""

如果原始对象中,有一个成员的值是undefined、函数或 XML 对象,这个成员会被过滤。

var obj = {
    a: undefined,
    b: function () {}
};
JSON.stringify(obj)  //"{}"

如果数组的成员是undefined、函数或 XML 对象,则这些值被转成null

var arr = [undefined, function() {}];
JSON.stringify(arr)  //"[null,null]"

正则对象会被转成空对象。

JSON.stringify(/foo/)  //"{}"

JSON.stringify方法会忽略对象的不可遍历属性。

JSON.parse()

JSON.parse方法用于将JSON字符串转化成对象。

JSON.parse('{}') // {}
JSON.parse('true') // true
JSON.parse('"foo"') // "foo"
JSON.parse('[1, 5, "false"]') // [1, 5, "false"]
JSON.parse('null') // null

var o = JSON.parse('{"name": "张三"}');
o.name // 张三

如果传入的字符串不是有效的JSON格式,JSON.parse方法将报错。

JSON.parse("'String'") // illegal single quotes
// SyntaxError: Unexpected token ILLEGAL

上面代码中,双引号字符串中是一个单引号字符串,因为单引号字符串不符合JSON格式,所以报错。

深拷贝的另一种写法

var obj = {
    name: 'jirengu',
    age: 3,
    friends: ['a','b','c']
}

var obj2 = JSON.parse(JSON.stringify(obj))
obj.age = 4
console.log(obj2.age) //3

因为已经将obj对象转换成字符串,再转换成对象,所以转换之后的obj已经不是转换之前的obj了。

JavaScript对象和JSON的关系

JavaScript 对象的字面量写法只是长的像 JSON 格式数据,二者属于不同的范畴,JavaScript 对象中很多类型(函数、正则、Date) JSON 格式的规范并不支持,JavaScript 对象的字面量写法更宽松。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,544评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,430评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,764评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,193评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,216评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,182评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,063评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,917评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,329评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,543评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,722评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,425评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,019评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,671评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,825评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,729评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,614评论 2 353

推荐阅读更多精彩内容

  • 概要 64学时 3.5学分 章节安排 电子商务网站概况 HTML5+CSS3 JavaScript Node 电子...
    阿啊阿吖丁阅读 9,182评论 0 3
  • 对象 简单说,所谓对象,就是一种无序的数据集合,由若干个“键值对”(key-value)构成。 varobj = ...
    那个汉堡在对我笑阅读 504评论 0 0
  • 对象 简单说,所谓对象,就是一种无序的数据集合,由若干个“键值对”(key-value)构成。 上面代码中,大括号...
    红豆丁244阅读 185评论 0 0
  • 数值 判断NaN更可靠的方法是,利用NaN为唯一不等于自身的值的这个特点,进行判断。 isFinite方法返回一个...
    guyigg阅读 1,157评论 0 2
  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 4,556评论 0 5