JavaScript函數

来源:http://www.cnblogs.com/haoxi/archive/2017/06/24/7074066.html
-Advertisement-
Play Games

函數的調用 直接調用 foo(); 對象方法 o.method(); 構造器 new Function(); call/apply/bind func.call(o); 函數聲明和表達式 函數聲明會被前置,函數表達式變數聲明會被前置,但是值為 。 函數聲明 函數表達式 將函數賦值給一個變數 匿名函數 ...


函數的調用

  • 直接調用 foo();
  • 對象方法 o.method();
  • 構造器 new Function();
  • call/apply/bind func.call(o);

函數聲明和表達式

函數聲明會被前置,函數表達式變數聲明會被前置,但是值為undefined

函數聲明

function func(a, b) {
    //do sth
}

函數表達式

  • 將函數賦值給一個變數

    var func = function(a, b) {
    //do sth
    };
  • 匿名函數(IEF 立即執行函數表達式)

    (function() {
    //do sth
    })();
  • 返回函數對象

    return function() {
    //do sth
    };
  • 命名函數表達式(NEF)

    var add = function foo(a, b) {
    //do sth
    };

命名函數表達式存在一些經典的bug,例如在執行如下代碼時:

var func = function nfe() {};
alert(func == nfe);

IE6會提示為false,IE9+中nfe外部並不可見,提示為nfe is undefined

命名函數表達式主要可以應用在調試和遞歸調用時。

var func = funtion nfe(){/* do sth */ nfe();};

但也可直接通過func變數名來執行遞歸調用,因此命名函數表達式並不常用。

function構造器

var func = new Function('a', 'b', 'console.log(a + b);');
func(1, 2); //3

var func = Function('a', 'b', 'console.log(a + b);');
func(1, 2)  //3

function構造器的作用域

Function('var localVal = "local"; console.log(localVal);')();
console.log(typeof localVal);
// result: local, undefined
// localVal仍未局部變數

var globalVal = 'global';
(function() {
    var localVal = 'local';
    Function('console.log(typeof localVal, typeof globalVal);')();
})();
//result: undefined, string
//local不可訪問,全局變數global可以訪問

各方式對比

函數聲明 函數表達式 函數構造器
前置
允許匿名
可立即調用
在定義該函數的作用域通過函數名訪問
沒有函數名

this

全局的this

全局的this一般即是瀏覽器

console.log(this.document === document); //true
consloe.loh(this == window);    //true

this.a = 37
console.log(window.a);  //37

一般函數的this

一般函數的this仍然指向全局對象,瀏覽器中即為window

fucntion f1(){
    return this;
}

f1() === window;    //true, global, object

嚴格模式下,this指向undefined

fucntion f2(){
    "use strict";
    return this;
}

f2() === undefined; //true

作為對象方法的函數this

對象方法中的函數this會指向具體的對象

var o = {
    prop: 37;
    f: function() {
        return this.prop;
    }
};

console.log(o.f()); //logs 37

也可以通過外部定義函數

var o = {prop: 37};
function independent() {
    return this.prop;
}

o.f = independent
console.log(o.f()); //logs 37

通過callapply調用指定this

function add(c, d) {
    return this.a + this.b + c + d;
}

var o = {a:1, b:3};

add.call(o, 5, 7);  // 1 + 3 + 5 + 7 = 16 
add.apply(o, [10, 20]); //1 + 3 + 10 + 20 = 34


function bar() {
    console.log(Object.prototype.toString.call(this));
}

bar.call(7);    //"[object Number]"

一般模式和嚴格模式下使用apply的區別

function foo(x, y) {
    console.log(x, y, this);
}

foo.apply(null);    // undefined, undefined, window
foo.apply(undefined);    // undefined, undefined, window

// 嚴格模式下
foo.apply(null);    // undefined, undefined, null
foo.apply(undefined);    // undefined, undefined, undefined

'bind'方法與this

通過ES5提供的bind方法,可以將函數的this綁定到一個對象上,bind之後this不可變。

function f(){
    return this.a
}

var g = f.bind({a: "test"}):
console.log(g());   //test

var o = {a: 37, f:f, g:g};
// g()中的this不會再改變
console.log(o.f(), o.g());  //37, test

函數屬性和對象

function foo(x, y, z) {
    
    arguments.length;   //2
    arguments[0];   //1
    arguments[0] = 10;
    x;  // change to 10; 嚴格模式下仍然是1
    
    arguments[2] = 100;
    z;  // still undefined !!!
    arguments.callee === foo;   // true 嚴格模式下不能使用
}

foo(1, 2)
foo.length;     // 3
foo.name;       // "foo"

使用bind()方法currying函數

function add(a, b, c) {
    return a + b + c;
}

var func = add.bind(undefined, 100);
func(1, 2); // 100綁定到a上,result:103

var func2 = func.bind(undefoned, 200);
func2(10);  // 200綁定到b上,result:310

bind和new的使用

function foo() {
    this.b =100;
    return this.a;
}

var func = foo.bind({a:1});

func(); // 1
// 使用new時,除非指定返回一個對象,否則會返回this,
// 同時this會被初始化為一個空對象的prototype
new func(); //{b: 100}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 很多項目都配置了日誌記錄的功能,但是,卻只有很少的項目組會經常去看日誌。原因就是日誌文件生成規則設置不合理,將嚴重的錯誤日誌跟普通的錯誤日誌混在一起,分析起來很麻煩。 其實,我們想要的一個日誌系統核心就這2個要求: 這樣的日誌系統最大的好處就是可以幫助我們一目瞭然的發現嚴重錯誤。結合管理員後臺直接訪 ...
  • 在開發 H5 應用的時候碰到一個問題,應用只需要一張小的縮略圖,而用戶用手機上傳的確是一張大圖,手機攝像機拍的圖片好幾 M,這可要浪費很多流量。 像我這麼為用戶著想的程式員,絕對不會讓這種事情發生的,於是就有了本文。 獲取圖片 通過 File API 獲取圖片。 預覽圖片 使用 createObje ...
  • 英文:Aurélien Hervé 譯文:眾成翻譯/msmailcode 這裡有一些 Javascript初學者應該知道的技巧和陷阱。如果你已經是專家了,順便溫習一下。 Javascript也只不過是一種編程語言。怎麼可能出錯嘛? 1. 你有沒有嘗試給一組數字排序? Javascript 的sort ...
  • 英文:Ben Northrop 譯文:開源中國 【導讀】:Ben Northrop 在 2016 年滿了 40 歲,本文是他對職業生涯的思考。他認為從長遠來看,應該多投資一些不容易過期、衰竭期較長的知識領域中。 我是一名程式員,幾個月前剛過完 40 歲生日。某個星期六的早晨,我參加了一個 React ...
  • 1,原型:$watch: function(watchExp, listener, objectEquality, prettyPrintExpression){}; 2,參數:watchExp(必須):{(function()|string)},可以字元串表達式,也可以帶當前scope為參數的函數 ...
  • 前幾天偶然看到了一個這樣的題: 這個題代碼不多,不過考察的知識點卻非常不錯。我們知道記憶體空間分為棧記憶體和堆記憶體。棧記憶體用來存放供js代碼來執行的環境,所以為了保證性能減少記憶體占用,我們一般把占用空間較小的類似於基本數據類型放在棧記憶體中,像引用數據類型這種占據空間比較大的東西,我們需要將它放在一個貯藏 ...
  • ES6 簡介 ECMAScript 6 簡稱 ES6,是 JavaScript 語言的下一代標準,已經在2015年6月正式發佈了。它的目標是使得 JavaScript 語言可以用來編寫複雜的大型應用程式,成為企業級開發語言。 ECMAScript 和 JavaScript 的關係:前者是後者的語法規 ...
  • 閉包的概念 維基百科中是這麼解釋閉包的: 電腦科學中,閉包(也稱為詞法閉包或函數閉包)是指一個函數或函數的引用,與一個引用環境綁定在一起。這個函數環境是一個存儲該函數每個非局部變數(也叫自由變數)的表。 閉包,不同意一般函數,它允許一個函數在立即詞法作用域外調用時,認可訪問非本地變數。 閉包的例子 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...