美文网首页
关于js的冷知识、小技巧(持续更新)

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

作者: 任无名F | 来源:发表于2017-08-24 17:47 被阅读0次

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更为保险。

相关文章

网友评论

      本文标题:关于js的冷知识、小技巧(持续更新)

      本文链接:https://www.haomeiwen.com/subject/ztpykxtx.html