JS 面向对象

数据类型与对象

对于JS中的数据类型,其本质都是基于各自的对象类进行创建的,因此如果我们可以通过修改这个对象类的原型来对某个数据类型整体都进行修改

数据类型对象

Number

数字,例如我们给数字类型都添加一个方法判断是否为正数:

Number.prototype.isPos = function() {
  return this > 0;
}
let b = 1;
b.isPos();
// true
Boolean
String
RegExp
Array
Function
参考

https://www.runoob.com/jsref/jsref-obj-array.html

对象API

Object.create

创建对象,第一个参数指定对象原型,第二个参数为对象本身,举例:

a = {x:1, y:2}
b = Object.create(a, {})
b.__proto__ === a
// true
Object.setPrototypeOf

设置原型,举例:

let a = {};
let b = {};
console.log(a.__proto__ === Object.prototype)  // true
Object.setPrototypeOf(a, b);
// 将a的原型指向b
console.log(a.__proto__ === Object.prototype)  // false
console.log(a.__proto__ === b)  // true
Object.getPrototypeOf

查看原型,举例:

Object.getPrototypeOf({});
// {}
Object.getOwnPropertyDescriptor

获取对象当中指定属性的特征,举例:

let a = {x:1};
console.log(Object.getOwnPropertyDescriptor(a, "x"));
// { value: 1, writable: true, enumerable: true, configurable: true }
// 几个属性分别表示:值、是否可修改、是否可遍历、是否可以修改属性特征
Object.defineProperty

定义、修改属性特征,举例:

let a = {};
Object.defineProperty(a, "x", {
    value: 1,
    writable: true,
    enumerable: false
})
a.x = 2;
console.log(a.x)
// 2
console.log(Object.keys(a));
// []

可以看出属性x可修改,但不可遍历
注:
如果定义的属性设置了访问器,则不能设置valuewritable属性,举例:

let o = {};
Object.defineProperty(o, "a", {
    enumerable: true,
    configurable: true,
    // 如果设置value或writable属性会报错
    set(val) {
      this._a = val;
      console.log(val);
    },
    get() {
      return this._a;
    }
});
console.log(o.a);
o.a = 100;
console.log(o.a);

// undefined
// 100
// 100
Object.preventExtensions

不允许对象添加属性,举例:

let a = {};
a.x = 1
Object.preventExtensions(a);
a.y = 2;
console.log(Object.keys(a));
// [ 'x' ]

可以看出设置不允许添加属性以后,属性y就没有被添加进去了
注:
可以通过Object.isExtensible查看对象是否允许添加属性

Object.seal

封闭属性,不允许添加、修改、重新定义属性(configurable变成false)
注:
可以通过Object.isSealed查看对象是否冻结

Object.freeze

对象冻结,无法改变、添加、删除对象内容(writable、configurable都变成false)
注:
可以通过Object.isFrozen查看对象是否冻结

参考:https://www.cnblogs.com/amiezhang/p/11312973.html

Object.keys()/Object.values()

访问对象下所有可迭代的key/value,举例:

a = {x:1, y:2}
Object.keys(a)
// ["x", "y"]
Object.is

判断对象是否为某个类型,举例:

Object.is("aaa", NaN)

注:
NaN类型需要用Number.isNaN或者Object.is来判断

Object.assign

将后面几个对象的内容合并到第一个对象里(重复属性名,后面的会覆盖前面的),举例:

console.log(Object.assign({x:1}, {y:2}))
// { x: 1, y: 2 }

我们可以用该方法实现浅拷贝,举例:

let a = {x:1};
let b = a;
let c = Object.assign({}, a);
// 将a合并到空对象里,从而实现浅拷贝
a.x = 2;
console.log(b.x, c.x);
// 2 1

可以看到此时c和a并不是同一个对象

对象操作

变量命名key

如果希望key的值是由变量决定,则可以通过[变量]来实现,举例:

a = "aaa";
b = {
    [a]: 1
}
console.log(b)
遍历可迭代对象
遍历key
a = {x:1, y:2};
for(let k in a) {
  console.log(k);
}

// x
// y

遍历数组时,获取的则是下标,举例:

a = ["a","b","c","d"];
for (let k in a){
  console.log(k);
}
// 0
// 1
// 2
// 3
遍历value
a = ["a","b","c","d"];
for (let v of a){
  console.log(v);
}
// a
// b
// c
// d
instanceof

判断是否是某个类实例(本质是看是否prototype是否在其原型链上),举例:

[] instanceof Array
hasOwnProperty

当前对象是否含有某个属性(不包括原型链上的),举例:

let a = {x:1};
a.__proto__.y = 1;
console.log(a.hasOwnProperty("x"))  // true
console.log(a.hasOwnProperty("y"))  // false
attr in o

当前对象是否含有某个属性(包括原型链),举例:

let a = {x:1};
a.__proto__.y = 1;
console.log("x" in  a)  // true
console.log("y" in  a)  // true
访问器

在对象中当要访问某个属性时,可以进行自定义操作,举例:

let a = {
    data: {
        x: 1
    },
    // 当设置x值的时候触发
    set x(val) {
        console.log("将x设置为:", val);
        this.data.x = val;
    },
    // 当获取x值的时候触发
    get x() {
        console.log("获取x的值");
        return this.data.x;
    }
}

a.x = 100;
console.log(a.x);
// 将x设置为: 100
// 获取x的值
// 100

定义

ES6提供了类相关的关键字用于创建里,主要有:class(定义类)、constructor(构造器)、extends(继承)、super(父类),static(定义静态属性/方法)举例:

class User {
    constructor(name, pwd) {
        // 构造方法
        this.name = name;
        this.pwd = pwd;
    }
    showName() {
        // 直接输入方法名定义方法
        console.log(this.name);
    }
    static showRule() {
        // 定义静态方法,可以从类直接调用而无需实例化
        console.log("this is User rule");
    }
    static rule = "this is rule";
    // 定义静态属性,如果报错可能是浏览器版本不支持,建议换成最新chrome尝试
}
class Vip extends User {
    // 继承父类
    constructor(name, pwd, level) {
        super(name, pwd);
        // 调用父类构造方法
        this.level = level;
    }
    showPwd() {
        console.log(this.pwd);
    }
    showLevel() {
        console.log(this.level);
    }
}
console.log(User.rule);
User.showRule();
// 直接从类调用静态方法
let user = new Vip("aaa", "111", "1");
user.showName();
user.showPwd();
user.showLevel();
  • 也可以通过直接在类中通过赋值来定义属性成员,举例:
class A {
    // 给实例设置私有成员,等价于在constructor中执行this.x = 1
    x = 1
    y = {a: 1}
    // 设置静态属性(@babel/plugin-proposal-class-properties里设置了关于这种语法的处理)
    static z = {a: 1}
}

let a = new A();
a.y.a = 2
A.z.a = 2
let a2 = new A();
console.log(a2.y.a);
console.log(A.z.a);
私有属性

前面加#即可,私有方法可以通过定义私有属性的值为函数即可,举例:

class A {
  #a = () => {};
}

深拷贝/浅拷贝

浅拷贝示例
a = {x:1};
// 实现浅拷贝a
// 第一种
b = { x: a.x };

// 第二种
b = {};
for (let key in a) {
  b[key] = a[key];
}

// 第三种
b = { ...a };

// 第四种
b = Object.assign({}, a);
深拷贝示例
  • 通过JSON序列化:
let o = {x:1};
let b = o
b === o
// true
let json = JSON.stringify(o)
let deepCopyO = JSON.parse(json)
deepCopyO === o
// false
  • JSON序列化只能操作基本类型,对于非基本类型,可以通过递归浅拷贝来实现
  • 实际开发中建议使用lodash相关库来进行深拷贝操作

lodash参考:
https://www.npmjs.com/package/lodash.clonedeep
https://github.com/lodash/lodash/tree/4.5.0-npm-packages/lodash.clonedeep

反射

参考:http://tech.dianwoda.com/2017/08/09/jsde-fan-she-xue-xi-he-ying-yong/

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

推荐阅读更多精彩内容