深究JS底层原理

一、JS中八种数据类型判断方法

在JavaScript中,数据类型分为两大类:基本(原始)数据类型和引用(对象)数据类型。

基本数据类型(Primitive Data Types)

基本数据类型是表示简单的数据片段,它们是直接存储在变量中的数据。JavaScript中的基本数据类型有以下几种:

  1. String - 表示文本数据,例如 "Hello, World!"
  2. Number - 表示整数和浮点数,例如 42 和 3.14
  3. Boolean - 表示逻辑值 true 或 false
  4. Undefined - 表示未定义的值,即变量已声明但未初始化时的值。
  5. Null - 表示空值或者不存在的对象。在JavaScript中,null 是一个表示“无”的对象,转为数值时为 0
  6. Symbol (ES6新增) - 表示独一无二的值,常用于创建对象的私有成员。
  7. BigInt (ES2020新增) - 表示非常大的整数。

基本数据类型的值是不可变的,也就是说,一旦创建,就不能更改。如果对基本数据类型的变量进行操作,实际上是创建了一个新的变量或值。

引用数据类型(Reference Data Types)

引用数据类型表示的是数据的集合,它们存储的是对数据的引用(内存地址),而不是数据本身。在JavaScript中,最常见的引用数据类型是对象(Object),还包括以下几种:

  1. Object - 表示一组无序的键值对,例如 {name: "Alice", age: 25}
  2. Array - 表示一组有序的数据集合,例如 [1, 2, 3, 4]
  3. Function - 表示函数,JavaScript中的函数是一等公民,可以作为对象使用。
  4. Date - 表示日期和时间。
  5. RegExp - 表示正则表达式。

引用数据类型的值是可变的,这意味着可以改变它们的内容而不改变它们的身份。例如,可以向对象添加属性或修改数组中的元素。

基本数据类型与引用数据类型的区别

以下是一些基本数据类型和引用数据类型之间的关键区别:

  • 存储方式:基本数据类型的值直接存储在变量中,而引用数据类型存储的是内存地址。
  • 赋值:基本数据类型赋值时是传递值,而引用数据类型赋值时是传递引用。
  • 比较:基本数据类型比较的是值,而引用数据类型比较的是引用地址。
  • 参数传递:在函数中,基本数据类型作为参数传递时,传递的是值的副本;引用数据类型传递的是引用地址。

理解这些差异对于编写高效和正确的JavaScript代码非常重要。

八种数据类型判断方式

二、NaN

三、数值千分位的六种方法

1.format with array

(number + '') 是一种常见的类型转换技巧,它将数字 number 转换为字符串。无论 number 是什么数字,加上一个空字符串 '' 后,结果都会是一个字符串形式的 number。

function format_with_array(number) {// 转为字符串,并按照.拆分var arr = (number + '').split('.');// 整数部分再拆分var int = arr[0].split('');// 小数部分var fraction = arr[1] || '';// 返回的变量var r = "";var len = int.length;// 倒叙并遍历int.reverse().forEach(function (v, i) {// 非第一位并且是位值是3的倍数, 添加 ","if (i !== 0 && i % 3 === 0) {r = v + "," + r;} else {// 正常添加字符r = v + r;}})// 整数部分和小数部分拼接return r + (!!fraction ? "." + fraction : '');
}const print = console.log;
print(format_with_array(938765432.02));938765432.02
  1. var arr = (number + '').split('.');

    这行代码首先通过 number + '' 将数字 number 转换为字符串。这是因为 split 方法是字符串的方法,不能直接在数字上使用。
    然后使用 split('.') 方法将字符串按照小数点(.)分割成两部分,结果存储在数组 arr 中。如果 number 是一个整数,那么 arr 将只有一个元素,即整数部分;如果 number 是一个小数,那么 arr将有两个元素,第一个元素是整数部分,第二个元素是小数部分。
  2. var int = arr[0].split('');

    这行代码将 arr 数组中的第一个元素(即整数部分)再次使用 split('') 方法进行分割。由于没有指定分隔符,实际上是将整数部分的每个字符分割成一个单独的字符串,并将结果存储在数组 int 中。例如,如果整数部分是 "123",那么 int 将是 ["1", "2", "3"]
  3. var fraction = arr[1] || '';

    这行代码尝试获取 arr 数组中的第二个元素(即小数部分)。如果 arr 只有一个元素(即 number 是一个整数),那么 arr[1] 将是 undefined
    使用逻辑或运算符 ||,如果 arr[1] 是 undefined(即没有小数部分),则将 fraction 设置为一个空字符串 ''。如果存在小数部分,则 fraction 将是小数部分的字符串。

return r + (!!fraction ? "." + fraction : '');

  1. r +:这是字符串拼接的开始,将数字 r 转换为字符串并准备拼接。

  2. !!fraction:这是双重逻辑非操作符,用于将 fraction 转换为布尔值。如果 fraction 是非空字符串、非零数字或任何“真值”对象,则 !!fraction 将为 true;如果 fraction 是空字符串、零、nullundefined 或任何“假值”,则 !!fraction 将为 false

  3. ? "." + fraction : '':这是一个条件(三元)运算符,它根据 !!fraction 的结果来决定返回的值。如果 !!fraction 为 true(即 fraction 有值),则返回 "." + fraction,即小数点加上 fraction 的值;如果 !!fraction 为 false(即 fraction 没有值或为假值),则返回空字符串 ''

整体来看,这段代码的作用是:

如果 fraction 有值,那么返回的字符串将是 r 后面跟着一个小数点和 fraction 的值。如果 fraction 没有值或为假值,那么返回的字符串就只是 r 转换成的字符串。

2.format with substring

function format_with_substring(number) {// 数字转字符串, 并按照 .分割var arr = (number + '').split('.');var int = arr[0] + '';var fraction = arr[1] || '';// 多余的位数var f = int.length % 3;// 获取多余的位数,f可能是0, 即r可能是空字符串var r = int.substring(0, f);// 每三位添加","和对应的字符for (var i = 0; i < Math.floor(int.length / 3); i++) {r += ',' + int.substring(f + i * 3, f + (i + 1) * 3)}//多余的位数,上面if (f === 0) {r = r.substring(1);}// 整数部分和小数部分拼接return r + (!!fraction ? "." + fraction : '');
}const print = console.log;
print(format_with_substring(938765432.02));

3.format with mod

function format_with_mod(number) {var n = number;  // 将传入的参数 number 赋值给变量 n,用于后续计算,保留原始数值不变。var r = "";     // 初始化一个空字符串 r,用于存储格式化后的结果。var temp;       // 声明一个变量 temp,用于暂存计算过程中的中间值。do {mod = n % 1000;  // 计算 n 除以 1000 的余数,即取 n 的最后三位数字(包括小数部分)。n = n / 1000;    // 将 n 除以 1000,即每次循环处理掉 n 的最后三位数字。temp = ~~mod;    // 使用双按位非操作符(~~)来取整,去除 mod 的小数部分,获取整数部分。// 以下代码用于构造格式化字符串:// 如果 n 大于等于 1,说明还有更多的数字需要处理,因此使用 padStart(3, "0") 来确保 temp 是三位数字,不足部分用 "0" 填充。// 如果 r 已经有内容,则在 temp 前面加上 ",",否则直接使用 temp。r = (n >= 1 ? `${temp}`.padStart(3, "0") : temp) + (!!r ? "," + r : "");} while (n >= 1);  // 当 n 大于等于 1 时,继续循环,直到 n 小于 1。var strNumber = number + "";  // 将数字 number 转换为字符串,以便后续查找小数点位置。var index = strNumber.indexOf(".");  // 查找小数点的位置。// 如果存在小数点(index >= 0),则将原始数字的小数部分添加到格式化字符串 r 的末尾。if (index >= 0) {r += strNumber.substring(index);}return r;  // 返回格式化后的字符串。
}const print = console.log;  // 将 console.log 方法赋值给变量 print,便于后续使用。
print(format_with_mod(38765432.02));  // 调用 format_with_mod 函数并打印结果。

4.format with regex

const print = console.log;
print(/hello (?=[a-z]+)/.test("hello a"));
print(/hello (?=[a-z]+)/.test("hello 1"));  
function format_with_regex(number) {var reg = /\d{1,3}(?=(\d{3})+$)/g;return (number + '').replace(reg, '$&,');
}// 987, 654, 321// const print = console.log;
// print(format_with_regex(987654321));function format_with_regex(number) {var reg = /\d{1,3}(?=(\d{3})+$)/g;return (number + '').replace(reg, function(match, ...args){    console.log(match, ...args);return match + ','});
}
format_with_regex(987654321)// function format_with_regex(number) {
//     var reg = /(\d)(?=(?:\d{3})+$)/g
//     return (number + '').replace(reg, '$1,');
// }

5.format with toLocaleString

function format_with_toLocaleString(number, minimumFractionDigits, maximumFractionDigits) {minimumFractionDigits = minimumFractionDigits || 2;maximumFractionDigits = (maximumFractionDigits || 2);maximumFractionDigits = Math.max(minimumFractionDigits, maximumFractionDigits);return number.toLocaleString("en-us", {maximumFractionDigits: maximumFractionDigits || 2,minimumFractionDigits: minimumFractionDigits || 2})
}function format_with_toLocaleString(number) {return number.toLocaleString("en-us")
}const print = console.log;
print(format_with_toLocaleString(938765432.02));

6.Intl.NumberFormat

function format_with_Intl(number, minimumFractionDigits, maximumFractionDigits) {minimumFractionDigits = minimumFractionDigits || 2;maximumFractionDigits = (maximumFractionDigits || 2);maximumFractionDigits = Math.max(minimumFractionDigits, maximumFractionDigits);return new Intl.NumberFormat('en-us', {maximumFractionDigits: maximumFractionDigits || 2,minimumFractionDigits: minimumFractionDigits || 2}).format(number)
}// 使用默认配置选项
// function format_with_Intl(number) {
//     return new Intl.NumberFormat('en-us').format(number)
// }const print = console.log;
print(format_with_Intl(938765432.02));

7.效果对比

对比一下这六种方法的性能

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><meta http-equiv="X-UA-Compatible" content="ie=edge"><title>数字千分位</title>
</head><body><div style="text-align:center"><p><input type="number" value="5000" id="textCount" /></p><p><input type="button" onclick="executeTest()" value="测试" /><input type="button" onclick="javascript:document.getElementById('messageEl').innerHTML=''" value="清除" /></p></div><div id="messageEl" style="width: 300px;margin: auto;"></div><script>function format_with_array(number) {var arr = (number + '').split('.');var int = arr[0].split('');var fraction = arr[1] || '';var r = "";var len = int.length;int.reverse().forEach(function(v, i) {if (i !== 0 && i % 3 === 0) {r = v + "," + r;} else {r = v + r;}})return r + (!!fraction ? "." + fraction : '');}</script><script>function format_with_substring(number) {var arr = (number + '').split('.');var int = arr[0] + '';var fraction = arr[1] || '';var f = int.length % 3;var r = int.substring(0, f);for (var i = 0; i < Math.floor(int.length / 3); i++) {r += ',' + int.substring(f + i * 3, f + (i + 1) * 3)}if (f === 0) {r = r.substring(1);}return r + (!!fraction ? "." + fraction : '');}</script><script>function format_with_mod(number) {var n = number;var r = "";var temp;var mod;do {mod = n % 1000;n = n / 1000;temp = ~~mod;r = (n > 1 ? `${temp}`.padStart(3, "0") : temp) + (!!r ? "," + r : "")} while (n > 1)var strNumber = number + "";var index = strNumber.indexOf(".");if (index > 0) {r += strNumber.substring(index);}return r;}</script><script>function format_with_regex(number) {var reg = /(\d{1,3})(?=(\d{3})+(?:$|\.))/g;// var reg = /(\d)(?=(?:\d{3})+$)/g//var reg = /\d{1,3}(?=(\d{3})+$)/g;return (number + '').replace(reg, '$1,');}</script><script>// function format_with_toLocaleString(number, minimumFractionDigits, maximumFractionDigits) {//     minimumFractionDigits = minimumFractionDigits || 2;//     maximumFractionDigits = (maximumFractionDigits || 2);//     maximumFractionDigits = Math.max(minimumFractionDigits, maximumFractionDigits);//     return number.toLocaleString("en-us", {//         maximumFractionDigits: maximumFractionDigits || 2,//         minimumFractionDigits: minimumFractionDigits || 2//     })// }function format_with_toLocaleString(number) {return number.toLocaleString("en-us")}</script><script>// function format_with_Intl(number, minimumFractionDigits, maximumFractionDigits) {//     minimumFractionDigits = minimumFractionDigits || 2;//     maximumFractionDigits = (maximumFractionDigits || 2);//     maximumFractionDigits = Math.max(minimumFractionDigits, maximumFractionDigits);//     return new Intl.NumberFormat('en-us', {//         maximumFractionDigits: maximumFractionDigits || 2,//         minimumFractionDigits: minimumFractionDigits || 2//     }).format(number)// }const format = new Intl.NumberFormat('en-us');function format_with_Intl(number) {return format.format(number)}</script><script>function getData(count) {var data = new Array(count).fill(0).map(function(i) {var rd = Math.random();var r = rd * Math.pow(10, Math.trunc(Math.random() * 12));if (rd > 0.5) {r = ~~r;}return r})return data;}function test(data, fn, label) {var start = performance.now();for (var i = 0; i < data.length; i++) {fn(data[i]);}var time = performance.now() - start;message((fn.name || label) + ":" + time.toFixed(2) + "ms");}function executeTest() {var data = getData(+textCount.value);test(data, format_with_array);test(data, format_with_mod);test(data, format_with_substring);test(data, format_with_regex);test(data, format_with_toLocaleString);test(data, format_with_Intl);message("-------------------")}function message(msg) {var el = document.createElement("p");el.innerHTML = msg;messageEl.appendChild(el);}</script></body></html>

三、[]+[],{}+{},[]+{},{}+[]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/59806.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ssm071北京集联软件科技有限公司信息管理系统+jsp(论文+源码)_kaic

毕 业 设 计&#xff08;论 文&#xff09; 题目&#xff1a;北京集联软件科技有限公司信息管理系统 \ 摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本信息…

Yocto 项目下通过网络更新内核、设备树及模块

Yocto 项目下通过网络更新内核、设备树及模块 前言 在 Yocto 项目的开发过程中&#xff0c;特别是在进行 BSP&#xff08;Board Support Package&#xff09;开发时&#xff0c;经常需要调整特定软件包的版本&#xff0c;修改内核、设备树以及内核模块。然而&#xff0c;每次…

k8s集群安装(kubeadm)

k8s集群安装&#xff08;kubeadm&#xff09; 1、环境准备&#xff08;master和node节点都执行&#xff09;1.1、替换yum源1.2、关闭selinux1.3、永久关闭防火墙1.4、永久关闭swap1.5、修改主机名添加host1.6、时间同步1.7、将桥接的IPv4流量传递到iptables的链1.8、docker安装…

【日常问题排查小技巧-连载】

线上服务CPU飙高排查 先执行 top&#xff0c;找到CPU占用比较高的进程 id&#xff0c;&#xff08;比如 21448&#xff09; jstack 进程 id > show.txt&#xff08;jstack 21448 > show.txt&#xff09; 找到进程中CPU占用比较高的线程&#xff0c;线程 id 转换为 16 进…

您与此网站之间建立的连接不安全解决方法

如果你打开网站&#xff0c;地址栏有警告&#xff0c;点进去是这样的提示&#xff1a;您与此网站之间建立的连接不安全&#xff0c;了解详细信息。 请勿在此网站上输入任何敏感信息&#xff08;例如密码或信用卡信息&#xff09;&#xff0c;因为攻击者可能会盗取这些信息。 …

【与AI+】学习SAP开发有什么渠道可以推荐

前言&#xff1a;好的&#xff0c;我又将开辟一个新的专栏&#xff0c;这个专栏呢&#xff0c;就准备放一些我向AI提问的问题&#xff0c;以及AI的回答。因为感觉真的好方便哈哈哈~ 我不是很确定我的专栏文章内容是否涉及版权&#xff0c;以及也不确定这些整合过的文字是否涉嫌…

江苏博才众创科技产业园集团拟投资10亿元在泰兴打造汽车零部件产业园

2024年11月7日&#xff0c;泰兴市高新技术产业开发区与江苏博才众创科技产业园集团举行新能源汽车零部件智能制造产业园项目签约仪式。 泰兴市高新区党工委委员、管理办副主任王峰表示&#xff1a;高新区是全市项目建设的主阵地&#xff0c;近年来聚焦高端化、智能化、绿色化&a…

【python】Flask

文章目录 1、Flask 介绍2、Flask 实现网页版美颜效果3、参考 1、Flask 介绍 Flask 是一个用 Python 编写的轻量级 Web 应用框架。它设计简单且易于扩展&#xff0c;非常适合小型项目到大型应用的开发。 以下是一些 Flask 库中常用的函数和组件&#xff1a; 一、Flask 应用对…

产品经理如何使用项目管理软件推进复杂项目按时上线

前言 相信很多产品同学或多或少都有过这样的经历&#xff1a;平时没有听到任何项目延期风险&#xff0c;但到了计划时间却迟迟无法提测……评审时没有任何argue&#xff0c;提测后发现开发的功能不是自己想要的……费劲九牛二虎之力终于让项目上线了&#xff0c;然而发现成果达…

新系统如何进行模型环境配置

在机器学习和深度学习中&#xff0c;一个良好的开发环境能够显著提高工作效率。本篇博客将详细介绍如何在新的Linux系统&#xff08;以Ubuntu为例&#xff09;上进行模型环境的配置&#xff0c;包括基础系统设置、Python虚拟环境搭建、常用库的安装以及GPU驱动和CUDA的安装等。…

OpenAI大事记;GPT到ChatGPT参数量进化

目录 OpenAI大事记 GPT到ChatGPT参数量进化 OpenAI大事记 GPT到ChatGPT参数量进化 ChatGPT是从初代 GPT逐渐演变而来的。在进化的过程中,GPT系列模型的参数数量呈指数级增长,从初代GPT的1.17亿个参数,到GPT-2的15 亿个参数,再到 GPT-3的1750 亿个参数。模型越来越大,训练…

Am I Isolated:一款安全态势基准测试工具

基于Rust的容器运行时扫描器作为一个容器运行&#xff0c;检测用户容器运行时隔离中的漏洞。 它还提供指导&#xff0c;帮助用户改善运行时环境&#xff0c;以提供更强的隔离保证。 容器的现状是它们并不包含&#xff08;隔离&#xff09;。 容器隔离的缺失在云原生环境中有…

SQLite的BLOB数据类型与C++二进制存储学习记录

一、BLOB数据类型简介 Blob&#xff08;Binary Large Object&#xff09;是一种用于存储二进制数据的数据类型&#xff0c;在数据库中常用于存储图片、音频和视频等大型&#xff08;大数据量&#xff09;的二进制数据[1-2]。需要注意的是&#xff0c;SQLite中BLOB类型的单对象最…

编写一个基于React的聊天室

前言 此前已经编写了一版后端的im&#xff0c;此次就用其作为服务端&#xff0c;可查看参考资料1 代码 使用WebStorm创建React项目 安装依赖包 PS C:\learn-demo\front\chatroom> npm installadded 183 packages, and audited 184 packages in 16s43 packages are looki…

Javascript事件循环流程分析

基础概念 事件循环&#xff08;Event Loop&#xff09;&#xff1a;事件循环是JavaScript运行时环境中的一个循环机制&#xff0c;它不断地检查调栈用和任务队列。当调用栈为空时&#xff0c;事件循环会首先检查微任务队列&#xff0c;并执行其中的所有任务。只有当微任务队列…

解决使用Golang的email库发送qq邮件报错short response,错误类型为textproto.ProtocolError

问题阐述 使用email库发送QQ邮件&#xff0c;采用587端口&#xff1a; package mainimport ("fmt""net/smtp""github.com/jordan-wright/email" )func SendEmail(sendTo string, subject string, body string) (err error) {e : email.NewEmai…

题目:Wangzyy的卡牌游戏

登录 - XYOJ 思路&#xff1a; 使用动态规划&#xff0c;设dp[n]表示当前数字之和模三等于0的组合数。 状态转移方程&#xff1a;因为是模三&#xff0c;所以和的可能就只有0、1、2。等号右边的f和dp都表示当前一轮模三等于k的组合数。以第一行为例&#xff1a;等号右边表示 j转…

【支付行业-支付系统架构及总结】

记得第一次看埃隆马斯克&#xff08;Elon Musk&#xff09;讲第一性原理的视频时&#xff0c;深受震撼&#xff0c;原来还可以这样处理复杂的事务。这篇文章也尝试化繁为简&#xff0c;探寻支付系统的本质&#xff0c;讲清楚在线支付系统最核心的一些概念和设计理念。 虽然支付…

模块化沙箱:深信达如何为数据安全提供全方位保护

在数字化时代&#xff0c;网络安全已经成为企业和个人不可忽视的重要议题。随着网络攻击手段的日益复杂和多样化&#xff0c;传统的安全防护措施已经难以应对日益严峻的安全挑战。在这样的背景下&#xff0c;模块化沙箱技术应运而生&#xff0c;成为网络安全领域的新宠。今天&a…

基于单片机的观赏类水草养殖智能控制系统的设计(论文+源码)

1总体设计 通过需求分析&#xff0c;本设计观赏类水草养殖智能控制系统的总体架构如图2.1所示&#xff0c;为系统总体设计框图。系统采用STM32单片机作为系统主控核心&#xff0c;利用DS18B20温度传感器、TDS传感器、CO2传感器、光敏传感器实现水草养殖环境中水温、CO2浓度、T…