JS之对象
2021-02-04 14:18
标签:数组 很多 保存 区分 document 一个 false 可见 ber 面向对象:可以创建自定义的类型,很好的支持继承和多态。 面向对象的特征:封装、继承、多态。 在 JavaScript 中,对象是一组无序的相关属性和方法的集合。 对象的作用是:封装信息。比如Student类里可以封装学生的姓名、年龄、成绩等。 对象具有特征(属性)和行为(方法)。 保存一个值时,可以使用变量,保存多个值(一组值)时,可以使用数组。 比如,如果要保存一个人的信息,通过数组的方式可以这样保存: ```javascript 上面这种表达方式比较乱。而如果用JS中的对象表达,结构会更清晰。如下: ```javascript person.name = ‘王二‘; 由此可见,对象里面的属性均是**键值对**: - 键:相当于属性名。 - 值:相当于属性值,可以是任意类型的值(数字类型、字符串类型、布尔类型,函数类型等)。 两条补充 补充1:对象的属性值可以是任何的数据类型,也可以是个**函数**:(也称之为方法) ```javascript console.log(obj.sayName); //没加括号,就是获取方法 ``` 补充2:对象中的属性值,也可以是一个对象。 举例: ```javascript //创建对象 obj2 //将整个 obj2 对象,设置为 obj1 的属性 console.log(obj1.test.name); 打印结果为:smyhvae - **基本数据类型(值类型)**:String 字符串、Number 数值、Boolean 布尔值、Null 空值、Undefined 未定义。 - **引用数据类型(引用类型)**:Object 对象。 **基本数据类型**: 基本数据类型的值直接保存在**栈内存**中,值与值之间是独立存在,修改一个变量不会影响其他的变量。 **对象**: 1)只要不是那五种基本数据类型,就全都是对象。 2)如果使用基本数据类型的数据,我们所创建的变量都是独立的,不能成为一个整体。 3)对象属于一种复合的数据类型,在对象中可以保存多个不同数据类型的属性。 4)对象是保存到**堆内存**中的,每创建一个新的对象,就会在堆内存中开辟出一个新的空间。变量保存的是对象的内存地址(对象的引用)。换而言之,对象的值是保存在**堆内存**中的,而对象的引用(即变量)是保存在**栈内存**中的。 5)如果两个变量保存的是同一个对象引用,当一个通过一个变量修改属性时,另一个也会受到影响。 一个经典的例子 ```javascript var obj2 = obj; //修改obj2的name属性 上面的代码中,当我修改 obj2 的name属性后,会发现,obj 的 name 属性也会被修改。因为obj和obj2指向的是堆内存中的同一个地址。 对于引用类型的数据,赋值相当于地址拷贝,a、b占用了同一段地址。所以改了b,a也会变;本质上a、b就是一个东西。 如果你打算把引用类型 A 的值赋值给 B,让A和B相互不受影响的话,可以通过 Object.assign() 来复制对象。效果如下: ```js // 复制对象:把 obj 赋值给 obj3。两者之间互不影响 属性和方法只能添加给对象,不能添加给基本数据类型。 1、基本数据类型: 注意,基本数据类型`string`是**无法绑定属性和方法**的。比如说: ```javascript str.aaa = 12; 上方代码中,当我们尝试打印`str.aaa`的时候,会发现打印结果为:undefined。也就是说,不能给 `string` 绑定属性和方法。 当然,我们可以打印str.length、str.indexOf("m")等等。因为这两个方法的底层做了数据类型转换(**临时**将 `string` 字符串转换为 `String` 对象,然后再调用内置方法),也就是我们在上一篇文章中讲到的**包装类**。 2、引用数据类型: 引用数据类型`String`是可以绑定属性和方法的。如下: ```javascript 打印结果: ![](http://img.smyhvae.com/20180202_1351.png) 内置对象Number也有一些自带的方法,比如: - Number.MAX_VALUE; - Number.MIN_VALUE; 内置对象Boolean也有一些自带的方法,但是用的不多。 - 由ES标准中定义的对象,在任何的ES的实现中都可以使用 - 比如:Object、Math、Date、String、Array、Number、Boolean、Function等。 2.宿主对象: - 由JS的运行环境提供的对象,目前来讲主要指由浏览器提供的对象。 - 比如 BOM DOM。比如`console`、`document`。 3.自定义对象: - 由开发人员自己创建的对象 - String():将基本数据类型字符串,转换为String对象。 - Number():将基本数据类型的数字,转换为Number对象。 - Boolean():将基本数据类型的布尔值,转换为Boolean对象。 通过上面这这三个包装类,我们可以**将基本数据类型的数据转换为对象**。 代码举例: ```javascript var str = new String("hello"); var bool = new Boolean(true); console.log(typeof num); // 打印结果:object 需要注意的是:我们在实际应用中不会使用基本数据类型的对象。如果使用基本数据类型的对象,在做一些比较时可能会带来一些**不可预期**的结果。 比如说: ```javascript console.log(boo1 === boo2); // 打印结果竟然是:false ```javascript if (boo3) { 当我们对一些基本数据类型的值去调用属性和方法时,浏览器会**临时使用包装类将基本数据类型转换为引用数据类型**,这样的话,基本数据类型就有了属性和方法,然后再调用对象的属性和方法;调用完以后,再将其转换为基本数据类型。 举例: ```js ``` 比如,上面的代码,执行顺序是这样的: ```js // 步骤(2):把临时变量的值 赋值给 str // 步骤(3):销毁临时变量 ``` 在底层,字符串以字符数组的形式保存 代码举例: ```javascript 上方代码中,`smyhvae`这个字符串在底层是以`["s", "m", "y", "h", "v", "a", "e"]`的形式保存的。因此,我们既可以获取字符串的长度,也可以获取指定索引index位置的单个字符。这很像数组中的操作。 再比如,String 对象的很多内置方法,也可以直接给字符串用。此时,也是临时将字符串转换为 String 对象,然后再调用内置方法。 JavaScript 中的对象分为3种:自定义对象 、内置对象、 浏览器对象。 前面两种对象:是JS的基础内容,属于 ECMAScript; 第三个浏览器对象:属于JS独有,即 JS 内置的API。 **内置对象**:就是指这个语言自带的一些对象,供开发者使用,这些对象提供了一些常用或者最基本而必要的功能(属性和方法)。 内置对象最大的优点就是帮助我们快速开发。 **JavaScript的内置对象**: | 内置对象 | 对象说明 | 创建自定义对象的几种方法 **对象的字面量**就是一个{}。里面的属性和方法均是**键值对**: - 键:相当于属性名。 - 值:相当于属性值,可以是任意类型的值(数字类型、字符串类型、布尔类型,函数类型等)。 使用对象字面量来创建一个对象,非常简洁,举例如下:: ```javascript 使用对象字面量,可以在创建对象时,直接指定对象中的属性。语法:{属性名:属性值,属性名:属性值....} 例如: ```javascript name: "猪八戒", ```javascript console.log(JSON.stringify(obj)); 对象字面量的属性名可以加引号也可以不加,建议不加。如果要使用一些特殊的名字,则必须加引号。 属性名和属性值是一组一组的键值对结构,键和值之间使用`:`连接,多个值对之间使用`,`隔开。如果一个属性的后面没有其他的属性了,就不要写`,`,因为它是对象的最后一个属性。 通过该方法可以大批量的创建对象。 ```javascript var obj2 = createPerson("猪八戒", 28, "男"); 第一次看到这种工厂模式时,你可能会觉得陌生。如果简化一下,可以写成下面这种形式,更容易理解:(也就是,利用new Object创建对象) ```javascript 使用工厂方法创建的对象,使用的构造函数都是Object。**所以创建的对象都是Object这个类型,就导致我们无法区分出多种不同类型的对象**。 ```javascript var stu2 = new Student("vae"); 构造函数可以创建自定义对象,所以这里顺带介绍构造函数 是一种特殊的函数,主要用来创建和初始化对象,也就是为对象的成员变量赋初始值。它与 `new` 一起使用才有意义。 我们可以把对象中一些公共的属性和方法抽取出来,然后封装到这个构造函数里面。 ```javascript var per = new Person("孙悟空", 18, "男"); // 创建一个构造函数,专门用来创建 Dog 对象 var dog = new Dog(); 1)构造函数的创建方式和普通函数没有区别,不同的是构造函数习惯上首字母大写。 2)构造函数和普通函数的区别就是**调用方式**的不同:普通函数是直接调用,而构造函数需要使用new关键字来调用。 3)this的指向也有所不同: - 1.以函数的形式调用时,this永远都是window。比如`fun();`相当于`window.fun();` - 2.以方法的形式调用时,this是调用方法的那个对象 - 3.以构造函数的形式调用时,this是新创建的实例对象 new 一个构造函数的执行流程 new 在执行时,会做下面这四件事: (1)开辟内存空间,在内存中创建一个新的空对象。 (2)让 this 指向这个新的对象。 (3)执行构造函数里面的代码,给这个新对象添加属性和方法。 (4)返回这个新对象(所以构造函数里面不需要return)。 因为this指的是new一个Object之后的对象实例。于是,下面这段代码: ```javascript 可以改进为: ```javascript ``` 类:使用同一个构造函数创建的对象,我们称为一类对象,也将一个构造函数称为一个类。 实例:通过一个构造函数创建的对象,称为该类的实例。 使用 instanceof 可以检查**一个对象是否为一个类的实例**。 **语法如下**: ```javascript 如果是,则返回true;否则返回false。 **代码举例**: ```javascript function Dog() {} var person1 = new Person(); var dog1 = new Dog(); console.log(person1 instanceof Person); // 打印结果: true console.log(dog1 instanceof Object); // 所有的对象都是Object的后代。因此,打印结果为:true 根据上方代码中的最后一行,需要补充一点:**所有的对象都是Object的后代,因此 `任何对象 instanceof Object` 的返回结果都是true**。 JS之对象 标签:数组 很多 保存 区分 document 一个 false 可见 ber 原文地址:https://www.cnblogs.com/dangjf/p/13140202.html一、对象
1、面向对象简介
2、对象的概念
var arr = [‘王二‘, 35, ‘男‘, ‘180‘];
```
var person = {};
person.age = 35;
person.sex = ‘男‘;
person.height = ‘180‘;
```
var obj = new Object();
obj.sayName = function () {
console.log(‘smyhvae‘);
};
console.log(‘-----------‘);
console.log(obj.sayName()); //加了括号,就是调用方法。即:执行函数内容,并执行函数体的内容
//创建对象 obj1
var obj1 = new Object();
obj1.test = undefined;
var obj2 = new Object();
obj2.name = "smyhvae";
obj1.test = obj2;
```二、对象和数据类型之间的关系
1、数据类型分类
var obj = new Object();
obj.name = "孙悟空";
obj2.name = "猪八戒";
```
var obj = {name: ‘孙悟空‘};
var obj3 = Object.assign({}, obj);
```2、基本数据类型不能绑定属性和方法
var str = "qianguyihao";
console.log(typeof str); //打印结果为:string
console.log(str.aaa); //打印结果为:undefined
```
var strObj = new String("smyhvae");
strObj.aaa = 123;
console.log(strObj);
console.log(typeof strObj); //打印结果:Object
console.log(strObj.aaa);
```三、对象的分类
1.内置对象:
通过 new 关键字创建出来的对象实例,都是属于对象类型,比如Object、Array、Date等。四、包装类(基本数据类型的数据转换为对象)
1、三个包装类:
var num = new Number(3);
```
var boo1 = new Boolean(true);
var boo2 = new Boolean(true);
```
再比如说:
var boo3 = new Boolean(false);
console.log(‘qianguyihao‘); // 这行代码竟然执行了
}
```2、基本包装类型【重要】
var str = ‘qianguyihao‘;
console.log(str.length); // 打印结果:11
// 步骤(1):把简单数据类型 string 转换为 引用数据类型 String,保存到临时变量中
var temp = new String(‘qianguyihao);
str = temp;
temp = null;
var str = "smyhvae";
console.log(str.length); // 获取字符串的长度
console.log(str[2]); // 获取字符串中的第2个字符
```五、内置对象
|:-------------|:-------------|
| Arguments | 函数参数集合|
| Array | 数组|
| Boolean | 布尔对象|
| Math | 数学对象|
| Date | 日期时间|
| Error | 异常对象|
| Function | 函数构造器|
| Number | 数值对象|
| Object | 基础对象|
| RegExp | 正则表达式对象|
| String | 字符串对象|六、自定义对象的创建
1、对象字面量
var obj = {};
```
var obj2 = {
age: 13,
gender: "男",
test: {
name: "沙僧"
}
sayName: function(){
console.log(‘smyhvae‘);
}
};
```
例如:
var obj = {
name: "张三",
age: 26,
isBoy: true,
// 还可以存放一个签到的对象
test: {
id: 123,
tel: 180
}
//我们还可以在对象中增加一个方法。以后可以通过obj.sayName()的方式调用这个方法
sayName: function() {
console.log(this.name);
}
};
```
2、工厂模式
/*
* 使用工厂方法创建对象
* 通过该方法可以大批量的创建对象
*/
function createPerson(name, age, gender) {
//创建一个新的对象
var obj = new Object();
//向对象中添加属性
obj.name = name;
obj.age = age;
obj.gender = gender;
obj.sayName = function() {
alert(this.name);
};
//将新的对象返回
return obj;
}
var obj3 = createPerson("白骨精", 16, "女");
var obj4 = createPerson("蜘蛛精", 18, "女");
```
var obj = new Obect();
obj.name = ‘猪八戒‘;
obj.age = 28;
obj.gender = ‘男‘;
obj.sayHi = function() {
alert(‘hello world‘);
};
```
**弊端:**3、利用构造函数
//利用构造函数自定义对象
var stu1 = new Student("smyh");
console.log(stu1);
stu1.sayHi();
console.log(stu2);
stu2.sayHi();
// 创建一个构造函数
function Student(name) {
this.name = name; //this指的是当前对象实例【重要】
this.sayHi = function () {
console.log(this.name + "厉害了");
}
}
```七、构造函数
1、构造函数概念
// 创建一个构造函数,专门用来创建Person对象
function Person(name, age, gender) {
this.name = name;
this.age = age;
this.gender = gender;
this.sayName = function() {
alert(this.name);
};
}
var per2 = new Person("玉兔精", 16, "女");
var per3 = new Person("奔波霸", 38, "男");
function Dog() {}
```2、构造函数和普通函数的区别
// 创建一个函数
function createStudent(name) {
var student = new Object();
student.name = name; //第一个name指的是student对象定义的变量。第二个name指的是createStudent函数的参数。二者不一样
}
```
// 创建一个函数
function Student(name) {
this.name = name; //this指的是构造函数中的对象实例
}3、类、实例
4、instanceof
对象 instanceof 构造函数
```
function Person() {}
console.log(dog1 instanceof Person); // 打印结果:false
```