var func = function(){
var a = 1;
alert ( a ); // 輸出: 1
};
func();
alert ( a ); // 輸出:Uncaught ReferenceError: a is not defined
複製代碼
var func = function(){
var a = 1; // 退出函數後局部變量 a 將被銷燬
alert ( a );
};
func();
複製代碼
var func = function(){
var a = 1;
return function(){
a++;
alert ( a );
}
};
var f = func();
f(); // 輸出:2
f(); // 輸出:3
f(); // 輸出:4
f(); // 輸出:5
複製代碼
var mult = (function(){
var cache = {};
var calculate = function(){ // 封閉 calculate 函數
var a = 1;
for(var i = 0, l = arguments.length; i < l; i++ ){
a = a * arguments[i];
};
return a;
}
return function(){
var args = Array.prototype.join.call( arguments, ',' );
if ( args in cache ){
return cache[ args ];
}
return cache[ args ] = calculate.apply( null, arguments );
}
})();
alert ( mult( 1,2,3 ) ); // 輸出:6
alert ( mult( 1,2,3 ) ); // 輸出:6
複製代碼
<html>
<body>
<button id="execute">點擊我執行命令</button>
<button id="undo">點擊我執行命令</button>
<script>
var Tv = {
open: function(){
console.log( '打開電視機' );
},
close: function(){
console.log( '關上電視機' );
}
};
var createCommand = function( receiver ){
var execute = function(){
return receiver.open();// 執行命令,打開電視機
}
var undo = function(){
return receiver.close();// 執行命令,關閉電視機
}
return {
execute: execute,
undo: undo
}
};
var setCommand = function( command ){
document.getElementById( 'execute' ).onclick = function(){
command.execute(); // 輸出:打開電視機
}
document.getElementById( 'undo' ).onclick = function(){
command.undo(); // 輸出:關閉電視機
}
};
setCommand(createCommand(Tv));
</script>
</body>
</html>
複製代碼
若是兩個對象之間造成了循環引用,那麼這兩個對象都沒法被回收,但循環引用形成的內存泄露在本質上也不是閉包形成的html
高階函數是指至少知足下列條件之一的函數node
- 函數能夠做爲參數被傳遞
- 函數能夠做爲返回值輸出
var getUserInfo = function( userId, callback ){
$.ajax( 'http://xxx.com/getUserInfo?' + userId, function( data ){
if ( typeof callback === 'function' ){
callback( data );
}
});
}
getUserInfo( 13157, function( data ){
alert ( data.userName );
});
複製代碼
var appendDiv = function( callback ){
for ( var i = 0; i < 100; i++ ){
var div = document.createElement( 'div' ); div.innerHTML = i;
document.body.appendChild( div );
if ( typeof callback === 'function' ){
callback( div );
}
};
};
appendDiv(function( node ){
node.style.display = 'none';
});
複製代碼
var Type = {};
for ( var i = 0, type; type = [ 'String', 'Array', 'Number' ][ i++ ]; ){
(function( type ){
Type[ 'is' + type ] = function( obj ){
return Object.prototype.toString.call( obj ) === '[object '+ type +']';
}
})(type)
};
Type.isArray( [] ); // 輸出:true
Type.isString( "str" ); // 輸出:true
複製代碼
var getSingle = function ( fn ) {
var ret;
return function () {
return ret || ( ret = fn.apply( this, arguments ) );
};
};
var getScript = getSingle(function(){
return document.createElement( 'script' );
});
var script1 = getScript();
var script2 = getScript();
alert ( script1 === script2 ); // 輸出:true
複製代碼
AOP(面向切面編程)的主要做用是把一些跟核心業務邏輯模塊無關的功能抽離出來,這些 跟業務邏輯無關的功能一般包括日誌統計、安全控制、異常處理等。把這些功能抽離出來以後, 再經過「動態織入」的方式摻入業務邏輯模塊中。ajax
Function.prototype.before = function( beforefn ){
var __self = this; // 保存原函數的引用
return function(){ // 返回包含了原函數和新函數的"代理"函數
beforefn.apply( this, arguments );
return __self.apply( this, arguments );
}
};
Function.prototype.after = function( afterfn ){
var __self = this;
return function(){
// 執行新函數,修正 this // 執行原函數
var ret = __self.apply( this, arguments );
afterfn.apply( this, arguments );
return ret;
}
};
var func = function(){
console.log( 2 );
};
func = func.before(function(){
console.log( 1 );
}).after(function(){
console.log( 3 );
});
func(); // 1 2 3
複製代碼
var currying = function( fn ){
var args = [];
return function(){
if ( arguments.length === 0 ){
return fn.apply( this, args );
}else{
[].push.apply( args, arguments );
return arguments.callee;
}
}
};
var cost = (function(){
var money = 0;
return function(){
for ( var i = 0, l = arguments.length; i < l; i++ ){
money += arguments[ i ];
}
return money;
}
})();
var cost = currying( cost ); // 轉化成 currying 函數
cost( 100 ); // 未真正求值
cost( 200 ); // 未真正求值
cost( 300 );// 未真正求值
alert ( cost() ); // 求值並輸出:600
複製代碼
Function.prototype.uncurrying = function () {
var self = this; // self 此時是 Array.prototype.push
return function() {
var obj = Array.prototype.shift.call( arguments );// 至關於 Array.prototype.push.apply( obj, 2 ) };
};
};
var push = Array.prototype.push.uncurrying();
var obj = {
"length": 1,
"0": 1
};
push( obj, 2 );
console.log( obj );// 輸出:{0: 1, 1: 2, length: 2}
複製代碼
var throttle = function ( fn, interval ) {
var __self = fn, // 保存須要被延遲執行的函數引用 timer, // 定時器
firstTime = true; // 是不是第一次調用
return function () {
var args = arguments,
__me = this;
if ( firstTime ) { // 若是是第一次調用,不需延遲執行
__self.apply(__me, args);
return firstTime = false;
}
if ( timer ) { // 若是定時器還在,說明前一次延遲執行尚未完成
return false;
}
timer = setTimeout(function () { // 延遲一段時間執行
clearTimeout(timer);
timer = null;
__self.apply(__me, args);
}, interval || 500 );
};
};
window.onresize = throttle(function(){
console.log( 1 );
}, 500 );
複製代碼