JavaScript中運算子規則的隱式類型轉換是什麼? 這是每個學習Javascript的新手們都應該知道的一個問題,下面這篇文章主要給大家介紹了關於JavaScript中運算符規則和隱式類型轉換的相關資料,需要的朋友可以參考借鑒,下面來一起看看吧。
前言
本文主要介紹給大家介紹了關於JavaScript運算子規則和隱含型別轉換的相關內容,並分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹吧。
隱含型別轉換
在JavaScript 中,當我們進行比較運算或加減乘除四則運算運算時,常常會觸發JavaScript 的隱式型別轉換機制;而這部分也往往是個令人困惑的地方。譬如瀏覽器中的 console.log 操作常常會將任何值都轉換為字串然後展示,而數學運算則會先將值轉換為數值類型(除了 Date 類型物件)然後進行操作。
我們先來看幾組典型的JavaScript 中運算子操作結果,希望閱讀完本部分之後能夠對每一個條目都能進行合理解釋:
// 比较 [] == ![] // true NaN !== NaN // true 1 == true // true 2 == true // false "2" == true // flase null > 0 // false null < 0 // false null == 0 // false null >= 0 // true // 加法 true + 1 // 1 undefined + 1 // NaN let obj = {}; {} + 1 // 1,这里的 {} 被当成了代码块 { 1 + 1 } + 1 // 1 obj + 1 // [object Object]1 {} + {} // Chrome 上显示 "[object Object][object Object]",Firefox 显示 NaN [] + {} // [object Object] [] + a // [object Object] + [] // 等价于 + "" => 0 {} + [] // 0 a + [] // [object Object] [2,3] + [1,2] // '2,31,2' [2] + 1 // '21' [2] + (-1) // "2-1" // 减法或其他操作,无法进行字符串连接,因此在错误的字符串格式下返回 NaN [2] - 1 // 1 [2,3] - 1 // NaN {} - 1 // -1
原始型別間轉換
#JavaScript 中我們常說的原始型別包含了數值型別、字串型別、布林型別與空類型這幾種;而我們常用的原始型別之間的轉換函數就是String、Number 與Boolean:
// String let value = true; console.log(typeof value); // boolean value = String(value); // now value is a string "true" console.log(typeof value); // string // Number let str = "123"; console.log(typeof str); // string let num = Number(str); // becomes a number 123 console.log(typeof num); // number let age = Number("an arbitrary string instead of a number"); console.log(age); // NaN, conversion failed // Boolean console.log( Boolean(1) ); // true console.log( Boolean(0) ); // false console.log( Boolean("hello") ); // true console.log( Boolean("") ); // false
最終,我們可以得到如下的JavaScript 原始型別轉換表(包括複合型別轉換為原始型別的範例):
原始值 | 轉換為數值型別 | #原始值 | |
---|---|---|---|
轉換為Boolean 類型 | #false | ||
#"false" | |||
#false | |||
true | 1 | "true" | true |
0 | 0 | "0" | false |
#1 | ##1 | "1" | true |
"0" | #0 | ##"0"#true | |
1 | "1" | true | |
NaN | "NaN" | false | |
Infinity | " Infinity" | true | |
#-Infinity | "-Infinity" | true | |
0 | "" | false | |
20 | "20" | true | |
NaN | |||
# "twenty" | true | [ ] | |
true | [20] | 20 | |
true | [10,20 ] | NaN | |
true | #["twenty"] | NaN | |
true | ["ten","twenty"] | NaN | |
true | function(){} | NaN |
ToPrimitive
在比较运算与加法运算中,都会涉及到将运算符两侧的操作对象转化为原始对象的步骤;而 JavaScript 中这种转化实际上都是由 ToPrimitive 函数执行的。实际上,当某个对象出现在了需要原始类型才能进行操作的上下文时,JavaScript 会自动调用 ToPrimitive 函数将对象转化为原始类型;譬如上文介绍的 alert 函数、数学运算符、作为对象的键都是典型场景,该函数的签名如下:
ToPrimitive(input, PreferredType?)
为了更好地理解其工作原理,我们可以用 JavaScript 进行简单地实现:
var ToPrimitive = function(obj,preferredType){ var APIs = { typeOf: function(obj){ return Object.prototype.toString.call(obj).slice(8,-1); }, isPrimitive: function(obj){ var _this = this, types = ['Null','Undefined','String','Boolean','Number']; return types.indexOf(_this.typeOf(obj)) !== -1; } }; // 如果 obj 本身已经是原始对象,则直接返回 if(APIs.isPrimitive(obj)) {return obj;} // 对于 Date 类型,会优先使用其 toString 方法;否则优先使用 valueOf 方法 preferredType = (preferredType === 'String' || APIs.typeOf(obj) === 'Date' ) ? 'String' : 'Number'; if(preferredType==='Number'){ if(APIs.isPrimitive(obj.valueOf())) { return obj.valueOf()}; if(APIs.isPrimitive(obj.toString())) { return obj.toString()}; }else{ if(APIs.isPrimitive(obj.toString())) { return obj.toString()}; if(APIs.isPrimitive(obj.valueOf())) { return obj.valueOf()}; } throw new TypeError('TypeError'); }
我们可以简单覆写某个对象的 valueOf 方法,即可以发现其运算结果发生了变化:
let obj = { valueOf:() => { return 0; } } obj + 1 // 1
如果我们强制将某个对象的 valueOf 与 toString 方法都覆写为返回值为对象的方法,则会直接抛出异常。
obj = { valueOf: function () { console.log("valueOf"); return {}; // not a primitive }, toString: function () { console.log("toString"); return {}; // not a primitive } } obj + 1 // error Uncaught TypeError: Cannot convert object to primitive value at <anonymous>:1:5
值得一提的是对于数值类型的 valueOf() 函数的调用结果仍为数组,因此数组类型的隐式类型转换结果是字符串。而在 ES6 中引入 Symbol 类型之后,JavaScript 会优先调用对象的 [Symbol.toPrimitive
] 方法来将该对象转化为原始类型,那么方法的调用顺序就变为了:
当
obj[Symbol.toPrimitive](preferredType)
方法存在时,优先调用该方法;如果 preferredType 参数为 String,则依次尝试
obj.toString()
与obj.valueOf()
;如果 preferredType 参数为 Number 或者默认值,则依次尝试
obj.valueOf()
与obj.toString()
。
而 [Symbol.toPrimitive]
方法的签名为:
obj[Symbol.toPrimitive] = function(hint) { // return a primitive value // hint = one of "string", "number", "default" }
我们同样可以通过覆写该方法来修改对象的运算表现:
user = { name: "John", money: 1000, [Symbol.toPrimitive](hint) { console.log(`hint: ${hint}`); return hint == "string" ? `{name: "${this.name}"}` : this.money; } }; // conversions demo: console.log(user); // hint: string -> {name: "John"} console.log(+user); // hint: number -> 1000 console.log(user + 500); // hint: default -> 1500
比较运算
JavaScript 为我们提供了严格比较与类型转换比较两种模式,严格比较(===)只会在操作符两侧的操作对象类型一致,并且内容一致时才会返回为 true,否则返回 false。而更为广泛使用的 == 操作符则会首先将操作对象转化为相同类型,再进行比较。对于
标准的相等性操作符(== 与 !=)使用了Abstract Equality Comparison Algorithm来比较操作符两侧的操作对象(x == y),该算法流程要点提取如下:
如果 x 或 y 中有一个为 NaN,则返回 false;
如果 x 与 y 皆为 null 或 undefined 中的一种类型,则返回 true(null == undefined // true);否则返回 false(null == 0 // false);
如果 x,y 类型不一致,且 x,y 为 String、Number、Boolean 中的某一类型,则将 x,y 使用 toNumber 函数转化为 Number 类型再进行比较;
如果 x,y 中有一个为 Object,则首先使用 ToPrimitive 函数将其转化为原始类型,再进行比较。
我们再来回顾下文首提出的 [] == ![] 这个比较运算,首先 [] 为对象,则调用 ToPrimitive 函数将其转化为字符串 "";对于右侧的 ![],首先会进行显式类型转换,将其转化为 false。然后在比较运算中,会将运算符两侧的运算对象都转化为数值类型,即都转化为了 0,因此最终的比较结果为 true。在上文中还介绍了 null >= 0 为 true 的这种比较结果,在 ECMAScript 中还规定,如果 = 为 true。
加法运算
对于加法运算而言,JavaScript 首先会将操作符两侧的对象转换为 Primitive 类型;然后当适当的隐式类型转换能得出有意义的值的前提下,JavaScript 会先进行隐式类型转换,再进行运算。譬如 value1 + value2 这个表达式,首先会调用 ToPrimitive 函数将两个操作数转化为原始类型:
prim1 := ToPrimitive(value1) prim2 := ToPrimitive(value2)
这里将会优先调用除了 Date 类型之外对象的 valueOf 方法,而因为数组的 valueOf 方法的返回值仍为数组类型,则会返回其字符串表示。而经过转换之后的 prim1 与 prim2 中的任一个为字符串,则会优先进行字符串连接;否则进行加法计算。
以上是詳細介紹JavaScript中運算子規則和隱式型別轉換的詳細內容。更多資訊請關注PHP中文網其他相關文章!

我使用您的日常技術工具構建了功能性的多租戶SaaS應用程序(一個Edtech應用程序),您可以做同樣的事情。 首先,什麼是多租戶SaaS應用程序? 多租戶SaaS應用程序可讓您從唱歌中為多個客戶提供服務

本文展示了與許可證確保的後端的前端集成,並使用Next.js構建功能性Edtech SaaS應用程序。 前端獲取用戶權限以控制UI的可見性並確保API要求遵守角色庫

JavaScript是現代Web開發的核心語言,因其多樣性和靈活性而廣泛應用。 1)前端開發:通過DOM操作和現代框架(如React、Vue.js、Angular)構建動態網頁和單頁面應用。 2)服務器端開發:Node.js利用非阻塞I/O模型處理高並發和實時應用。 3)移動和桌面應用開發:通過ReactNative和Electron實現跨平台開發,提高開發效率。

JavaScript的最新趨勢包括TypeScript的崛起、現代框架和庫的流行以及WebAssembly的應用。未來前景涵蓋更強大的類型系統、服務器端JavaScript的發展、人工智能和機器學習的擴展以及物聯網和邊緣計算的潛力。

JavaScript是現代Web開發的基石,它的主要功能包括事件驅動編程、動態內容生成和異步編程。 1)事件驅動編程允許網頁根據用戶操作動態變化。 2)動態內容生成使得頁面內容可以根據條件調整。 3)異步編程確保用戶界面不被阻塞。 JavaScript廣泛應用於網頁交互、單頁面應用和服務器端開發,極大地提升了用戶體驗和跨平台開發的靈活性。

Python更适合数据科学和机器学习,JavaScript更适合前端和全栈开发。1.Python以简洁语法和丰富库生态著称,适用于数据分析和Web开发。2.JavaScript是前端开发核心,Node.js支持服务器端编程,适用于全栈开发。

JavaScript不需要安裝,因為它已內置於現代瀏覽器中。你只需文本編輯器和瀏覽器即可開始使用。 1)在瀏覽器環境中,通過標籤嵌入HTML文件中運行。 2)在Node.js環境中,下載並安裝Node.js後,通過命令行運行JavaScript文件。

如何在Quartz中提前發送任務通知在使用Quartz定時器進行任務調度時,任務的執行時間是由cron表達式設定的。現�...


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

PhpStorm Mac 版本
最新(2018.2.1 )專業的PHP整合開發工具

ZendStudio 13.5.1 Mac
強大的PHP整合開發環境

Atom編輯器mac版下載
最受歡迎的的開源編輯器

SecLists
SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

WebStorm Mac版
好用的JavaScript開發工具