1. 代码求值机制

  JavaScript中,有很多不同的代码求值机制。

  ● eval()函数

  ● 函数构造器

  ● 定时器

  ● <script>元素

- 用eval()方法进行求值

  作为定义在全局作用域内的eval()方法,该方法将在当前上下文内,执行所传入字符串形式的代码。

  基本功能

  ● 该方法将执行传入代码的字符串

  ● 在调用eval()方法的作用域内进行代码求值

例如:

eval("5+5")     //

(function(){
eval("var ninja = 6");
console.assert(ninja === 6, "evaluated within the current scope.");
})();

  求值结果

  eval()方法将返回传入字符串中最后一个表达式的执行结果。例如,如果我们调用如下语句:

  eval("3+4; 5+6");    //  结果将返回11

  应该指出的是,任何不适简单变量、原始值、赋值语句的内容都需要在外面包装一个括号以便返回正确的结果。例如,如果我们想使用eval()创建一个简单的对象,可能会编写如下语句:

  var o = eval('({ninja: 1})');

  但是,结果不是我们所期望的。我们需要在对象字面量外面包装一个括号,示例如下:

  var o = eval('({ninja: 1})');

  在IE8或之前的版本,要使用布尔表达式,才能让eval()进行正确的调用,如下代码:

var fn = eval("false||function(){return true;}");

  就像我们用普通方式在特定作用域内创建函数一样,eval()创建的函数会继承该作用域的闭包——局部作用域内执行eval()时的衍生结果。

- 用函数构造器进行求值

  使用Function构造器来实例化函数,示例如下:

var add = new Function("a", "b", "return a+b;");

  Function构造器可变参数列表的最后一个参数,始终是要创建函数的函数体内容。前面的参数则表示函数的形参名称。所以,上述示例代码等价于如下代码:

var add = function(a,b){return a + b}

  虽然这些代码在功能上是等同的,但采用Function构造器方式有一个明显的区别,函数体尤运行时的字符串所提供。另外一个极其重要的实现区别是,使用Function构造器创建函数的时候,不会创建闭包。在不想承担任何不相关闭包开销时,这可能是一件好事。

- 用定时器进行求值

  还有一种方式,可以让代码字符串进行求值,而且是异步的,那就是通过定时器进行求值。示例如下:

var tick = window.setTimeout("alert("Hi!"), 100")

- 全局作用域内的求值操作

function globalEval(data){
data = data.replace(/^\s|\s*$/g, "");
if(data){
var head = document.getElementsByTagName("head")[0]||document.documentElement, script = document.createElement("script");
script.type = "text/javascript";
script.text = data;
head.appendChild(script);
head.removeChild(script);
}
} window.onload = function(){
(function(){
globalEval("var test = 5;");
})();
}

  在eval()方法中,我们定义了一个名为globalEval()的函数,以便在全局作用域内求值任何想要要求的内容。该函数去除了所传字符串中的所有前导和尾部空白字符,然后定位DOM中的<head>元素或文档本身,并创建一个分离的<script>元素。设置script元素的类型,然后把需要求值的字符串加载到该script元素的body内。将script元素附加到DOM上,作为head元素的一个子节点,将会导致该脚本在全局作用域内进行求值。

  这段代码最常见的场景是动态执行从服务器端返回的代码。它几乎总是要求代码在全局作用域内进行执行,从而让新函数的使用变成必然。

2.函数反编译

反编译意味着将程序集或字节码重组成源代码。例如,将函数反编译成字符串:

function test(a){return a+a; }
console.assert(test.toString() === "function test(a){return a+a; }","Function decompiled");

3.代码求值实战

- JSON转化

  运行时求值的最广泛使用方式是将JSON字符串转换为JavaScript对象表示法。例:

var json = '{"name":"Ninja"}';
var object = eval("("+ json +")");
assert(object.name === "Ninja", "My name is Ninja!");

- 导入有命名空间的代码

  对于将命名空间导入到当前上下文,base2库提供了一个非常有趣的解决方案。因为没有办法将该问题进行自动化操作,因此我们可以利用运行时求值让该实现变得更简单。

  每当一个新类或模块添加到base2包的时候,构造可执行代码的字符串,对其进行求值,可以将产生的函数引入到当前上下文中,示例如下:

base2.namespace ==
"var Base = base2.Base; var Package = base2.Package;" +
"var Abstract = base2.Abstract; var Module = base2.Module;" +
"var Enumberable = base2.Enumberable; var Map = base2.Map;" +
"var Collection = base2.Collection; var RegGrp = base2.RegGrp;" +
"var Undefined = base2.Undefined; var Null = base2.Null;" +
"var This = base2.This; var True = base2.True; var False = base2.False;" +
"var assignID = base2.assignID; var detect = base2.detect;" +
"var global = base2.global; var lang = base2.lang;"+
"var JavaScript base2.JavaScript; var JST = base2.JST;" +
"var JSON = base2.JSON; var IO = base2.IO; var MiniWeb = base2.MiniWeb;" +
"var DOM = base2.DOM; var JSB = base2.JSB; var code = base2.code;" +
"var doc = base2.doc;"

JavaScript忍者秘籍——运行时代码求值的更多相关文章

  1. 《Secrets of the JavaScript Ninja》:JavaScript 之运行时代码

    最近,在阅读 jQuery 之父 John Resig 力作:Secrets of the JavaScript Ninja(JavaScript忍者秘籍).关于第九章提及的 JavaScript 之 ...

  2. JavaScript 对引擎、运行时、调用堆栈的概述理解

    JavaScript 对引擎.运行时.调用堆栈的概述理解  随着JavaScript越来越流行,越来越多的团队广泛的把JavaScript应用到前端.后台.hybrid 应用.嵌入式等等领域. 这篇文 ...

  3. 如何将oc代码转换成运行时代码

    // 运行时 其实就是oc的底层  平时写的代码 最终都是转成底层的运行时代码以下面程序为例子: 如果我们想要看我们的main.m文件底层转换成了怎样的运行时代码 ,我们可以这样做. 1.打开终端  ...

  4. webpack学习:uni运行时代码解读一 (页面初始化加载)

    uni的vue代码是如何在微信小程序里面执行的,对此比较感兴趣所以去调试学习了一波. 准备工作 // 在vue.config.js里打开非压缩的代码 module.exports = { config ...

  5. 解读JavaScript 之引擎、运行时和堆栈调用

    转载自开源中国 译者:Tocy, 凉凉_, 亚林瓜子, 离诌 原文链接 英文原文:How JavaScript works: an overview of the engine, the runtim ...

  6. 解读 JavaScript 之引擎、运行时和堆栈调用

    https://www.oschina.net/translate/how-does-javascript-actually-work-part-1 随着 JavaScript 变得越来越流行,很多团 ...

  7. JavaScript忍者秘籍——驯服线程和定时器

    1.定时器和线程 - 设置和清除定时器 JavaScript提供了两种方式,用于创建定时器以及两个相应的清除方法.这些方法都是window对象上的方法. 方法 格式 描述 setTimeout   i ...

  8. JavaScript忍者秘籍——原型

    概要:本篇博客主要介绍JavaScript的原型 1.对象实例化 - 初始化的优先级 初始化操作的优先级如下: ● 通过原型给对象实例添加的属性 ● 在构造器函数内给对象实例添加的属性 在构造器内的绑 ...

  9. JavaScript忍者秘籍——函数(下)

    概要:本篇博客主要介绍函数的一些类型以及常见示例 1.匿名函数 使用匿名函数的常见示例: window.onload = function(){ assert(true,'power!'); }; / ...

随机推荐

  1. Linux网络编程(六)

    网络编程中,使用多路IO复用的典型场合: 1.当客户处理多个描述字时(交互式输入以及网络接口),必须使用IO复用. 2.一个客户同时处理多个套接口. 3.一个tcp服务程序既要处理监听套接口,又要处理 ...

  2. hive left outer join的问题

    最近BA用户反馈有两句看似很像的语句返回的结果数不一样,比较奇怪,怀疑是不是Hive的Bug Query 1 返回结果数6071 select count(distinct reviewid) as ...

  3. 查看TOMCAT内存使用情况 以及修改方法

    查看TOMCAT内存使用情况 <% double total = (Runtime.getRuntime().totalMemory()) / (1024.0 * 1024); double m ...

  4. python 中文字数统计/分词

    因为想把一段文字分词,所以,需要明确一定的词语关系. 在网上随便下载了一篇中文小说.随便的txt小说,就1mb多.要数数这1mb多的中文到底有多少字,多少分词,这些分词的词性是什么样的. 这里是思路 ...

  5. javaScript 消除错误,并将错误记录在控制台,阻止浏览器错误警告

    当我们使用jquery,和其他各种框架时,有的时候会出现各种错误,  例如jquery文件报错,但又不影响功能,  又不能对jquery做出更改,怎么办呢? window.onerror=functi ...

  6. hdu1205(类似 分布垃圾数列)

    Problem Description A Fibonacci sequence is calculated by adding the previous two members the sequen ...

  7. sharepoint 2013 文档库eventhandle权限控制

    记录一下如何在sharepoint server 2013文档库中,使用eventhandle控制文档库document library的条目item权限. ///<summary> // ...

  8. [置顶] Oracle学习经验谈

    经常遇到朋友问oracle学习难不难,怎么才能成为高手等等,我想结合我的个人经验简单说几点: 1.打好基础,由浅入深 学习Oracle不能急于求成,寄希望于一天成为一个大侠.学习有个过程,应该由浅入深 ...

  9. java调用存储过程超时及DBCP参数配置说明

    问题:            生产环境实时打标超时: 分析原因:        “实时打标java服务中,只创建数据库Connection,没有关闭数据库Connection,导致数据库连接池耗尽,无 ...

  10. Linux负载均衡软件LVS之三(配置篇)

    LVS集群有DR.TUN.NAT三种配置模式,可以对www服务.FTP服务.MAIL服务等做负载均衡,下面通过搭建www服务的负载均衡实例,讲述基于DR模式的LVS集群配置. 一. Director ...