javascript: 類型轉換

来源:https://www.cnblogs.com/guolao/archive/2019/08/16/11364779.html
-Advertisement-
Play Games

strat javascript 的類型轉換一直是個大坑,但其實它也減少了代碼量。 ToPrimitive Symbol.toPrimitive 是一個內置的 Symbol 值,它作為對象的函數值屬性存在,當一個對象轉換為原始值時,會調用此函數。 該函數被調用時,會被傳遞一個字元串參數 ,表示要轉換 ...


strat


javascript 的類型轉換一直是個大坑,但其實它也減少了代碼量。

ToPrimitive


Symbol.toPrimitive 是一個內置的 Symbol 值,它作為對象的函數值屬性存在,當一個對象轉換為原始值時,會調用此函數。

該函數被調用時,會被傳遞一個字元串參數 hint ,表示要轉換到的原始值的預期類型。 hint 參數的取值是 "number""string""default" 中的任意一個。

// 一個沒有提供 Symbol.toPrimitive 屬性的對象,參與運算時的輸出結果
let obj1 = {};
console.log(+obj1);     // NaN
console.log(`${obj1}`); // "[object Object]"
console.log(obj1 + ""); // "[object Object]"

// 接下麵聲明一個對象,手動賦予了 Symbol.toPrimitive 屬性,再來查看輸出結果
let obj2 = {
  [Symbol.toPrimitive](hint) {
    if (hint == "number") {
      return 10;
    }
    if (hint == "string") {
      return "hello";
    }
    return true;
  }
};
console.log(+obj2);     // 10      -- hint 參數值是 "number"
console.log(`${obj2}`); // "hello" -- hint 參數值是 "string"
console.log(obj2 + ""); // "true"  -- hint 參數值是 "default"

從上面可以看出,toPrimitive 轉換過程依靠 hint 參數:

  • number: valueOf() → toString() → TypeError
  • string: toString() → valueOf() → TypeError
  • default: 同 number

valueOf

對象 返回值
Array 返回數組對象本身。
Boolean 布爾值。
Date 存儲的時間是從 1970 年 1 月 1 日午夜開始計的毫秒數 UTC。
Function 函數本身。
Number 數字值。
Object 對象本身。這是預設情況。
String 字元串值。
Symbol Symbol本身
Math 和 Error 對象沒有 valueOf 方法。

toString

對象 返回值
Array [1, 2, 3] => "1,2,3"
Boolean false => "false"
Date 返回表示 UTC 的字元串。
Function 返回表示當前函數源代碼的字元串。
Number 返回表示當前數值的字元串。
Object "[object Object]"
String 字元串本身。
Symbol "Symbol()"

註意:[null].toString()以及[undefined].toString()均返回空字元串""

ToBoolean


ES5 規範 9.2中列舉了布爾強制類型轉換 (ToBoolean) 會出現假值 (false) 的僅有以下幾個,其餘都為真值 (true):

  • undefined
  • null
  • false
  • +0、-0、NaN
  • ''(空字元串)
/*
 以下 a、b、c 存儲的是指向對象的指針,並非假值
*/
let a = new Number(0);
let b = new Boolean(false);
let c = new String('');

Boolean(a) // true
Boolean(b) // true
Boolean(c) // true

Boolean(0) // false
Boolean(false) // false
Boolean('') // false

ToNumber


對象 返回值
Undefined NaN
Null 0
Boolean true => 1, false => 0
Number 返回自身
String 不能解析為 StringNumericLiteral 的,均返回 NaN
Object ToPrimitive(input argument, hint Number)

註: StringNumericLiteral

強制類型轉換符


加號 (+)

+作為一元運算符,單獨使用,會強制將右側操作數類型轉為 number,即對右側操作數使用 ToNumber()。

+1 // 1
+'1.2' // 1.2
+[] // 0
+[1, 2, 3] // NaN
+{} // NaN

嘆號 (!)

!會強制將右側操作數類型轉為 boolean,並反轉真、假值,即對右側操作數使用 !ToBoolean()。

!true // false
!0 // true
![] // false
!'' // true
!undefined // true
!null // true

!!true // true
!!undefined // false
!!null // false

四則運算符


加法運算遵循以下規則:

  1. 運算的其中一方為字元串,就會把另一方轉換為字元串。

    1 + '1' // '11'
    42 + '' // '42'
  2. 如果其中一方不是字元串或數字,就會將它轉換為字元串或數字。

    false + true // 1
    3 + [1, 2, 3] // '31,2,3'
    ([] + {}) // '[object Object]'
    
    /* {} + [] 的結果為 0, 是因為從左往右解析,{} 為一個代碼塊,+[] 被解析為將 [] 轉為 number, 即 0。*/
    {} + [] // 0
    ({} + []) // "[object Object]"

    註意:

    /* 會出現以下情況,是因為 + 'b' 解釋為 ToNumber('b') */
    'a' + + 'b' // "aNaN"

對於加法運算以外的運算來說,雙方會被轉為數字進行運算。

1 * '2' // 2
[] * {} // NaN
1 * [1, 2, 3] // NaN

let obj = {
    valueOf: () => {
        return 1
    }
}

obj * 2 // 2

== and ===


對於==(相對等於)、===(絕對等於),絕大部分的書籍和博客都解釋為前者僅檢查值是否相等,後者檢查值和類型是否相等,其實這樣是不對的,正確的解釋應該是:前者允許在比較的時候進行強制類型轉換,後者不允許

ES5 規範 11.9.3 定義了相對等於的行為,涵蓋了所有的類型,具體可分為以下幾種情況:

  1. 雙方類型相同

    類型 結果
    Undefined true
    Null true
    Number 1. 如果其中一方為NaN,返回false。2. 如果 x 與 y 的值相同,則返回true,否則false。3.如果其中一方為+0-0且另一方為+0-0,返回true
    String 雙方為完全相同的字元序列,返回true。否則返回 false
    Boolean 雙方為truefalse,返回true,否則返回false
    Object 雙方引用同一個對象,返回 true。否則,返回false
    NaN == NaN // false
    -0 == +0 // true
  2. null 與 undefined

    null == undefined // true
  3. 字元串與數字

    會將字元串轉為數字進行比較,即ToNumber(字元串) == 數字

    10 == '10' // true
    10 == 'a' // false
    /* 十六進位 '0xa' => 十進位 10 */
    10 == '0Xa' // true
  4. 布爾類型與其他類型

    會將布爾類型轉為數字,再與其他類型進行比較,即ToNumber(布爾類型) == 其他類型

    0 == false // true
    '1' == true // true
    null == false // false
    undefined == false // false
  5. 對象類型與非對象類型

    會將對象類型轉為原始類型,再進行比較,即 ToPrimitive(對象類型) == 非對象類型

    [1] == 1 // true
    [1, 2] == 1 // false
    
    /* b.toString() 返回 '111' */
    let a = '111';
    let b = Object(a);
    a == b // true
    
    /* null 與 undefined 不能被封裝為 object, 即 Object(null) 的返回結果與 Object() 的一樣 */
    let c = null;
    let d = Object(c);
    c == d // false
    
    let e = undefined;
    let f = Object(e);
    e == f // false

難以理解的情況


  1. [] == ![]

    [] == ![] // true
    
    /*
    第一步: !的優先順序比 == 高,所以 ![] 解析為 !Boolean([]),結果為 true.
    現在: [] == true
    
    第二布: 布爾類型與其他類型進行比較,解析為 ToNumber(true), 結果為 0.
    現在: [] == 0
    
    第三步: 對象類型與非對象類型進行比較,解析為 ToPrimitive([], 'number'),結果為 0.
    現在: 0 == 0 // true
    */
  2. [null] == ''

    [null] == '' // true
    [undefined] == '' // true
    
    /*
    [null].toString() 以及 [undefined].toString() 均返回空字元串 ''
    因為 null 與 undefined 均沒有 toString 和 valueOf 方法。
    */
  3. 0 == '\n'

    0 == '\n' // true
    0 == '\t\r\n' // true
    
    /*
    上述語句被解析為 ToNumber('\n'), 返回結果為 0.
    */

    具體解釋:'\n\t\r' == 0 is true?

備註


理解了類型轉換,你會發現並非一定要拋棄==去使用===


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

-Advertisement-
Play Games
更多相關文章
  • 簡單來說GPDB是一個分散式資料庫軟體,其可以管理和處理分佈在多個不同主機上的海量數據。對於GPDB來說,一個DB實例實際上是由多個獨立的PostgreSQL實例組成的,它們分佈在不同的物理主機上,協同工作,呈現給用戶的是一個DB的效果。Master是GPDB系統的訪問入口,其負責處理客戶端的連接及 ...
  • 前言 涉及到的源碼有 frameworks\base\services\core\java\com\android\server\policy\PhoneWindowManager.java vendor\mediatek\proprietary\packages\apps\SystemUI\src ...
  • 原文鏈接:https://blog.csdn.net/Jaden_hool/article/details/78437947 方法調用流程 調用一個方法是一個壓棧和出棧的過程,調用方法時將棧針壓入方法棧,然後執行方法體,方法結束時將棧針出棧,這個壓棧和出棧的過程會耗費資源,這個過程中傳遞形參也會耗費 ...
  • 多人開發時找回丟失的代碼 1.先保證所有分支的代碼都已經提交並拉取到最新狀態。 2.最重要的是需要找到最近一條自己代碼還存在的記錄,然後複製到最近的更早一條的提交記錄的id,比如e36e9e76da144536c196147b17335288da52c2ff;查找當前分支自己最新的一次提交記錄的id ...
  • CSS cascading(層疊) style(樣式) sheet(表) css註釋 引入方式 css選擇器 基本選擇器 元素選擇器 div{ } id選擇器 #demo 在html中(id="demo") 類選擇器 .demo 在html中(class="demo") 通用選擇器 *{ } 組合選 ...
  • 在之前,我寫了一個websql的封裝類庫,代碼如下: (function(win) { function smpWebSql(options){ options = options || {}; this.database = null; this.DateBaseName = options.Da ...
  • 08.16自我總結 django渲染高階 一.利用母版渲染 1.創建母版文件 如: 2.導入模板 二.一部分文件渲染 1.組件 2.導入組件 :這裡導入多個相同的內容會出現多個內容 三.函數渲染 1.內置函數渲染 使用方法: {{後臺傳輸的內容|內置函數}} 2.自定義函數渲染 1.在app中創建t ...
  • i標簽中的內容會以斜體顯示,b標簽中的內容會以加粗顯示 h5規範中規定,對於不需要著重的內容而是單純的加粗或者是斜體,就可以使用b和i標簽 在h5中使用small標簽來表示一些細則一類的內容 比如:合同中小字,網站的版權聲明都可以放到small 網頁中所有的加書名號的內容都可以使用cite標簽,表示 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...