面向对象
对象的组成:方法和属性
属性关键词:静止的,状态
方法关键词:动态的,过程,处理,带括号
————————————————————————
js中的面向对象不是其他语言那样的面向对象。
<script type="text/javascript">
//定义arr为一个数组(数组也是一个对象实例,所以arr是个对象实例)
var my_arr=[];
//以前变量这样用,var number=10;
//现在定义属性,arr.number = 10; 就是用“点”即可!
my_arr.number = 10;
//定义对象的方法
my_arr.test = function(){
console.log(this);
//这里this就是这个数组,直接用数组名也可以
//alert(my_arr.number);
//alert(this.number);
}
//这样使用方法
my_arr.test();
</script>
这里需要注意的是,能不能将第6行放在第3行前面?
不可以,没有到第3行赋值,my_arr还是undefined,而undefined是不能有特性的。
1<script type="text/javascript">
2 var a ='ddd';
3 a.kkk = 'eee';
4 console.log(a.kkk); //火狐,打印出undefined ,可见,a不是一个对象类型的数据时,还不可以直接加“点”呢
5 a.ka = function (){
6 console.log('aaaaaaaaaaaaaaaaaa');
7 }
8 a.ka(); //报错,a.ka is not a function 可见,a不是一个对象类型的数据时,还不可以直接加“点”赋予方法呢
9 </script>
1 <script type="text/javascript">
2 var a ='ddd';
3 String.prototype.kkk = 'eee';
4 a.kkk = 'eeeddd';
5 console.log(a.kkk); //火狐,打印出eee ,可见,a不是一个对象类型的数据时,还不可以直接加“点”呢,但是从字符串的原型上可以加
6 String.prototype.ka = function (){
7 console.log('aaaaaaaaaaaaaaaaaa');
8 }
9 a.ka(); //aaaaaaaaaaaaaaaaaa
10 </script>
结果是数组有个number属性和test方法,但是数组内容为空,length为0,但是这样alert(my_arr[‘number’]);也可以弹出10
//TODO 以后再研究数组,但是可见,数组的内容是内容,属性和内容不是一回事,内容是数组里面存了啥,属性是这个数组实例有什么属性。如同,一个是汽车里装了谁谁,一个是汽车有车大灯的感觉。
新建一个纯净的对象实例
1 <script type="text/javascript">
2 var my_obj = new Object(); //新建一个对象实例my_obj,换成 var my_obj = {}; 也行,一个意思
3 my_obj.name = '张三'; //加个属性
4 my_obj.test = function(){ //加个方法
5 this.test2(); //对象的A方法中用到对象的B方法
6 }
7 my_obj.test2 = function(){
8 console.log(this.name);
9 }
10 my_obj.test(); //使用
11 </script>
对象中思考变量作用域的问题
以前学到变量作用域,都是函数和变量,现在加入对象,看看是怎么回事
<script type="text/javascript">
2 console.log(my_obj); // 显示undefined,说明解析器中和其他类型一样,根据var,先放个undefined再说
3 var my_obj = {};
4 console.log(my_obj); // 显示 Object {}
5 my_obj.name = '张三';
6 console.log(my_obj); // 显示Object {name:"张三"}
7 </script>
其实可以这样理解,js解析器,预解析,根据var找到了my_obj,赋值undefined,如果放到不用对象的时候,出个name,需要var name一下,这里由于对象var了,对象下各个目前未定义的属性就相当于已经var但没赋值了,一旦测试都是undefined。第4行,由于还没有赋予name属性,如果查看my_obj.name的话,就是undefined,而第5行赋值,第6行再看,就是“张三”了。
1 <script type="text/javascript">
2 var my_obj = {};
3 my_obj.name = '张三';
4 console.log(my_obj.test); //undefined
5 my_obj.test(); //浏览器提示出错,说test不是一个函数。可见my_obj.test = function(){...} 只是一个赋值,并没有因为是function,就扔到预解析仓库中
6 my_obj.test = function(){
7 console.log(this.name);
8 }
9 </script>
1 <script type="text/javascript">
2 var my_obj = {};
3 my_obj.name = '张三';
4 my_obj.test = xxx;
5 my_obj.test(); //这样写是可以执行的,因为函数xxx扔到预解析仓库了,上一行又赋予test这个函数了,显示张三
6 function xxx(){
7 console.log(this.name);
8 }
9 </script>
<script type="text/javascript">
var my_obj = {};
my_obj.name = '张三';
my_obj.test = function xxx(){
console.log(this.name);
};
my_obj.test(); // 正常运行
</script>
1 <script type="text/javascript">
2 var my_obj = {};
3 my_obj.name = '张三';
4 my_obj.test = function xxx(){
5 console.log(this.name);
6 };
7 xxx(); //浏览器报错,火狐认为,xxx未定义,也就是说,在等号后面的函数不会被预解析,这里不求预解析,连解析都没有
8 </script>
1 <script type="text/javascript">
2 var my_obj = {
3 name : "张三",
4 test : function(){
5 //对象也是个变量作用域,这里面是可以使用对象外面的函数的
6 //console.log(name); 不可以这样用,不能因为name和test平级,就不加this,属性名和变量是不同的,要加上对象名才能用
7 console.log(this.name); //console.log(my_obj.name); 这样也行
8 }
9 };
10 my_obj.test(); //显示张三
11 console.log(my_obj.name); //想使用对象的属性,就要加对象名
12
13 </script>
这里新建对象不够灵活,如果要建的对象名不是张三呢,而是好几个,张三,李四,王二,麻子等很多人,那么就要用到工厂方式
还按照上面的方式就是如下代码
1 <script type="text/javascript">
2 var my_obj1 = {
3 name: "张三",
4 test: function() {
5 console.log(this.name); //如果是console.log(my_obj2.name); 将显示李四,也就是用到了其他对象的属性。
6 }
7 };
8
9 var my_obj2 = {
10 name : "李四",
11 test : function(){
12 console.log(this.name);
13 }
14 }
15 my_obj1.test(); //显示张三
16 my_obj2.test(); //显示李四
17 </script>
js并不是其他语言那样的面向对象,这里的对象,其实是其他语言中 对象实例的意思。
上面代码怎么精简呢,共同部分用一个函数来生成即可,也就是所谓的工厂模式,如下
1 <script type="text/javascript">
2 function CreatePerson(name){ //这个function就相当于一个生产对象的工厂
3 var obj = new Object();
4 obj.name = name;
5 obj.test = function() {
6 console.log(this.name);
7 }
8 return obj;
9 }
10
11 var my_obj1 = CreatePerson('张三'); //开始生产了
12 var my_obj2 = CreatePerson('李四');
13 my_obj1.test();//使用工厂生产出来的对象
14 my_obj2.test();
15 </script>
在js中,用new去调用一个函数的话,这个函数中的this就是创建出来的对象,而且函数的返回值直接就是this 隐式返回
难以理解这句吧,就是js规定的,函数加new, 函数中的this就是对象实例本身,并且函数返回该对象实例(不用你操作了,js帮你搞定)。
1 <script type="text/javascript">
2 function CreatePerson(name){
3 this.name = name;
4 console.log(this); //打印出的this就是张三 这里就相当于js帮你做了 var my_obj1 = new Object(); 最后return my_obj1;
5 }
6 var my_obj1 = new CreatePerson('张三');
7 console.log(my_obj1); //打印出的my_obj1就是张三
8 </script>
这样调用
var my_obj1 = new CreatePerson(‘张三’); 是不是很像 var date = new Date();
1 <script type="text/javascript">
2 function CreatePerson(name){
3 this.name = name;
4 this.test = function(){
5 console.log(this.name);
6 }
7 }
8
9 var my_obj1 = new CreatePerson('张三');
10 my_obj1.test();
11 </script>
但是这样还是有问题,每个对象实例的方法,都是在内存总新建的,用prototype来对相同的对象实例方法进行定义,使之共用内存。改造如下
1 <script type="text/javascript">
2 function CreatePerson(name){
3 this.name = name;
4 }
5
6 CreatePerson.prototype.test = function(){
7 console.log(this.name);
8 }
9
10 var my_obj1 = new CreatePerson('张三');
11 var my_obj2 = new CreatePerson('李四');
12
13 alert(my_obj1.test == my_obj2.test); //true
14 my_obj1.test(); // 张三
15 my_obj2.test(); // 李四
16
17 </script>
这里面的对象这样去记忆理解容易点。 my_obj1和my_obj2是具体的某个人,是对象的实例。而CreatePerson是人这一类,是构造函数。在CSS中,有行内样式
或者写成 这里。用构造函数加上prototype的方式,类似于class,能广泛用于很多个个体对象实例,而直接加在个体上的方法就只能用于个体,同时,二者有重名冲突的时候,个体上的方法优先级较高。 1 <script type="text/javascript">
2 function CreatePerson(name){
3 this.name = name;
4 }
5
6 CreatePerson.prototype.test = function(){
7 console.log(this.name);
8 }
9
10 var my_obj1 = new CreatePerson('张三');
11 var my_obj2 = new CreatePerson('李四');
12
13 //为my_obj1这个个体又单独加个test。prototype定义的test对该个体来说失效
14 my_obj1.test = function(){
15 console.log(this.name+'111');
16 }
17
18 alert(my_obj1.test == my_obj2.test); //false
19 my_obj1.test(); // 张三111
20 my_obj2.test(); // 李四
21
22 </script>
再看看最开始怎么定义my_obj1,直接 var my_obj1 = new Object(); 或者写 var obj = {};这也是js的面向对象,到现在用构造函数的方式,一个是直接定义到个体,一个是先定义一个种类,然后再出个体。
使用原型再举个例子,求数组中元素的和,如下
1 <script type="text/javascript">
2 var arr_1 = [1,9,3,5];
3 Array.prototype.sum = function(){
4 var result = 0;
5 for (var i=0;i<this.length;i++) {
6 result += this[i];
7 }
8 return result;
9 }
10
11 console.log(arr_1.sum());//18
12 </script>
原型要加在构造函数上
1 <script type="text/javascript">
2 var arr_1 = [1,9,3,5];
3 var arr_2 = [3,5,7];
4 Array.prototype.sum = function(){
5 var result = 0;
6 for (var i=0;i<this.length;i++) {
7 result += this[i];
8 }
9 return result;
10 }
11 arr_1.sum = function(){
12 return '我要覆盖prototype的方法!';
13 }
14
15 console.log(arr_1.sum());//我要覆盖prototype的方法!
16 console.log(arr_2.sum());//15
17 </script>
js是基于原型的程序
每个基本类型数据都有一个原型,字符串 String 数字 Number 布尔值 Boolean 数组 Array
尽量不要去修改系统对象下面的方法和属性。
例子:重写数组的push方法
1 <script type="text/javascript">
2 var arr = [1,2,3];
3 Array.prototype.push = function(){//这里系统自带的push就被覆盖了
4 for(var i=0; i<arguments.length;i++){
5 this[this.length] = arguments[i];
6 }
7 return this.length;
8 }
9
10 arr.push(9,8,7,6,5);
11 console.log(arr);
12 </script>
包装对象:基本类型(除了null 和 undefined)都有自己对应的包装对象,如String Number Boolean,这里也就知道了,null空对象和undefined未定义是不能再添加属性和方法的
1 <script type="text/javascript">
2 var str1 = 'hello123';
3 var str2 = new String('hello123'); //通过new创建的都是对象
4
5 console.log(typeof str1); //string
6 console.log(typeof str2); //object
7 str2.lastvalue = function() {
8 return this.charAt(this.length - 1);
9 }
10 console.log(str2.lastvalue());//最后一个字符3
11
12 str1.lastvalue = function() { //报错,因为str1是字符串,不是对象,不能添加方法,只能使用其包装对象给予的方法
13 return this.charAt(this.length - 1);
14 }
15 console.log(str1.lastvalue());
16 </script>
如果str1这个字符串类型数据,要添加方法,那可以添加到它的包装对象上。如下
1 <script type="text/javascript">
2 var str1 = 'hello123';
3 String.prototype.lastvalue = function(){
4 return this.charAt(this.length-1);
5 }
6 console.log(str1.lastvalue());
7 </script>
原型链:个体实例和原型之间的链接 , 原型链的最外层是Object.prototype
举例:个体人物:张三 / 原型:男人
张三具有属性:姓名:张三;
1 <script type="text/javascript">
2 function Man(name){
3 this.name = name;
4 }
5 Man.prototype.name = '男人';
6 Object.prototype.name = '人类';
7
8 var Zhanshan = new Man('张三');
9 console.log(Zhanshan.name); //显示张三,如果没有第3行代码,则根据原型链,显示为男人,如果连第5行也没有,则显示为人类
10 //console.log(Man.prototype.name); //显示男人,若无第5行,则显示为人类
11 </script>
也就是注意层级关系。没有找到的属性就往原型链上找,优先级当然是内层优先。
面向对象的一些属性和方法
hasOwnProperty : 对象个体实例自身是不是有某个属性?如果是个体自身的属性返回true; 如果属性是原型下的,返回false;
1 <script type="text/javascript">
2 function Man(name){
3 this.name = name;
4 }
5 Man.prototype.name = '男人';
6
7 var Zhanshan = new Man('张三');
8 console.log(Zhanshan.hasOwnProperty('name')); //true
9 </script>
1 <script type="text/javascript">
2 function Man(name){
3 //this.name = name;
4 }
5 Man.prototype.name = '男人';
6
7 var Zhanshan = new Man('张三');
8 console.log(Zhanshan.hasOwnProperty('name')); //false
9 </script>
<script type="text/javascript">
function Man(name){
//this.name = name;
}
Man.prototype.name = '男人';
var Zhanshan = new Man('张三');
Zhanshan.name = '张三三';
console.log(Zhanshan.hasOwnProperty('name')); //true
</script>
<script type="text/javascript">
function Man(){
this.name = '张三';
}
Man.prototype.name = '男人';
var Zhanshan = new Man('张三');
console.log(Zhanshan.hasOwnProperty('name')); //true 虽然新建个其他的个体实例,name也会是张三,但是该个体实例自身有name属性,所以还是true
</script>
constructor : 查看对象的构造函数
1 <script type="text/javascript">
2 function Man(name){
3 this.name = '张三';
4 }
5 Man.prototype.name = '男人';
6
7 var Zhanshan = new Man('张三');
8 console.log(Zhanshan.constructor == Man); //true
9 </script>
1 <script type="text/javascript">
2 var a1 = [1,2,3];
3 console.log(a1.constructor = Array); //true
4 </script>
当我们写一个函数时,程序自动会生成constructor
1 <script type="text/javascript">
2 function Aaa(){
3 }
4 //注意,下面这句话是系统内部自动实现的,这里写出来,好看到系统做了什么,只要定义一个函数,系统都会"自动生成"下面一句
5 //Aaa.prototype.constructor = Aaa; //这里可以自行更改,将覆盖,如Aaa.prototype.constructor = Array;但是乱改后将影响使用
6 console.log(Aaa.prototype);
7 //只有constructor是函数原型自身的,hasOwnProperty这个方法是Object上的,是通过原型链找到的
8 </script>
现在知道不要更改constructor,但是有时候,不注意我们就更改了,如下:
1 <script type="text/javascript">
2 function Aaa(){
3 }
4 Aaa.prototype.name = '张三';
5 Aaa.prototype.age = 30;
6
7 var a1 = new Aaa();
8 console.log(a1.constructor); //Aaa
9 </script>
上面我们知道,Aaa.prototype也是一个对象,上面的改写为如下形式,就更改了constructor
1 <script type="text/javascript">
2 function Aaa(){
3 }
4 //这里更改了,不是更改属性,而是重新对Aaa.prototype进行了赋值,也就弄丢了constructor属性
5 Aaa.prototype = {`在这里插入代码片`
6 name:"张三",
7 age :30
8 }
9 var a1 = new Aaa();
10 console.log(a1.constructor); //Object
11
12 </script>
所以,当我们按照上面这张方法写的时候,要加上修正的,变成如下
1 <script type="text/javascript">
2 function Aaa(){
3 }
4 //弄丢的constructor属性再找回来
5 Aaa.prototype = {
6 constructor:Aaa, //注意不要加引号,注意这句话就是为了修正对象constructor属性
7 name:"张三",
8 age :30
9 }
10 var a1 = new Aaa();
11 console.log(a1.constructor); //Aaa
12 </script>
还有一个地方注意:用for in循环对象的属性的时候,系统自带的属性是循环不到的
1 <script type="text/javascript">
2 function Aaa(){
3 }
4 Aaa.prototype.name = '张三';
5 Aaa.prototype.age = 30;
6 Aaa.prototype.constructor = Aaa;
7 var a1 = new Aaa();
8 for (var attr in Aaa.prototype) {
9 console.log(attr); //显示了name和age,但是不显示constructor,即使还单独写一次
10 }
11 </script>
但是,如果用= 重新赋值的方式,再循环打印是可以打印出来的哦!如下
1 <script type="text/javascript">
2 function Aaa(){
3 }
4 //这种方式是重新赋值
5 Aaa.prototype = {
6 constructor:Aaa, //注意不要加引号,注意这句话就是为了修正对象constructor属性
7 name:"张三",
8 age :30
9 }
10 var a1 = new Aaa();
11 for (var attr in Aaa.prototype) {
12 console.log(attr); //显示constructor name age
13 }
14 </script>
instanceof 运算符,判断个体对象和原型函数 是否有原型链上的关系,返回true或false 如下
1 <script type="text/javascript">
2 function Man(name){
3 this.name = name;
4 }
5 function Woman(name){
6 this.name = name;
7 }
8 var zhanshan = new Man();
9 console.log(zhanshan instanceof Man); //true
10 console.log(zhanshan instanceof Object); //true
11 console.log(zhanshan instanceof Woman); //false
12 </script>
用instanceof判断某变量是否是数组
1 <script type="text/javascript">
2 var a1 = [1,2,3,4,5];
3 console.log(a1 instanceof Array); //true;
4
5 var b1 = 'hi';
6 console.log(b1 instanceof Array); //false;
7 </script>
toString 方法 把对象转成字符串 。 系统对象下面都是自带的,而自己创建的对象toString在object上
如下
1 <script type="text/javascript">
2 var a1 = [1,2,3,4,5];
3 console.log(a1.toString == Object.prototype.toString); //false 系统对象下面都是自带的,不是Object上的;
4 console.log(a1.toString == Array.prototype.toString); //true 系统对象下面都是自带的,在Array上 不在Object
5 function Bbb(){
6 }
7 var b1 = new Bbb();
8 console.log(b1.toString == Object.prototype.toString); //true 自己创建的对象toString在object上
9 </script>
1 <script type="text/javascript">
2 var a1 = [1,2,3,4,5];
3 console.log(a1.toString()); //1,2,3,4,5
4 console.log(typeof a1.toString()); //string
5 </script>
1 <script type="text/javascript">
2 var a1 = [1,2,3,4,5];
3 console.log(a1.toString()); //1,2,3,4,5
4 console.log(typeof a1.toString()); //string
5
6 //按照系统toString生成的1,2,3,4,5 如果不符合我们使用的格式的话,我们还可以自己去改写
7 Array.prototype.toString = function(){
8 return this.join('+'); //将数组内容按照加号连接
9 }
10 console.log(a1.toString()); //1+2+3+4+5
11 </script>
复
制代码
1 <script type="text/javascript">
2 var num = 255;
3 console.log(num.toString(16)); //ff, 参数16是16进制的意思,输出16进制的数据 rgb色彩可以用~~
4 console.log(num.toString(2)); // 11111111 参数2是2进制的意思
5 </script>
上面是利用toString转换数字进制
toString还可以用来判断变量的数据类型,比typeof instanceof 更精准,如下
1 <script type="text/javascript">
2 var num = 255;
3 console.log(Object.prototype.toString.call(num)); // [object Number]
4 console.log(Object.prototype.toString.call(num) == '[object Number]'); // true;
5
6 var str = 'hello';
7 console.log(Object.prototype.toString.call(str)); // [object String]
8 var arr = [1,2,5];
9 console.log(Object.prototype.toString.call(arr)); // [object Array]
10
11 console.log(Object.prototype.toString.call({name:'张三'})); //[object Object]
12
13 </script>
对象的继承 继承要求原类不变,新出子类
如何做?
1 <script type="text/javascript">
2 function CreateUser(name,age){
3 this.name = name;
4 this.age = age;
5 }
6
7 CreateUser.prototype.dosometing = function(){
8 console.log(this.name+'dosometing');
9 }
10
11 var Zhanshan = new CreateUser('张三',39);
12 Zhanshan.dosometing();
13
14 function CreateStar(name,age,job){
15 CreateUser.call(this,name,age); //这句话完成了属性的继承
16 this.job = job;//这句话添加父级没有的新属性
17 }
18
19 CreateStar.prototype = CreateUser.prototype; //这句话完成了方法的继承,但是根据对象赋值可知,这里,子类对象的变化,父类对象亦变。不是完美的方式。
20
21
22 var Lichen = new CreateStar('李晨',40,'演员');
23 console.log(Lichen.name + '是个' + Lichen.job);
24 Lichen.dosometing();
25
26 console.log(Lichen.constructor); //CreateUser 因为直接CreateStar.prototype = CreateUser.prototype; 所以这里不是CreateStar
27
28 console.log(CreateStar.prototype.constructor); //CreateUser
29 console.log(CreateUser.prototype.constructor); //CreateUser
30 console.log(CreateUser.prototype.constructor == CreateStar.prototype.constructor); //true;
31
32 </script>