关于js的冷知识、小技巧(持续更新)

8. 正确分割字符串

由于JavaScript只能处理UCS-2编码,造成所有字符在这门语言中都是2个字节,如果是4个字节的字符,会当作两个双字节的字符处理。JavaScript的字符函数都受到这一点的影响,无法返回正确结果。

比如包含emoji的字符串:🐂🍺🐂🍺🐂🍺🐂🍺

而要想正确分割这种特殊的字符串,可以使用如下方式

function splitString(string) {
  const length = string.length
  const output = []
  let index = 0
  while (index < length) {
    let charCode = string.charCodeAt(index)
    let character = string[index]
    if (charCode >= 0xD800 && charCode <= 0xDBFF) {
      output.push(character + string[++index]);
    } else {
      output.push(character);
    }
    index++
  }
  return output
}

具体原理请参考链接

8. 奇妙的隐式转换

(!+[]+[])[+!+[]]+([!![]]+[][[]])[+!+[]+[+!+[]]]+([][[]]+[])[+!+[]]; // ren

(!+[]+[])[+[]]+([![]]+[][[]])[+!+[]+[+[]]]+(![]+[])[+!+[]]+([][[]]+[])[+!+[]]; // tian

(![]+[])[+[]]+([][[]]+[])[+[]]; // fu

7. 关于js各种变量类型

  1. function类型变量,其length值为参数个数
  2. number/boolean/string类型的变量,其值一旦声明便无法改变
  3. 关于数字:
42.0 === 42; // true
42.toFixed(2); // error
42..toFixed(2); // 42.00
42 .toFixed(2); // 42.00
NaN !== NaN; // true

6. 实现call、apply

Function.prototype.call = function(context) {
  if (typeof this !== "function") {
    throw new Error("Function.prototype.bind - what is trying to be bound is not callable");
  }
  let L = arguments.length,
      args = [];
  for(let i=1;i<L;i++) {
    args.push(arguments[i]);
  }
  let fn = Symbol('fn');
  context[fn] = this;
  eval('context[fn](' + args + ')');
  delete context[fn];
}

Function.prototype.apply = function(context) {
  let args = arguments[1];
  let fn = Symbol('fn');
  context[fn] = this;
  eval('context[fn](' + args + ')');
  delete context[fn];
}

5. encodeURI 与 encodeURIComponent的区别

首先要了解下,URI (也就是网址) 中包含三种有效的字符:

  • 保留字符(reserved characters):这类字符是URI中的保留关键字符,它们用于分割URI中的各个部分,包括:";" | "/" | "?" | ":" | "@" | "&" | "=" | "+" | "$" | ","
  • Mark字符(mark characters):这类字符没有特别说明用途,可能与其他的RFC标准相关,包括:"-" | "_" | "." | "!" | "~" | "*" | "'" | "(" | ")"
  • 基本字符(alphanum characters):这类字符是URI中的主体部分,它包括所有的大写字母、小写字母和数字

而 encodeURI 会将这三种字符类型之外的其他字符进行转义编码(escape),所有的需要转义的字符都按照UTF-8编码转化成为一个、两个或者三个字节的十六进制转义字符(%x | %xx | %xxx)。
encodeURIComponent 相比 encodeURI 则是将保留字符也进行转义编码。这是为了满足当URI参数中带有URI的场景~

4. 一行代码生成随机字符串

const str = Math.random().toString(36).substr(2, 10);
console.log(str);

运行以上简短的代码,你会得到一个长度为10的随机字符串,短短一行代码得到这样的结果,可以说是很 cooooooooool

原理其实很简单,Math.random()获得了一个[0,1)的随机数,toString(36)将这个小数转化成了36进制的表示形式,最后substr(2,10)截取了从第2位开始的10个字符。

其中仅有的知识点,也就是小数的进制转换了。

所以实际上toString(30)、toString(20)也是可以获得随机字符串的,只不过36进制才能覆盖所有的英文字母。

很多开源库都使用此方式为DOM元素创建随机ID。

3. 将数字货币格式化的正则

let num = "19194575842775752";
num = num.replace(/(\d+?)(?=(\d{3})+\b)/g, '$1,');
console.log(num); // 19,194,575,842,775,752

2. 获取window对象

下面的代码可以在任何作用域中获得全局对象window(非严格模式下),因为一个单独的function中的this会指向window。

var global = (function () {
   return this;
}());

1. void 666

在一些源码中会看到这样的片段

function foo() {
    // do sth.
    return void 666; // 等同于 return undefined
}

解析:void是javascript的一个操作符,它后面可以接一个表达式,并且会立即执行后面的表达式,然后统一返回undefined。

所以无论接什么数字,结果都是undefined,一般为void 0,用666或23333纯属程序员的幽默哈哈哈~

至于为什么不直接用undefined,是因为undefined是一个合法的标识符,有可能被重新赋值,所以用void更为保险。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,226评论 0 4
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 第三章 基本概念 3.1 语法 ECMAScript标识符一般采用驼峰大小写格式,也就是第一个字母小写,剩下的每个...
    小雄子阅读 536评论 0 1
  • 插入模式 a -- 在光标后插入 o -- 在当前行后插入新的一行 O -- 在当前行前插入新的一行 cw -- ...
    onzing阅读 368评论 0 0