RYF javascript笔记3


4. 面向对象编程

4.1 面向对象编程概述

4.1.1 简介

4.1.1.1 构造函数

js没有“类”,而改用构造函数作为对象的模板。

var Vehicle = function() {
    this.price = 1000;
};

构造函数是一个正常函数。但是:它使用new命令调用;函数体内部使用this代表要生成的对象实例。

4.1.1.2 new命令

new命令执行构造函数,返回一个实例对象。

使用new调用构造函数时,传多少个参数都可以,甚至可以可以省略括号。

var Car = function(p) {
    this.price = p;
};
var c1 = new Car(); //price:undefined
var c2 = new Car(1); //price:1
var c3 = new Car(1,2); //price:1
var c2 = new Car; //price:undefined

不使用new调用构造函数时,构造函数就变成了普通函数,并不会生成实例对象。而且由于this这时代表全局对象,将造成一些意想不到的结果。

var c4 = Car();  //c4:undefined

因此,为了保证构造函数必须与new命令一起使用,可以在构造函数内部使用严格模式

var Lorry = function() {
    "use strict" // 严格模式
    this.price = 1;
};
var l = Lorry(); // 无法设置未定义或 null 引用的属性“price”

或者在构造函数内部判断是否使用了new命令。

function Vehicle (p) {
  if (!(this instanceof Vehicle)) {
    return new Vehicle(p);
  } 
  this._p = p;
}
var v1 = new Vehicle(9); // Vehicle里的this指向的是new出来的对象
var v2 = Vehicle(9); // Vehicle里的this指向的是window对象

4.1.1.3 instanceof运算符

instanceof用来确定对象是否是某个构造函数的实例。

在JavaScript之中,所有对象都有对应的构造函数。

[1, 2, 3] instanceof Array // true
({}) instanceof Object // true

但是,由于原始类型的值不是对象,所以不能使用instanceof运算符判断类型。

"" instanceof String // false
1 instanceof Number // false

如果存在继承关系,那么instanceof运算符对这些构造函数都返回true。

var a = [];
a instanceof Array // true
a instanceof Object // true

4.1.2 this关键字

4.1.2.1 涵义

this是指函数当前的运行环境。

难点是:JavaScript支持运行环境动态切换。

function f(){ console.log(this.x); };
var a = {x:'a'};
var b = {x:'b'};
a.m = f;
b.m = f;
a.m() // a
b.m() // b

4.1.2.2 this的使用场合

1、全局环境
在全局环境使用this,它指的就是顶层对象window。

this === window // true 
function f() {
    console.log(this === window); // true
}

2、构造函数
构造函数中的this,指的是实例对象。

3、对象的方法
将对象的方法赋值给另一个对象,会改变this的指向。

var o1 = {m:1};
o1.f = function (){ console.log(this.m);};
o1.f() // 1

var o2 = {m:2};
o2.f = o1.f
o2.f() // 2

o1.f() // 1

将对象内部的方法赋值给一个变量,也可能改变this的指向。

var a = {
        b : {
            m : function() {
                console.log(this.p);
            },
            p : 'Hello'
        }
};
var t = a.b.m;
t(); // undefined
a.b.m(); // Hello

4.1.2.3 使用注意点

1、避免多层this
2、避免数组处理方法中的this
数组的map和foreach方法,允许提供一个函数作为参数。这个函数内部不应该使用this。

var o = {
    v: 'hello',
    p: [ 'a1', 'a2' ],
    f: function f() {
        this.p.forEach(function (item) {
            console.log(item); 
            console.log(this.v);
        });
    }
}
o.f(); // f函数里的item正确,但this.v是undefinded

解决问题的一种方法,是使用中间变量。

var o = {
    v: 'hello',
    p: [ 'a1', 'a2' ],
    f: function f() {
        var that = this;
        this.p.forEach(function (item) {
            console.log(this.v);
        });
    }
}

3、避免回调函数中的this
回调函数中的this往往会改变指向,最好避免使用。

4.1.3 固定this的方法

4.1.3.1 call方法

函数的call方法,可以改变指定该函数内部this的指向,然后再调用该函数。

var n = 123;
var o = { n : 456 };

function a() {
  console.log(this.n);
}

a.call() // 123
a.call(null) // 123
a.call(undefined) // 123
a.call(window) // 123
a.call(o) // 456

如果this所要指向的对象,设定为null或undefined,则等同于指向全局对象。

call方法的完整使用格式如下:

func.call(thisValue, arg1, arg2, ...)

4.1.3.2 apply方法

apply方法的作用与call方法类似,也是改变this指向,然后再调用该函数。它的使用格式如下。

func.apply(thisValue, [arg1, arg2, ...])

apply的第一个参数如果设为null或undefined,则等同于指定全局对象。第二个参数则是一个数组。

function f(x,y){ console.log(x+y); }
f.call(null,1,1) // 2
f.apply(null,[1,1]) // 2
f.apply(null,[1,1,1]) // 2
f.apply(null,[]) // NaN

4.1.3.3 bind方法

bind方法用于将函数体内的this绑定到某个对象,然后返回一个新函数。它的使用格式如下。

func.bind(thisValue, arg1, arg2,...)

4.2 封装

4.2.1 prototype对象

4.2.1.1 构造函数的缺点

构造函数定义的属性和方法是属于具体对象的,无法共享。(当然了)

4.2.1.2 prototype属性的作用

js的每个对象都有一个原型对象(prototype)。

  • 定义在prototype上面的属性和方法,能被所有实例对象共享。
  • 修改prototype对象会影响所有实例对象。
  • 实例优先调用自身的属性或方法,然后才是prototype的属性或方法。

示例:

function Animal () {
}
var cat1 = new Animal();
var cat2 = new Animal();

Animal.prototype.color = "white";
cat1.color // 'white'
cat2.color // 'white'

Animal.prototype.color = "yellow";
cat1.color // 'yellow'
cat2.color // 'yellow'

cat1.color = 'black';
cat1.color // 'black'
cat2.color // 'yellow'

4.2.1.3 原型链

从自身到原型,再到原型的原型,这样就形成了原型链。直至Object.prototype,它的原型是null,因此原型链结束。

function MyArray (){}
MyArray.prototype = new Array();

var mine = new MyArray();
mine.push(1, 2, 3);

mine.length // 3
mine instanceof Array // true

4.2.1.4 constructor属性

prototype对象有一个constructor属性,默认指向prototype对象的构造函数。

function P() {}
P.prototype.constructor === P // true

4.2.2 Object.getPrototypeOf方法

返回对象的原型。

// 空对象的原型是Object.prototype
Object.getPrototypeOf({}) === Object.prototype // true

// 函数的原型是Function.prototype
function f() {}
Object.getPrototypeOf(f) === Function.prototype // true

// 假定F为构造函数,f为F的实例对象
// 那么,f的原型是F.prototype
var f = new F();
Object.getPrototypeOf(f) === F.prototype // true

4.2.3 Object.create方法

Object.create方法用于生成新的对象,可以替代new命令。它接受一个原型对象作为参数,返回一个新对象,后者完全继承前者的属性。

var oldObj = {
    id: 123,
    "name": "Hello"
};
var newObj = Object.create(oldObj)

4.2.4 isPrototypeOf方法

isPrototypeOf方法用来判断一个对象是否是另一个对象的原型。

var o1 = {};
var o2 = Object.create(o1);
var o3 = Object.create(o2);
o2.isPrototypeOf(o3) // true
o1.isPrototypeOf(o3) // true

4.3 继承

4.3.1 proto属性

在Object对象的实例有一个非标准的proto属性,指向该对象的原型对象,即构造函数的prototype属性。

var o = new Object();
o.__proto__ === o.constructor.prototype // true
o.__proto__ === Object.getPrototypeOf(o) // true

4.3.2 属性的继承

属性有两种。一种是对象自身的原生属性,另一种是继承自原型的继承属性。

4.3.2.1 对象的原生属性

对象的原生属性,可以用Object.getOwnPropertyNames获得。

Object.getOwnPropertyNames(Date)
// ["parse", "arguments", "UTC", "caller", "name", "prototype", "now", "length"]

只获原生属性中可枚举的,用Object.keys

Object.keys(Date)// []

判断对象是否具有某个属性,使用hasOwnProperty

Date.hasOwnProperty('length') // true
Date.hasOwnProperty('toString') // false

4.3.2.2 对象的继承属性

用Object.create创建的对象,会继承所有原型对象的属性。

var proto = { p1: 123 };
var o = Object.create(proto);
o.p1 // 123
o.hasOwnProperty("p1") // false

4.3.2.3 获取所有属性

判断一个对象是否具有某个属性(不管是自身的还是继承的),使用in运算符。

"length" in Date // true
"toString" in Date // true

可用for-in循环所有可枚举属性。

var o1 = {p1:123};
var o2 = Object.create(o1,{
    p2: { value: "abc", enumerable: true }
});
for (p in o2) {
    console.log(p);
}
// p2
// p1

4.4 JavaScript模块化编程

4.4.1 使用构造函数封装私有变量

4.4.2 IIFE封装私有变量

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

推荐阅读更多精彩内容

  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 4,556评论 0 5
  • 第5章 引用类型 引用类型的值(对象)是引用类型的一个示例。在ECMAScript 中,引用类型是一种数据结构,用...
    力气强阅读 716评论 0 0
  • 1.在导入高通的AR项目的时候,犹豫只提供了armeabi-v7a没有提供其他处理器so库,到时再64位机器直接闪...
    Lazy1阅读 2,741评论 0 5
  • 什么是APP消息推送? APP运营通过自带的或者第三方的工具,对已安装用户进行消息告知的一种形式。 为什么要做AP...
    Air7777777阅读 1,054评论 1 5
  • 有人说,喜欢回顾往事的人是因为不满当下现实。是啊,人们总觉得从前是最好的,从前的天很蓝,水很清,车马邮件慢,一生只...
    他间阅读 431评论 0 0