继承:
Call:
var lipi={
name:"李皮",
play:function(foo){
console.log(this.name+"正在玩"+foo);
}
}
lipi.play("和泥")
liujiaming想拿到lipi里面的play方法
var liujiaming = {
name:"刘嘉明"
}
lipi.play.call(liujiaming,"回旋踢");
call第一个参数改变this的指向,第二个参数是play这个方法要传递的值,可以多个,用逗号隔开,比如,lipi.play.call(liujiaming,"回旋踢",3,4);但是如果方法里面没传两个形参play:function(foo,a,b)的话3,4是不会输出,也不会报错,但是如果形参传了实参没传就是undefined
案例1
var aLi = document.getElementById("list").getElementsByTagName("li");
console.log(aLi instanceof Array) //fasle,用这种方法获取的数组是伪数组
var aLi = document.getElementById("list").getElementsByTagName("li");
var arr = [].slice.call(aLi) //true,借用真数组arr的方法使aLi变为真数组
console.log(arr instanceof Array)
案例2
function fn(){}
var a = new fn();
console.log(typeof a)//object typeof只能判断基本数据类型,当判断复杂数据类型是typeof只会判断为object
1 function fn(){}
console.log(Object.prototype.toString.call(fn))//[object Function]
2 var obj ={};
console.log(Object.prototype.toString.call(obj))//[object object]
后面为数据类型,使用这个方法就可以判断复杂数据类型了
老师笔记:
Object.prototype.toString.call():检测一个复杂数据类型是一个什么样的类型
instanceof:判断一个对象是不是另一个对象创建出来的
typeof:只能判断基本数据类型 引用数据类型统计返回OBject
apply:(同样具有call的用法,但它的第二个参数可以传数组)
data:image/s3,"s3://crabby-images/46be8/46be8709005ab92548a9ec2d54ffc4312dced664" alt=""
Math.max(取出最大值)这个方法没办法放数组
所以。。。
var arr = [10,20,30,40]
var arr1 = [];
console.log(Math.max.apply(arr1,arr));//40
先创建arr1再让this指向arr1跟直接传 [ ]是一样的,但是var这样占用内存空间
因此:
var arr = [10,20,30,40]
console.log(Math.max.apply([],arr));//40
max是Math的方法,数组通过apply跟Math借,第一个依然是改变this的指向,所以传this指向一个数组,第二个传判断哪个数组的最大值
---------------------------------------------------------------------------------------------
1、属性继承:
data:image/s3,"s3://crabby-images/1bc9f/1bc9f599cb6a7b63c85634dd5db8eed159903695" alt=""
data:image/s3,"s3://crabby-images/f3ed4/f3ed4469c689b61c7395babf4090c5a4cee1848e" alt=""
但是lipi并没有继承到Person里面的属性
data:image/s3,"s3://crabby-images/3c923/3c9239be6d06b730b526bcb81ebbf2292d724aa4" alt=""
使用call跟Person借属性,第一个参数改变this指向,指向Man,后面传要借的属性
data:image/s3,"s3://crabby-images/682ae/682ae9df1ac3f42624e8969838e54e015c63e31e" alt=""
这时候call已经成功继承到Person的属性和方法了
但是一般为了节省内存空间,都会把方法放在prototype上面
data:image/s3,"s3://crabby-images/66eb8/66eb8be06a637e4d590761c9b6659b7905a53623" alt=""
data:image/s3,"s3://crabby-images/97234/9723496d3274b9e9247c80bc92e7cf76d172d9fe" alt=""
但是放了之后却继承不到方法了,所以call只能继承属性,不能继承方法,想继承的话又会耗费内存,所以一般只用call和apply来做属性继承
笔记:call和apply一般情况下我们都用来做属性继承
---------------------------------------------------------------------------------------------
prototype__proto__constructor:原型链
笔记(背会)
prototype:每一个函数里面都有一个prototype ,这个属性叫做原型,这个原型指向一个对象 我们把这个对象叫做原型对象
prototype原型对象里面有2个东西
1、constructor:构造器--作用指向创建自己的那个构造函数
2、__proto__:1 每一个对象里面都会有一个__proto__这个属性
2 __proto__指向了一个对象 这个对象就是原型对象
3 实例化对象可以直接访问__proto__里面的一些方法
原型链:由__proto__组成的链条就叫做原型链
data:image/s3,"s3://crabby-images/3f130/3f130b59fa3203d1ff9151d445ca17641ce3a32e" alt=""
data:image/s3,"s3://crabby-images/bc933/bc9332bc6702460e69d38ccbee73539017ae7ca8" alt=""
---------------------------------------------------------------------------------------------
实例化的过程就是创建对象的过程,new的过程就是创建对象的过程
data:image/s3,"s3://crabby-images/17dc0/17dc0a2ef5b109ec794ff17190a883006753d20d" alt=""
data:image/s3,"s3://crabby-images/50e0c/50e0cfaaccded22ffd2bd45d08dfd55d391ab9cc" alt=""
方法在_proto_里面
data:image/s3,"s3://crabby-images/784d8/784d8080f2ab8ad64a41d609b990295adf4e391d" alt=""
//结果为true
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/4ec04/4ec04a3c60aaea5a923b1338f6e90cde12086fca" alt=""
本来访问实例化对象访问__proto__里面的方法需要console.log(p1.__proto__.eat)
但却直接console.log(p1.eat)就可以直接访问,所以说 实例化对象可以直接访问__proto__里面的一些方法
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/7d531/7d531deb43cdb00fa661c34b78a12348bf3b0348" alt=""
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/8db21/8db21e9a44ce99460b6d72b9cb32585c4a02f806" alt=""
原型链:
可以通过 console.log(p1.__proto__.__proto__.toString)
访问父级Person的方法,p1里没有tostring这个方法,是使用链条访问到父级的方法(p1的父级是Person,Person的父级是object)
---------------------------------------------------------------------------------------------
2、原型继承(极度简单,但也极度不推荐使用,因为会污染父级)
data:image/s3,"s3://crabby-images/1f55c/1f55ce404fc3a862b53d238e63a00d46dff457f4" alt=""
方法work要加在原型上面
data:image/s3,"s3://crabby-images/572e9/572e967c2901df822c589ee512f28f68ea8ea63c" alt=""
属性都通过call继承到了,但方法却只继承到了work一个,因为他本来就放在Man上面,但是它也要继承到person的方法
data:image/s3,"s3://crabby-images/0116f/0116fa644e5b9fada80a8e441ab774af71f2b18f" alt=""
data:image/s3,"s3://crabby-images/45b10/45b100b33a180066876d86bf671881e73200b207" alt=""
这时候所有属性和方法就都继承到了
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/196d5/196d50a172b63008ba494ed716fe529cff8dd59b" alt=""
输出父级
data:image/s3,"s3://crabby-images/89103/8910339cce6f10fa056efb1e6dc6c5bd889a25bf" alt=""
发现Man的work方法也加到了父级上面,这就是污染父级
---------------------------------------------------------------------------------------------
3、原型拷贝 (不能直接继承父级以上的东西,只能一层层嵌套)
data:image/s3,"s3://crabby-images/ab656/ab656faa3f9e876944439c3ea4f3c48620dcf8f5" alt=""
那既然会污染父级,我们就把Man.prototype = Person.prototype;改为下面的拷贝继承
data:image/s3,"s3://crabby-images/9e6d2/9e6d25486e94cf71278d854a4566ba69a878c667" alt=""
复制父级的方法,这样就父级的方法就不会发生变化,输出p1和父级看一下
data:image/s3,"s3://crabby-images/4800c/4800cc71fa545e245fcb12e0bfcfc176cba1cdb5" alt=""
子级继承到父级Person的方法了,父级的方法也没有被子级污染到,貌似看起来没什么问题.
但是不能继承父级以上的东西(也就是Person的以上就不能继承了)
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/32d43/32d43dcaabd4ac78cda68cf8941cd077d1cba7e2" alt=""
data:image/s3,"s3://crabby-images/3521b/3521be437fbc72e0a58346a4320a285110c10de1" alt=""
依然保留Man拷贝Person的循环,再加上shengwu这个方法,也加一个拷贝的循环,一层层嵌套,依然输出p1和person
data:image/s3,"s3://crabby-images/26f80/26f8017079d9ca2642dad19e8be86d2cebd0dbea" alt=""
这时候Person继承到了父级shengwu的increase的方法,P1继承也到了父级的父级的方法
但这样需要多层嵌套,一层一层的获取,是原型拷贝的唯一缺点
---------------------------------------------------------------------------------------------
4 、原型链继承 (多了无用的属性,并且constructor构造器没了,原型对象的指向也发生改变)
原型链:由__proto__组成的链条就叫做原型链
data:image/s3,"s3://crabby-images/321ce/321ce16faaf5c243fecef367ea422e7d36651768" alt=""
因为prototype里面的东西跟实例化对象里面的__proto__的东西是一样的,
所以我们让子级Man的prototype等于父级的实例化里面的东西(p1.eat=p1.___proto___.eat,所以直接省略___proto___就可以访问了)
data:image/s3,"s3://crabby-images/aa8c9/aa8c9f2e1cda0ade68697509c3450e00ae17301f" alt=""
因为是原型链继承 ,所以是形成链条一样的,但这时候确实是访问到了父级的方法,但是多了无用的属性,并且constructor构造器没了,原型对象的指向也发生改变
data:image/s3,"s3://crabby-images/9cd5d/9cd5d50d742ddb2238f0a4d8c0eb2f1330a86b24" alt=""
console.log( [Person] )这时候父级也没有被污染
原型链继承的缺点:丢失构造器,多了一些无用的属性,原型对象的指向也发生改变
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/3f089/3f089c91a9665816bafaf09ca9eb85a2cec7992a" alt=""
原型链链条
data:image/s3,"s3://crabby-images/451f9/451f9b3aee791aa045a000b226e3fcb3c88e4b35" alt=""
object再上面就是null,所以说null是原型链的顶端
---------------------------------------------------------------------------------------------
5 、混合继承(完美继承方式)
data:image/s3,"s3://crabby-images/8dbda/8dbda9649ea776e89c89b85485590737662708da" alt=""
person.prototype里面的东西本身就跟new Person 的__proto__里面的东西一样,所以跟上面的原型链继承差不多,只不过自己加个constructor构造器上去
data:image/s3,"s3://crabby-images/ce4d6/ce4d662d1e5d0e5f8450057c16417cabb7612845" alt=""
父级并没有被污染
data:image/s3,"s3://crabby-images/ba570/ba57092e6684f408448d76771ab10c3414cf2e9b" alt=""
constructor构造器有了,父级的方法也继承到了,所以说混合继承是完美的继承方式
---------------------------------------------------------------------------------------------
6 、寄生继承 (没有混合继承的方法代码少)
data:image/s3,"s3://crabby-images/9997f/9997f46e2e27c3b813c360950ded8aca1b6f3140" alt=""
data:image/s3,"s3://crabby-images/cffbf/cffbf8761d2b4b0c23e9030b769aa66001407f46" alt=""
所以又要再给它加上构造器
data:image/s3,"s3://crabby-images/44e88/44e885d4d570634d16151ff6db3ebc92d5517d30" alt=""
data:image/s3,"s3://crabby-images/876fa/876fa3097a87bcc92ad32afa8e7d1ace8c922c06" alt=""
这时候就有了,所以这种方法也是好,但是没有混合继承的方法代码少,所有混合继承的方式最好
总结:
<script>
(模板例子)
function Person(name,age,sex){
this.name = name;
this.age = age;
this.sex = sex;
}
Person.prototype.eat = function(){}
Person.prototype.sleep = function(){}
function Man(y,name,age,sex){
this.y = y;
Person.call(this,name,age,sex)
}
-----------------------------------
//继承语句
Man.prototype = {
constructor:Man,
__proto__:Person.prototype
}
-----------------------------------
Man.prototype.work = function(){} //写在继承语句下面,不然添加不到Man上
var p1 = new Man();
console.log(p1);
console.log([Person])
-----------------------------
继承主要语句:
//混合继承(完美继承方式)
Man.prototype = {
constructor:Man, //添加少了的构造器
__proto__:Person.prototype
}
//寄生继承
function fn(){}; //寄生壳
fn.prototype = Person.prototype;
Man.prototype = new fn();
//原型拷贝(不能拷贝父级的父级)
for(var key in Person.prototype){
Man.prototype[key] = Person.prototype[key];
}
//原型链继承(多了无用东西,少了构造器)
Man.prototype = new Person();
//原型继承(污染父级,不推荐)
Man.prototype = Person.prototype;
//属性继承(只能继承属性)
Person.call(this,name,age,sex)
Person.apply(this,name,age,sex)
</script>
闭包:
闭包:闭包就是能够读取其他函数内部的变量的函数
全局变量:任何范围之内都能访问
局部变量:只能在当前作用域内访问
好处:1 可以让局部的变量在全局进行访问
2 保留i的值
坏处:1 占用内存空间
2 在IE浏览器下可能会造成内存溢出
3 所以不要滥用闭包,如果用完记得销毁
data:image/s3,"s3://crabby-images/43228/43228c07816250f1683af5b430f75a0a9e238849" alt=""
通过return接受值
data:image/s3,"s3://crabby-images/0d39b/0d39bb36e66dc7a76426310ee1616d5e453e78e0" alt=""
正常情况下每次调用后都会被销毁,但在闭包里面就不会
垃圾回收机制:
当函数运行完毕以后如果内部的变量或者函数没有在全局挂载的话,
那么这个函数就会被销毁掉
如果第二次执行这个函数的时候里面的代码就会重新执行(所以就是因为这样才每次调用都会销毁重置让a=10,所以外面输出的结果都是11)
data:image/s3,"s3://crabby-images/ae8cf/ae8cf6053e5f9bd9f058d44c0f9811387dfa59d0" alt=""
a已经被return出去变成全局的了,所以在外面调用两次都会进行++ //11 //12
面试题:
data:image/s3,"s3://crabby-images/56e18/56e18198e8e78541d396500d72ec09cdc645ad37" alt=""
同上,也就是把局部的变量挂载到全局上面,所以全局可以访问到进行累加
data:image/s3,"s3://crabby-images/6584b/6584bb4af78f6a5e11f3a9912282136f87fa5ae7" alt=""
挂载在全局后如果不销毁的话会一直在里面进行++,不会销毁重置,所以值才能在全局一直累加,会占用内存,所以不要滥用闭包,如果用完记得销毁
data:image/s3,"s3://crabby-images/26d26/26d267f0608a4a777767380ea702bb5637ac0933" alt=""
让b=null;就是销毁了,再调用就是没有了
---------------------------------------------------------------------------------------------
data:image/s3,"s3://crabby-images/b2622/b2622c97c51852b54fb43d96020dbc94f5b6d557" alt=""
这时候i一下子就等于4了
data:image/s3,"s3://crabby-images/27a5c/27a5c52fa9c8c69f918ae7588f5053fec5020262" alt=""
放在立即执行函数里面,就形成闭包,这样子就可以保留i的值了