搜索
首页web前端js教程JavaScript之自定义类型_javascript技巧

1、直接创建模式。这是最简单也是最直接的一种模式,首先创建一个引用类型的对象,然后为其添加自定义属性和方法。示例代码如下:

复制代码 代码如下:

var person = new Object();
person.name = "Sam";
person.age = 16;
person.speak = function(){
alert(this.name + "is " + this.age + "years old");
}
person.speak();

可以看到,上面创建了一个Object类型的对象,然后为其添加了name和age属性以及一个speak方法。直接创建模式虽然简单,但其缺点是显而易见的:当我们需要创建许多相同的对象时,每次都要重复编写代码。为了解决这个问题,我们可以将创建对象的过程进行封装,于是便有了下面的工厂模式。
2、工厂模式。工厂模式是程序设计中一种常用的设计模式,它主要是将创建对象的过程进行了封装,示例代码如下:

复制代码 代码如下:

function createPerson(name, age){
var person = new Object();
person.name = name;
person.age = age;
person.speak = function(){
alert(this.name + "is " + this.age + "years old");
}
return person;
}
var person1 = createPerson("Sam", 16);
var person2 = createPerson("Jack", 18);

使用工厂模式后,创建相同类型的对象变得简单了。但工厂模式没有解决对象识别的问题,即我们无法确定创建的对象的具体类型。有过面向对象编程经验的开发人员都知道,对象的创建应当基于类,有了具体的自定义类,再来创建该类的对象。幸好,在JavaScript中,我们可以通过构造函数模式来模拟一个类。
3、构造函数模式。构造函数和普通函数没有任何区别。任何普通函数都可以作为构造函数,只要使用new操作符即可;任何构造函数也都可以作为普通函数来调用。只不过在JavaScript中,有一个约定,就是用作构造函数的函数名需要首字母大写。示例代码如下:

复制代码 代码如下:

function Person(name, age){
this.name = name;
this.age = age;
this.speak = function(){
alert(this.name + "is " + this.age + "years old");
}
}
var person1 = new Person("Sam", 16);
var person2 = new Person("Jack", 18);

可以看到,在构造函数内部,我们使用了this来添加属性和方法,那么,这个this是指什么呢?当我们创建了一个Person的对象时,this即是指这个创建的对象。现在,我们可以识别出对象person1和person2的具体类型了。使用alert(person1 instanceOf Person)后可以发现,输出的值为true。但构造函数模式也有自己的缺点,就是构造函数内声明的方法在每次创建新对象时都会重新创建(在JavaScript中,函数也是对象)。也就是说,构造函数内的方法是与对象绑定的,而不是与类绑定的。下面代码的输出可以验证我们的推断。
alert(person1.speak == person2.speak); // false 解决这个缺点的一种比较简单的方法就是将函数的声明放到构造函数的外面,即:

复制代码 代码如下:

function Person(name, age){
this.name = name;
this.age = age;
this.speak = speak;
}
function speak(){
alert(this.name + "is " + this.age + "years old");
}
var person1 = new Person("Sam", 16);
var person2 = new Person("Jack", 18);
alert(person1.speak == person2.speak); // true

问题解决了,但这种方法又带来了新的问题。首先,函数speak是在全局作用域中声明的,但它却只能被用于Person构造函数,放在全局作用域中有被误用的风险;其次,如果一个自定义类型有很多的方法,则需要声明很多的全局函数,这既将导致全局作用域的污染,也不利于代码的封装。那么,有没有什么办法能让自定义类型的方法成为与类绑定的,又不污染全局作用域呢?答案是使用原型模式。
4、原型模式。在我们声明一个新的函数后,该函数(在JavaScript中,函数也是对象)就会拥有一个prototype的属性。prototype是一个对象,表示会被该函数创建的所有对象拥有的公共属性和方法。示例代码如下:

复制代码 代码如下:

function Person(){}
Person.prototype.name="Sam";
Person.prototype.age=16;
Person.prototype.speak = function(){
alert(this.name + "is " + this.age + "years old");
}
var person1 = new Person();
person1.speak();
var person2 = new Person();
alert(person1.speak == person2.speak); // true


可以看到,虽然构造函数内没有声明speak方法,但我们创建的对象person1还是能调用speak方法,这是因为JavaScript有一个搜索规则,先搜索实例属性和方法,找到则返回;如果没找到,则再到prototype中去搜索。原型模式使得方法是与类相关的,并且没有污染全局作用域,但其也有自身的缺点:一是所有属性也都与类相关,这意味着所有对象共享一份属性,这显然是不合理的;二是没有办法向构造函数传入初始化数据了。解决的方法很简单,就是混合使用构造函数模式和原型模式。
5、组合模式。示例代码如下:

复制代码 代码如下:

function Person(name, age){
this.name = name;
this.age = age;
}
Person.prototype.speak = function(){
alert(this.name + "is " + this.age + "years old");
}
var person1 = new Person();
person1.speak();
var person2 = new Person();
alert(person1.speak == person2.speak); // true

不难发现,组合模式实现了我们的所有需求,这也是目前应用得比较广泛的一种模式。有面向对象编程经验的开发人员可能会觉得将prototype的声明放在构造函数外面有点别扭,那么能否将其放到构造函数里去呢?答案是肯定的,使用动态组合模式即可。
6、动态组合模式。其原理就是先判断原型中的某个属性或方法是不是已经声明过,如果没有声明,则声明整个原型;否则,什么也不用做。示例代码如下:

复制代码 代码如下:

function Person(name, age){
this.name = name;
this.age = age;
if (Person.prototype.speak == "undefined"){
Person.prototype.speak = function(){
alert(this.name + "is " + this.age + "years old");
}
}
}
声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
如何使用 Go 语言定义和使用自定义类型?如何使用 Go 语言定义和使用自定义类型?Jun 05, 2024 pm 12:41 PM

在Go中,自定义类型可使用type关键字定义(struct),包含命名字段。它们可以通过字段访问运算符访问,并可附加方法来操作实例状态。在实际应用中,自定义类型用于组织复杂数据和简化操作。例如,学生管理系统使用自定义类型Student存储学生信息,并提供计算平均成绩和出勤率的方法。

C++ 函数返回自定义类型时有什么要求?C++ 函数返回自定义类型时有什么要求?Apr 19, 2024 pm 03:33 PM

C++函数可以返回自定义类型,满足如下要求:类型完整定义。默认构造函数。值类型需要复制构造函数。

如何在 Golang 中创建不可变自定义类型?如何在 Golang 中创建不可变自定义类型?Jun 02, 2024 am 09:41 AM

是的,在Go中创建不可变的自定义类型可以提供许多好处,包括线程安全性、易于推理和更强的错误处理。要创建不可变类型,需要遵循以下步骤:定义类型:声明一个包含成员变量的自定义类型,不应包括指针。声明不可变性:确保所有成员变量都是基础类型或其他不可变类型,避免使用切片、映射或指针。使用值接收器方法:为与类型关联的方法使用值接收器,禁止结构体字面量分配,强制方法仅对自身进行操作。

在 Golang 中,如何比较自定义类型的值?在 Golang 中,如何比较自定义类型的值?Jun 05, 2024 pm 01:04 PM

在Golang中,可通过以下方式比较自定义类型的值:对于具有相同底层表示形式的类型,直接使用==操作符进行比较。对于更复杂的类型,使用reflect.DeepEqual函数递归比较两个值的全部内容。

使用 Golang 时如何对自定义类型进行序列化?使用 Golang 时如何对自定义类型进行序列化?Jun 03, 2024 pm 05:07 PM

在Go中,序列化自定义类型的方法有:使用JSON序列化时实现json.Marshaler接口,使用Gob序列化时实现encoding/gob包中的GobEncoder和GobDecoder接口。

深入探究Go语言中的自定义类型方法深入探究Go语言中的自定义类型方法Mar 24, 2024 am 09:45 AM

深入探究Go语言中的自定义类型方法在Go语言中,我们可以为自定义类型添加方法,以扩展该自定义类型的功能。通过在类型上定义方法,我们可以实现面向对象编程的特性,使代码更加模块化和可维护。本文将深入探究在Go语言中自定义类型方法的使用,同时提供具体的代码示例。1.什么是自定义类型方法在Go语言中,我们可以通过在结构体上定义方法,实现自定义类型的方法。这些方法可

如何在Go中实现自定义类型?如何在Go中实现自定义类型?May 10, 2023 pm 03:34 PM

Go是一种面向对象的静态类型编程语言,因此允许开发者通过自定义类型来简化代码并提高可读性。在本文中,我们将探讨如何在Go中实现自定义类型,以便开发者可以更好地利用这个特性。结构体Go中的结构体是一种自定义类型,它允许开发者通过将不同类型的变量组合在一起来实现更复杂的数据结构。以下是一个结构体的示例:typePersonstruct{name

如何在 Golang 中为自定义类型定义方法?如何在 Golang 中为自定义类型定义方法?Jun 03, 2024 am 11:08 AM

在Go中,可以为自定义类型定义方法,即方法接收者。只需定义接收器类型、方法名和参数,即可为特定类型添加相应行为。

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
2 周前By尊渡假赌尊渡假赌尊渡假赌
仓库:如何复兴队友
4 周前By尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒险:如何获得巨型种子
4 周前By尊渡假赌尊渡假赌尊渡假赌

热工具

安全考试浏览器

安全考试浏览器

Safe Exam Browser是一个安全的浏览器环境,用于安全地进行在线考试。该软件将任何计算机变成一个安全的工作站。它控制对任何实用工具的访问,并防止学生使用未经授权的资源。

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

功能强大的PHP集成开发环境

SublimeText3 英文版

SublimeText3 英文版

推荐:为Win版本,支持代码提示!

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具