函数词法作用域 是在函数定义时的作用域,一直向上找到全局作用域,不是运行时作用域, 以下例子正好说明这个结果
this 既不指向函数自身也不指向函数的词法作用域。
this 实际上是在函数被调用时发生的绑定,它指向什么完全取决于函数在哪里被调用。
// JavaScript 环境中内置的 setTimeout() 函数实现和下面的伪代码类似:
// function setTimeout(fn,delay) { // 等待 delay 毫秒
// fn(); // <-- 调用位置!
// }
// 所以 setTimeout就是一个函数调用,然后其参数就是传递的参数, 有时候会误认为参数中的this 是在运行这个回调函数时才查找的,
// 还是对于函数不够深入理解
// setTimeout(this.fn, 1000)
第3天
typeof null === 'object'
在计算机底层,对象都是二进制形式,JS对前三位都是0的判定为对象,而null 都是0。
这样就好理解了 起码有个底层道理,不然凭空认为是这样,觉得太儿戏。
这是一个JS bug, 但是无法修复了,因为很多系统项目在用。
null
是关键字,不可以被覆盖。
原文为:注 1:原理是这样的,不同的对象在底层都表示为二进制,在 JavaScript 中二进制前三位都为 0 的话会被判 断为 object 类型,null 的二进制表示是全 0,自然前三位也是 0, 所以执行 typeof 时会返回“object”。
Undefined
: 不是关键字 ,可以被覆盖,而使用void(0) 可以很安全的使用undefined值
4、基本类型的变量 调用的方法,本质是引擎自动将字面量形式的转换为对象,然后在调用其原型上的方法。
let str = 'i am is string'; str.slice(0);
实际转换成String对象 ,这样就能理解了 为什么能调用这些方法,good
第4天
-
对象:数组也是一种对象,可以添加属性,但是因为不符合下标规范,所以数组的length没有变化,所以下面第4个值不存在 进一步讲,要是添加的属性为数字型的字符串呢,那当然可以被JS引擎转化成下标形式
let arr = [1, 2, 3]; arr.len = 4; arr['3'] = 4
console.log(arr, arr.length) // [ 1, 2, 3, len: 3 ], 3
-
遍历对象的几种方法
Object.keys()
//会遍历对象本身上的所有属性成一个数组集合,不包括设置为不能枚举的属性Object.getOwnPropertyNames
//会遍历对象本身上的所有属性成一个数组集合,但包括设置为不能枚举的属性for in
会遍历对象自身及其原型链上的属性myObject.hasOwnProperty()
只针对某个属性是否是在对象本身上,所以需要和for in配合
第5天
事件循环
:
- 宿主环境和引擎关系,如同浏览器和V8关系。
- 在ES6之前引擎负责执行宿主环境调取过来的代码,所以事件循环是由宿主环境搭建的。而ES6之后,引擎负责制定事件循环,因为有了Promise
setTimeout
并不是立马把程序放在事件循环队列中,而是设置了一个定时器,时间到了才把回调放进循环队列中,此时,队列中可能有多个程序了,所以 它的程序运行时间,可能在那个时刻,也可能在那个时刻之后。- Promise发生在一次轮询tick的末尾,和下一轮轮询Tick之前。
- 程序的完整性运行:是指函数内部的代码都是同步代码,而在函数这个级别上它的顺序是多种可能的。非完整性程序内部有异步程序,比如ajax promise setTimeout等
第6天
为什么用promise 首先为什么摒弃回调,因为顺序的同步流代码对人们的大脑来说是可接受的,而回调的存在会打乱这些步骤,造成一些列不可理解、不可维护、糟糕的代码。 异步流代码不单单存在回调地狱(嵌套)的问题,还有一个最重要的问题就是上面说的代码的不好维护,太糟糕。比如下面的代码
listen( "click", handler );
function handler() {
setTimeout( request, 500 );
}
function request(){
ajax( "http://some.url.1", response );
}
function response(text){ if (text == "hello") {
handler(); }
else if (text == "world") { request();
} }
我们的眼睛需要不断的来回上下翻看程序执行顺序,对于程序员来说 这是一种最大的心智负担了吧。
- promise 不是对回调的替代,其实是对回调代码和将来执行这个任务的异步代码之间提供了一种可靠的中间机制来管理回调。
- promise可以优雅的处理 回调地狱的劣势。
- Promise 决议后就是外部不可变的值,这是 Promise 设计中最基础和 最重要的因素,这是关于 Promise 需要理解的最强大也最重要的一个概念。
第7天
JS的继承 不是面向对象的那种继承,也不是对象的复制,就是基于原型。 相对于继承,用委托更为确切,一个对象通过一种链接到另一个对象,然后可以取到另一个对象的属性和方法。 原文:继承意味着复制操作,JavaScript(默认)并不会复制对象属性。相反,JavaScript 会在两 个对象之间创建一个关联,这样一个对象就可以通过委托访问另一个对象的属性和函数。 委托(参见第 6 章)这个术语可以更加准确地描述 JavaScript 中对象的关联机制
第8天
JS继承的理解 JS根本不是所谓的继承机制,因为继承是需要复制父辈的属性和方法的,而JS是通过原型链获取属性和方法,没有复制。 通过'构造函数'举例来说
function fun() {
console.log('fun')
}
fun.prototype = {}
const a = new fun()
console.log('a.constructor=', a.constructor) // ?
原来的固有印象是通过构造函数fun new出来的对象a 会有一个属性指向其构造函数fun,其实错了,a根本没有属性指向fun,
a.constructor
是通过原型链去找,如果在fun.prototype
找到了就是fun.prototype
的指向,如果没找到,那么继续往上找,
一直找到Object.prototype
,而它的constructor =》Object
函数,故上面的结果为
console.log(a.constructor) // function Object //注意这里的Object 为函数
继续,故需要在其原型上加这个constructor
属性指向fun
函数, 并且是不可枚举的,那就用到了属性描述
Object.defineProperty(fun.prototype, 'constructor', {
enumerable: false,
writable: true,
configurable: true,
value: fun,
})
console.log(a.constructor) // function fun
再往下深入的话,那其实这个constructor
有什么作用呢,既然我可以随便定义这个属性。
以上可以深刻理解了 JS的所谓继承,根本不存在,本质还是原型,或者链条更为确切。
和其他语言的面向对象不是一回事。
可以这样形容上述的函数构造对象的过程,本质是创建一个对象a,然后其原型是fun.prototype
第9天
vue源码中用到了Object.create(null)
创建空的对象,去掉了原型链(本质是原型关联指向了null)储存一些数据。
Object.create
ES5兼容写法
if (!Object.create) {
Object.create = function (o) {
const f = function () {}
f.prototype = o
return new f()
// or
// const obj = {}
// obj.__proto__ = o //IE不支持 哈哈 只是表达一种思路
// return obj
}
}
这个兼容性(polyfill)写法,再次说明了委托原型链的本质,创建一个对象,这个对象的原型指向了要关联的对象。 怎么创建一个对象 可以通过一个new 一个函数:
/* 官方解释
* 使用 new 来调用函数,或者说发生构造函数调用时,会自动执行下面的操作。
1. 创建(或者说构造)一个全新的对象。
2. 这个新对象会被执行[[原型]]连接。
3. 这个新对象会绑定到函数调用的this。
4. 如果函数没有返回其他对象,那么new表达式中的函数调用会自动返回这个新对象。
* */
第10天
通过两个对象之间建立委托联系的形式,更好的实现了关注点分离。举例 首先是类的形式,'子类'需要call 一下'父类',获取父的属性,然后通过创建一个新对象指向父函数的原型并赋到子类prototype上实现原型的关联
function fa(name) {
this.name = name
}
fa.prototype.getName = function () {
return this.name
}
function fb(name, age) {
fa.call(this, name)
this.age = age
}
fb.prototype = Object.create(fa.prototype)
const f = new fb('li', 100)
console.log(f.getName())
// 这种方式一点都不分离化,也很费解。
// 以下是对象形式的委托关联
const obj1 = {
getName() {
return this.name
},
}
const obj2 = {
name: 'li',
getAge() {
return this.age
},
}
Object.setPrototypeOf(obj2, obj1)
console.log(obj2.getName())
// 这种方式 很容易发现 两者之间并不存在很紧密的关系,通过原型链进行关联。
// 联想到日常开发中, 写组件或者写业务,尽量做到分离、低耦合。
ES6中 class 中原型的探究
es6 class 说到底是隐藏了一些细节关系,不用通过显示的prototype来关联两个对象, 总结其实就是一种语法糖.
class Widget {
constructor(width, height) {
this.width = width || 50
this.height = height || 50
this.$elem = null
}
render($where) {
if (this.$elem) {
this.$elem
.css({ width: this.width, height: this.height })
.appendTo($where)
}
}
}
class Button extends Widget {
constructor(width, height, label) {
super(width, height) // 类似执行了 Widget.call(this, width, height)
this.label = label || 'Default'
this.$elem = 'button'
}
render($where) {
super.render($where) // 往上找原型widget中的render方法 也是一种委托关联关系,本质不变
this.$elem.click(this.onClick.bind(this))
}
onClick(evt) {
console.log("Button '" + this.label + "' clicked!")
}
}
const btn = new Button(20, 30, null)
console.log(btn)
第十一、十二天
数字
0.1 + 0.2 === 0.3 // false 5.1+5.2 === 5.3 // false 由于二进制浮点数对于0.1 0.2实现都不是真正的这个值,故相加结果不是0.3
原文:使用二进制浮点数最著名的副作用(请记住,是所有使用IEEE 754的语言——不仅是JavaScript) 故前端在涉及这种计算的时候 要特别小心. 规避措施有 使用一个微小的误差值Number.EPSILON 其实就是2^-52值(Math.pow(2,-52))
const num1 = 0.3
const num2 = 0.1 + 0.2
console.log(Math.abs(num1 - num2) < Number.EPSILON) // true // Math.abs取一下绝对值
// 整数的范围为下面两个
// Number.MAX_SAFE_INTEGER; // Math.pow( 2, 53 ) - 1
// Number.MIN_SAFE_INTEGER //
// 测试是不是安全整数的方法
Number.isSafeInteger(Math.pow(2, 53))
// isNaN 是唯一一个不等于其自身的值, 检测方法有ES6的Number.isNaN() 和ES5的window.isNaN() //后者会把非数字类型的也检测成true
第14天
类型转换
console.log(0 == [null]) // true // [null]会先通过toString()转换成字符串'' ,然后数字和字符串相比, 会将字符串''转换成数字 0
// null == undefined 除了这俩 其他任何和他俩比较都是false
第15天
try finally: finally中语句总是会执行,即便try中有return, 但是会让代码变得有些难懂。最好不要这样
const tryFinally = function () {
try {
return 1
} finally {
console.log(111)
}
}
console.log(tryFinally())
总结:语法、操作符、TDZ、容错等就是日常开发中的细节,这要在平日里注意积累,不要用那些难懂的语句编程。 ECMAScript 是规范,而JS是在浏览器上的实现
第16天
// Benchmark.js 库 测试代码的性能,抛弃以前设置开始事件、结束事件的劣质做法。
// jsPerf(http://jsperf.com) 在线测试
// 尾调用优化TCO,函数内部最后执行的函数,引擎会优化内存空间
第20天
// 使用generator的原因是为了用同步的书写方式实现异步,故需要把异步 promise的一些实现细节隐藏起来,作为一个函数,而不必都放在当前函数中
// 下面的代码就是把promise.all封装起来,这俩并发请求作为一个函数,在实际开发过程中很有用,让代码看起来更好维护
// 注:普通函数,不是生成器
function bar(url1,url2) {
return Promise.all( [request( url1 ), request( url2 )] );
}
function *foo() {
// 隐藏bar(..)内部基于Promise的并发细节
const results = yield bar("http://some.url.1", "http://some.url.2" )
const r1 = results[0];
const r2 = results[1];
const r3 = yield request( "http://some.url.3/?v=" + r1 + "," + r2);
console.log( r3 );
}
在 *foo() 内部,我们所做的一切就是要求 bar(..) 给我们一些 results,并通过 yield 来等待结果,这样更简洁也更清晰。 我们不需要关心在底层是用Promise.all([ .. ]) Promise 组合来实现这一切。
第21天
引擎 编译器 作用域 以var a = 2 为例 // 编译器: 词法分析、语法分析,生成可执行代码,过程如下: // 词法分析: 将该语句,解析成var, a , = , 2 这几个token // 语法分析: 将tokens分词转换成AST语法树结构。 // 生成代码:在当前作用域内查找是否存在a,如果存在,则忽略var声明,如果不存在,则声明一个a变量,在这部分不会进行赋值操作 // 引擎:作为运行时执行编译器生成的代码,过程如下: // 在当前作用域内LHS(左查询)a变量,如果不存在,则继续向上作用域内查找,如果找到了,则把值2赋值给a,如果找不到,直到追寻到全局作用域。 // 如果都找不到,在非严格模式下,那么就会在全局环境声明a,并把2赋值。如果在严格模式下,报ReferenceError. // 上面提到的左查询对应的右查询,可以理解为赋值右边的变量或者取值的行为。比如console.log(b); return a + b等等
第22天
// 判断类型
//instanceof 和多全局对象(例如:多个 frame 或多个 window 之间的交互)
// 在浏览器中,我们的脚本可能需要在多个窗口之间进行交互。多个窗口意味着多个全局环境,不同的全局环境拥有不同的全局对象,从而拥有不同的内置类型构造函数。这可能会引发一些问题。比如,表达式 [] instanceof window.frames[0].Array 会返回 false,因为 Array.prototype !== window.frames[0].Array.prototype,并且数组从前者继承。
语言精粹这本书学到的
深刻理解了小时候经常说的好记性不如烂笔头
- 先随时记录下糟粕
// 1、let a; typeof a === 'undefined'; typeof b === 'undefined'
- 对于未声明的变量,typeof 由于有容错机制,会和未赋值的变量一样返回undefined 这样无法区分出来,如果后者能返回 'undeclared' 就能明了.但是呢,有了这种防范机制,在日常开发中,如果进行polyfill,那用typeof
先检查是否存在,再补充当前环境缺失的功能。
比如
if (typeof Promise !== 'undefined') {....}
- 2、Number类型中非数字值 NaN 不等于其自身,这是JS中唯一一个。
注释
多行注释也需要用 '//'
因为 /* let reg = /\*/g */
这种情况会导致后面的没注释
以前觉得2种都可以 没细究。
{}
代码块,在ES6之前 代码块内部定义的变量会在上一层也能获取,之后的话,通过let const 只会在内部作用域生效。- 引申到词法作用域,与闭包。
- 为什么对象属性中带有- 的键必须用引号包裹,因为在JS中 这种标识符是不合法的,而下划线是合法的,比如a_b