eval解析JSON字符串的一个小问题,深入理解

eval解析JSON字符串的一个小问题

2016/02/24 · JavaScript
· JSON

原文出处: 韩子迟   

之前写过一篇 关于 JSON
的介绍文章,里面谈到了
JSON 的解析。我们都知道,高级浏览器可以用  JSON.parse() API 将一个 JSON 字符串解析成
JSON 数据,稍微欠妥点的做法,我们可以用 eval() 函数。

JavaScript

var str = ‘{“name”: “hanzichi”, “age”: 10}’; var obj = eval(‘(‘ + str +
‘)’); console.log(obj); // Object {name: “hanzichi”, age: 10}

1
2
3
var str = ‘{"name": "hanzichi", "age": 10}’;
var obj = eval(‘(‘ + str + ‘)’);
console.log(obj); // Object {name: "hanzichi", age: 10}

 

是否注意到,向 eval() 传参时,str 变量外裹了一层小括号?为什么要这样做?

我们先来看看 eval 函数的定义以及使用。

eval() 的参数是一个字符串。如果字符串表示了一个表达式,eval()
会对表达式求值。如果参数表示了一个或多个 JavaScript 声明, 那么 eval()
会执行声明。不要调用 eval() 来为算数表达式求值; JavaScript
会自动为算数表达式求值。

简单地说,eval 函数的参数是一个字符串,如果把字符串 “noString”
化处理,那么得到的将是正常的可以运行的 JavaScript 语句。

怎么说?举个栗子,如下代码:

JavaScript

var str = “alert(‘hello world’)”; eval(str);

1
2
var str = "alert(‘hello world’)";
eval(str);

执行后弹出 “hello world”。我们把 str 变量 “noString”
化,粗暴点的做法就是去掉外面的引号,内部调整(转义等),然后就变成了:

JavaScript

alert(‘hello world’)

1
alert(‘hello world’)

very good!这是正常的可以运行的 JavaScript 语句!运行之!

再回到开始的问题,为什么 JSON
字符串要裹上小括号。如果不加,是这个样子的:

JavaScript

var str = ‘{“name”: “hanzichi”, “age”: 10}’; var obj = eval(str); //
Uncaught SyntaxError: Unexpected token :

1
2
var str = ‘{"name": "hanzichi", "age": 10}’;
var obj = eval(str);  // Uncaught SyntaxError: Unexpected token :

恩,报错了。为什么会报错?试试把 str “noString” 化,执行一下:

JavaScript

{“name”: “hanzichi”, “age”: 10}; // Uncaught SyntaxError: Unexpected
token :

1
{"name": "hanzichi", "age": 10};  // Uncaught SyntaxError: Unexpected token :

毫无疑问,一个 JSON 对象或者说是一个对象根本就不是能执行的 JavaScript
语句!等等,试试以下代码:

JavaScript

var str = ‘{name: “hanzichi”}’; var obj = eval(str); console.log(obj);
// hanzichi

1
2
3
var str = ‘{name: "hanzichi"}’;
var obj = eval(str);
console.log(obj); // hanzichi

这又是什么鬼?但是给 name 加上 “” 又报错?

JavaScript

var str = ‘{“name”: “hanzichi”}’; var obj = eval(str); // Uncaught
SyntaxError: Unexpected token : console.log(obj);

1
2
3
var str = ‘{"name": "hanzichi"}’;
var obj = eval(str);  // Uncaught SyntaxError: Unexpected token :
console.log(obj);

好吧,快晕了,其实还是可以将 str “nostring” 化,看看是不是能正确执行的
JavaScript 语句。前者的结果是:

JavaScript

{name: “hanzichi”}

1
{name: "hanzichi"}

这确实是一条合法的 JavaScript 语句。{} 我们不仅能在 if、for
语句等场景使用,甚至可以在任何时候,因为 ES6 之前 JavaScript
只有块级作用域,所以对于作用域什么的并不会有什么冲突。去掉 {}
后 name: "hanzichi"也是合法的语句,一个 label 语句,label
语句在跳出嵌套的循环中非常好用,具体可以参考 label,而作为
label 语句的标记,name 是不能带引号的,标记能放在 JavaScript
代码的任何位置,用不到也没关系。

一旦一个对象有了两个 key,比如 {name: “hanzichi”, age: 10} ,ok,两个 label 语句?将
“hanzhichi” 以及 10
分别看做是语句,但是 语句之间只能用封号连接!(表达式之间才能用逗号)。所以改成下面这样也是没有问题的:

JavaScript

var str = ‘{name: “hanzichi”; age: 10}’; var obj = eval(str);
console.log(obj); // 10

1
2
3
var str = ‘{name: "hanzichi"; age: 10}’;
var obj = eval(str);  
console.log(obj); // 10

越扯越远,文章开头代码的错误的原因是找到了,为什么套个括号就能解决呢?简单来说,()
会把语句转换成表达式,称为语句表达式。括号里的代码都会被转换为表达式求值并且返回,对象字面量必须作为表达式而存在

本文并不会大谈表达式,关于表达式,可以参考文末链接。值得记住的一点是,表达式永远有一个返回值。大部分表达式会包裹在()
内,小括号内不能为空,如果有多个表达式,用逗号隔开,也就是所谓的逗号表达式,会返回最后一个的值。

说到表达式,不得不提函数表达式,以前翻译过一篇关于立即执行函数表达式的文章,可以参考下,文末。

Read More:

  • [译]JavaScript中:表达式和语句的区别
  • (译)详解javascript立即执行函数表达式(IIFE)
  • 深入探究javascript的 {}
    语句块

    1 赞 1 收藏
    评论

图片 1

你会用setTimeout吗

2016/03/22 · JavaScript
· 1 评论 ·
settimeout

本文作者: 伯乐在线 –
唐光尧
。未经作者许可,禁止转载!
欢迎加入伯乐在线 专栏作者。

教科书里面的setTimeout

定义很简单
setTimeout() 方法用于在指定的毫秒数后调用函数或计算表达式。

广泛应用场景
定时器,轮播图,动画效果,自动滚动等等

上面一些应该是setTimeout在大家心中的样子,因为我们平常使用也不是很多。

但是setTimeout真的有那么简单吗?

测试题

一个题目,如果你在一段代码中发现下面内容

var startTime = new Date(); setTimeout(function () { console.log(new
Date() – startTime); }, 100)

1
2
3
4
var startTime = new Date();
setTimeout(function () {
    console.log(new Date() – startTime);
}, 100)

请问最后打印的是多少?
我觉得正确答案是,取决于后面同步执行的js需要占用多少时间。
MAX(同步执行的时间, 100)

再加一个题目,只有下面代码

setTimeout(function () { func1(); }, 0) func2();

1
2
3
4
setTimeout(function () {
    func1();
}, 0)
func2();

func1和func2谁会先执行?

这个答案应该比较简单,func2先执行,func1后面执行。

再来一题

setTimeout(function () { func1() }, 0)

1
2
3
setTimeout(function () {
    func1()
}, 0)

setTimeout(function () { func1() })

1
2
3
setTimeout(function () {
    func1()
})

有什么差别?

0秒延迟,此回调将会放到一个能立即执行的时段进行触发。javascript代码大体上是自顶向下的,但中间穿插着有关DOM渲染,事件回应等异步代码,他们将组成一个队列,零秒延迟将会实现插队操作。
不写第二个参数,浏览器自动配置时间,在IE,FireFox中,第一次配可能给个很大的数字,100ms上下,往后会缩小到最小时间间隔,Safari,chrome,opera则多为10ms上下。

上面答案来自《javascript框架设计》

好了,看了上面几个题目是不是感觉setTimeout不是想象中那样了。

深入理解 JSON

2017/03/29 · JavaScript
· 2 评论 ·
JSON

原文出处:
Apriltail   

我们先来看一个JS中常见的JS对象序列化成JSON字符串的问题,请问,以下JS对象通过JSON.stringify后的字符串是怎样的?先不要急着复制粘贴到控制台,先自己打开一个代码编辑器或者纸,写写看,写完再去仔细对比你的控制台输出,如果有误记得看完全文并评论,哈哈。

JavaScript

var friend={ firstName: ‘Good’, ‘lastName’: ‘Man’, ‘address’: undefined,
‘phone’: [“1234567”,undefined], ‘fullName’: function(){ return
this.firstName + ‘ ‘ + this.lastName; } };
JSON.stringify(friend);//这一行返回什么呢?

1
2
3
4
5
6
7
8
9
10
11
var friend={  
    firstName: ‘Good’,
    ‘lastName’: ‘Man’,
    ‘address’: undefined,
    ‘phone’: ["1234567",undefined],
    ‘fullName’: function(){
        return this.firstName + ‘ ‘ + this.lastName;
    }
};
 
JSON.stringify(friend);//这一行返回什么呢?

第二个问题,如果我想在最终JSON字符串将这个’friend’的姓名全部变成大写字母,也就是把”Good”变成”GOOD”,把”Man”变成”MAN”,那么可以怎么做?

基于以上两个问题,我们再追本溯源问一下,JSON究竟是什么东西?为什么JSON就是易于数据交换?JSON和JS对象的区别?JS中JSON.parseJSON.stringify和不常见的toJSON,这几个函数的参数和处理细节到底是怎样的?

欢迎进入本次“深挖JSON之旅”,下文将从以下几个方面去理解JSON:

  • 首先是对“JSON是一种轻量的数据交换格式”的理解;
  • 然后来看经常被混为一谈的JSON和JS对象的区别;
  • 最后我们再来看JS中这几个JSON相关函数具体的执行细节。

希望全文能让如之前的我一样对JSON一知半解的亲能说清楚JSON是什么,也能熟练运用JSON,不看控制台就知道JS对象序列化成JSON字符串后输出是啥。

setTimeout和单线程

下面是我自己的一些理解
首先需要注意javascript是单线程的,特点就是容易出现阻塞。如果一段程序处理时间很长,很容易导致整个页面hold住。什么交互都处理不了怎么办?

简化复杂度?复杂逻辑后端处理?html5的多线程?

上面都是ok的做法,但是setTimeout也是处理这种问题的一把好手。

setTimeout一个很关键的用法就是分片,如果一段程序过大,我们可以拆分成若干细小的块。
例如上面的情况,我们将那一段复杂的逻辑拆分处理,分片塞入队列。这样即使在复杂程序没有处理完时,我们操作页面,也是能得到即使响应的。其实就是将交互插入到了复杂程序中执行。

换一种思路,上面就是利用setTimeout实现一种伪多线程的概念。

有个函数库Concurrent.Thread.js 就是实现js的多线程的。

一个简单使用的例子,引入Concurrent.Thread.js

Concurrent.Thread.create(function(){ for (var i = 0;i<1000000;i++) {
console.log(i); }; }); $(‘#test’).click(function () { alert(1); });

1
2
3
4
5
6
7
8
Concurrent.Thread.create(function(){
    for (var i = 0;i<1000000;i++) {
        console.log(i);
    };
});
$(‘#test’).click(function  () {
    alert(1);
});

虽然有个巨大的循环,但是这时不妨碍你去触发alert();

是不是很厉害~

还有一种场景,当我们需要渲染一个很复杂的DOM时,例如table组件,复杂的构图等等,假如整个过程需要3s,我们是等待完全处理完成在呈现,还是使用一个setTimeout分片,将内容一片一片的断续呈现。

其实setTimeout给了我们很多优化交互的空间。

一、JSON是一种格式,基于文本,优于轻量,用于交换数据

如果没有去过JSON的官方介绍可以去一下这里,官方介绍第一、二段已经很清楚地表述了JSON是什么,我将JSON是什么提炼成以下几个方面:

如何使用

setTimeout这么厉害,那么我们是需要在在项目中大量使用吗?
我这边的观点是非常不建议,在我们业务中,基本上是禁止在业务逻辑中使用setTimeout的,因为我所看到的很多使用方式都是一些问题不好解决,setTimeout作为一个hack的方式。
例如,当一个实例还没有初始化的前,我们就使用这个实例,错误的解决办法是使用实例时加个setTimeout,确保实例先初始化。
为什么错误?这里其实就是使用hack的手段
第一是埋下了坑,打乱模块的生命周期
第二是出现问题时,setTimeout其实是很难调试的。

我认为正确的使用方式是,看看生命周期(可参考《关于软件的生命周期
》),把实例化提到使用前执行。

综上,setTimeout其实想用好还是很困难的,
他更多的出现是在框架和类库中,例如一些实现Promis的框架,就用上了setTimeout去实现异步。
所以假如你想去阅读一些源码,想去造一些轮子,setTimeout还是必不可少的工具。

 

打赏支持我写出更多好文章,谢谢!

打赏作者

1. 一种数据格式

什么是格式?就是规范你的数据要怎么表示,举个栗子,有个人叫“二百六”,身高“160cm”,体重“60kg”,现在你要将这个人的这些信息传给别人或者别的什么东西,你有很多种选择:

  • 姓名“二百六”,身高“160cm”,体重“60kg”
  • name="二百六"&height="160cm"&weight="60kg"
  • 二百六16060
  • {"name":"二百六","height":160,"weight":60}
  • … …

以上所有选择,传递的数据是一样的,但是你可以看到形式是可以各式各样的,这就是各种不同格式化后的数据,JSON是其中一种表示方式。

打赏支持我写出更多好文章,谢谢!

任选一种支付方式

图片 2
图片 3

1 赞 7 收藏 1
评论

2. 基于文本的数据格式

JSON是基于文本的数据格式,相对于基于二进制的数据,所以JSON在传递的时候是传递符合JSON这种格式(至于JSON的格式是什么我们第二部分再说)的字符串,我们常会称为“JSON字符串”。

关于作者:唐光尧

图片 4

百度凤巢FE
个人主页 ·
我的文章 ·
2 ·
    

图片 1

3. 轻量级的数据格式

在JSON之前,有一个数据格式叫xml,现在还是广泛在用,但是JSON更加轻量,如xml需要用到很多标签,像上面的例子中,你可以明显看到xml格式的数据中标签本身占据了很多空间,而JSON比较轻量,即相同数据,以JSON的格式占据的带宽更小,这在有大量数据请求和传递的情况下是有明显优势的。

4. 被广泛地用于数据交换

轻量已经是一个用于数据交换的优势了,但更重要的JSON是易于阅读、编写和机器解析的,即这个JSON对人和机器都是友好的,而且又轻,独立于语言(因为是基于文本的),所以JSON被广泛用于数据交换。

以前端JS进行ajax的POST请求为例,后端PHP处理请求为例:

  1. 前端构造一个JS对象,用于包装要传递的数据,然后将JS对象转化为JSON字符串,再发送请求到后端;
  2. 后端PHP接收到这个JSON字符串,将JSON字符串转化为PHP对象,然后处理请求。

可以看到,相同的数据在这里有3种不同的表现形式,分别是前端的JS对象、传输的JSON字符串、后端的PHP对象,JS对象和PHP对象明显不是一个东西,但是由于大家用的都是JSON来传递数据,大家都能理解这种数据格式,都能把JSON这种数据格式很容易地转化为自己能理解的数据结构,这就方便啦,在其他各种语言环境中交换数据都是如此。

二、JSON和JS对象之间的“八卦”

很多时候都听到“JSON是JS的一个子集”这句话,而且这句话我曾经也一直这么认为,每个符合JSON格式的字符串你解析成js都是可以的,直到后来发现了一个奇奇怪怪的东西…

发表评论

电子邮件地址不会被公开。 必填项已用*标注