JavaScript 你不知道的事 -- 关于函数
接上篇Javascript 你不知道的事,直接条列了:
- 每个函数创建时默认带有一个prototype属性,其中包含一个constructor属性,和一个指向Object对象的隐藏属性__proto__。constructor属性的值为该函数的对象。在一个函数前面加上new来调用,则会创建一个隐藏连接到该函数prototype成员的新对象(由__proto__属性来链接),同时函数的this将会被绑定到那个新对象上。
 - 函数总是返回一个值;如果没有指定返回值,就返回undefined;如果当做构造函数来调用,且返回值不是对象,则返回this(该新对象);如果返回值是对象,则它作为构造函数是没有意义的!
- function A(){
 - this.p = 'haha';
 - return {p:'heihei'};
 - }
 - var a = new A();
 
alert(a.p);//显示'heihei',与var a = A();的效果一样。
 - 函数A内部直接调用一个函数B,B的this绑定到全局对象而不是其外部函数A,这是JS设计的一个错误。我们不得不用别的方式来解决这个问题,比如在A中用一个变量(通常是that)来保存A的this作用域的引用。
 - JS函数拥有一个length属性,表示函数定义时指定的形参的个数。
 - 函数的arguments属性包含了调用函数时传入的所有参数,而不管函数的声明中是否定义了这些形参;arguments不是数组,只是一个“类似数组”的对象(在函数中运行arguments instanceof Array;返回false)。可以通过Array.prototype.slice.apply(arguments)将其转化为JS数组。
 - 给JavaScript函数的原型增加方法(method),则所有的(构造)函数都可以用了!例如,可以给JS函数的构造者 Function 的原型增加一个method方法,则包括Object、Number等构造函数在内的所有函数都继承了该方法,这是很强大的:
- Function.prototype.method = function(name, func){
 - this.prototype[name] = func;
 - return this;
 - };
 
这样,调用Object.method方法,就可以为所有的JS对象(包括Function对象)增加新的方法,调用Number.method方法,可以为所有的数值类型增加新的方法,下面一条就是这样的一个例子。 注意Object、Number等类型的对象此时并没有继承method方法。如果想达到这样的目的,可以运行类似下面的语句:
- Object.method('method',Object.method);
 
 我们可以通过修改数值类型的原型,来给数值类型增加新的方法,这里我们借用上一条中提到的method方法来给Number的原型增加一个negative方法:
- Number.method(negative,function(){
 - return 0–this;
 - })
 
调用方法的时候稍微有一点绕。在JavaScript的语法中,数字后面直接跟点号,然后跟方法调用的语法是错误的;也就是说,3.negative()这样写是不对的。要想调用数值类型的方法,需要在数字后面加n个空格(n>=1),或者使用小括号将数字括起来,将其强制转化为表达式,然后再调用方法,或者干脆定义一个数值变量,也可以直接调用方法。也就是说,下面的写法都是正确的:
- (3).negative();
 - 3 .negative();
 - var n = 3; n.negative();
 - 3['negative']();
 
- 当使用函数表达式方法定义函数时,function后面的函数名可以用来递归地调用自己,并且这个名字不会被覆盖!我们来看下面的例子,
- function a(n){
 - if(n>1)
 - return a(n-1)+1;
 - else
 - return 1;
 - };
 
上述代码定义了一个函数a,并且其内部递归对自身进行了调用;现在我们用一个新的引用aa指向函数a,然后将原来的a改变,比如变为一个整数1,然后调用函数aa,如下面代码所示:- var aa = a;
 - a = 1;
 - aa(3);
 
则控制台报错:TypeError: Property 'a' of object [object Window] is not a function;很显然,原来的递归函数已经被破坏了。关于这个问题,我们可以在函数a的内部,用arguments.callee.caller来代替a,或者用一个命名函数表达式(Named Function Expressions)来定义函数:- var b = function a(n){
 - if(n>1)
 - return a(n-1)+1;
 - else
 - return 1;
 - };
 - var bb = b;
 - a = 3;
 - bb(3);
 
此时,bb函数能正确返回我们想要的结果。
 命名函数表达式不是真正的函数声明,该函数名只在该函数的作用域内有用。 下面这段代码充分说明了它的意义:
- var b = function a(){
 - return typeof a;
 - };
 - typeof a; // "undefined"
 - b(); // "function"
 
- 为了提高JavaScript函数的封装性,我们可以定义函数化的构造器,下面是一个例子:
- var funcCons = function(spec){
 - var that = {};
 - that.getName = function(){
 - return spec.name;
 - };
 - that.says = function(){
 - return spec.saying || '';
 - };
 - return that;
 - };
 - var myFunc = funcCons({name:'NearEast'});
 
这样,我们可以在构造器中定义一些私有变量(如字典表)和函数,而不必把它们全部暴露在外面。
 
JavaScript 你不知道的事 -- 关于函数的更多相关文章
- Javascript 你不知道的事,好吧,是我不知道的事
		
NaN表示一个不能产生正常结果的运算结果.它不等于任何值,包括它自己.可以用isNaN(number)来检测. 同Java中的字符串一样,JS中的字符串是不可变的.也就是说一旦字符串被创建,就无法改变 ...
 - Javascript 你不知道的事
		
NaN表示一个不能产生正常结果的运算结果.它不等于任何值,包括它自己.可以用isNaN(number)来检测. 同Java中的字符串一样,JS中的字符串是不可变的.也就是说一旦字符串被创建,就无法改变 ...
 - javascript线程解释(setTimeout,setInterval你不知道的事)---转载
		
在工作中,可能我们经常遇到在有很多 setInterval 的页面, 再手动触发 setTimeout 的时候经常失败, 尤其是 jquery做动画的时候,一些渐入溅出的东西,很多东西都不被触发……, ...
 - javascript线程解释(setTimeout,setInterval你不知道的事)
		
john resig写的一篇文章: 原文地址:http://ejohn.org/blog/how-javascript-timers-work/ 作为入门者来说,了解JavaScript中timer的 ...
 - setTimeout,setInterval你不知道的事
		
javascript线程解释(setTimeout,setInterval你不知道的事) 标签: javascript引擎任务浏览器functionxmlhttprequest 2011-11-21 ...
 - JavaScript——callback(回调函数
		
1.回调函数定义 回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数.回调函数不是由该函数的实现方直 ...
 - javascript你不知道的This
		
<你不知道的javascript>这本书读了有好几遍了,似乎每一次读都有新发现,有些内容并不是一下子可以弄懂的,每次读似乎都能明白一些概念.再重读一下this关键字.这个概念非常灵活,也非 ...
 - Javascript自执行匿名函数(function() { })()的原理分析
		
匿名函数指没有指定函数名或指针的函数,自执行匿名函数只是其中一种,下文中称这种函数为:自执行函数 下面是一个最常见的自执行函数: // 传统匿名函数 (function() { alert('hell ...
 - 借助JavaScript中的时间函数改变Html中Table边框的颜色
		
借助JavaScript中的时间函数改变Html中Table边框的颜色 <html> <head> <meta http-equiv="Content-Type ...
 
随机推荐
- java总结(方法与对象)
			
包 1.用于管理类 2.包名采用公司的域名倒序+项目名+模块名 3.包的引入 类 1.main方法用static 它调用的方法也要static 2.程序要运行必须要main方法 3.类是一种引用数据类 ...
 - Spring学习——什么是AOP
			
在网上查找什么是AOP,查到了下面这段话,我感觉写得清晰明了,一下子就明白了什么是面向切面编程AOP. ———————————————————————————————————————————————— ...
 - 2017/11/5 Leetcode 日记
			
2017/11/5 Leetcode 日记 476. Number Complement Given a positive integer, output its complement number. ...
 - 【BZOJ 1062】 1062: [NOI2008]糖果雨 (二维树状数组)**
			
1062: [NOI2008]糖果雨 Description 有一个美丽的童话:在天空的尽头有一个" 糖果国" ,这里大到摩天大厦,小到小花小草都是用糖果建造而成的.更加神奇的是, ...
 - 主席树+dfs SPOJ BZOJ2588 Count on a tree
			
这道题我由于智障错误导致一直错. 在树上建主席树,加上lca思想,很简单. #include<bits/stdc++.h> using namespace std; ; struct no ...
 - hihoCoder #1695 公平分队II
			
题目大意 Alice 和 Bob 在玩一个游戏.Alice 将 $1$ 到 $2n$ 这 $2n$ 个整数分成两组,每组 $n$ 个.Bob 从中选一组,剩下一组归 Alice.Alice 可以与 B ...
 - [BZOJ1032][P1840] 祖玛 记忆化搜索 动态规划
			
描述 Description 某天,小x在玩一个经典小游戏——zumo.zumo游戏的规则是,给你一段长度为n的连续的彩色珠子,珠子的颜色不一定完全相同,但是,如果连续相同颜色的珠子大 ...
 - JBoss 7 配置成windows启动服务
			
将Jboss7 server 配置成一个windows启动服务的两个文件,部署步骤如下: 1. 先检查是否配置java_home和jboss_home的环境变量,如没配置上先配置,如我的是JBO ...
 - 内功心法 -- java.util.LinkedList<E> (2)
			
写在前面的话:读书破万卷,编码如有神--------------------------------------------------------------------下文主要对java.util ...
 - keras实现mnist数据集手写数字识别
			
一. Tensorflow环境的安装 这里我们只讲CPU版本,使用 Anaconda 进行安装 a.首先我们要安装 Anaconda 链接:https://pan.baidu.com/s/1AxdGi ...