Heim >Web-Frontend >js-Tutorial >So schreiben Sie wartbaren objektorientierten JavaScript-Code

So schreiben Sie wartbaren objektorientierten JavaScript-Code

伊谢尔伦
伊谢尔伦Original
2016-11-29 10:08:521023Durchsuche

Die Fähigkeit, wartbaren objektorientierten JavaScript-Code zu schreiben, wird Ihnen nicht nur Geld sparen, sondern Sie auch beliebt machen. Glauben Sie es nicht? Es ist möglich, dass Sie oder jemand anderes eines Tages zurückkommt und Ihren Code wiederverwendet. Wenn Sie diese Erfahrung so schmerzlos wie möglich gestalten können, können Sie viel Zeit sparen. Jeder auf der Erde weiß, dass Zeit Geld ist. Ebenso können Sie sich die Gunst einer anderen Person verdienen, indem Sie ihr Kopfschmerzen ersparen. Aber bevor wir damit beginnen, zu untersuchen, wie man wartbaren objektorientierten JavaScript-Code schreibt, werfen wir einen kurzen Blick darauf, was objektorientiert ist. Wenn Sie bereits objektorientierte Konzepte verstehen, können Sie den nächsten Abschnitt direkt überspringen.


Was ist objektorientiert?


Objektorientierte Programmierung repräsentiert hauptsächlich physische Objekte in der realen Welt durch Code. Um ein Objekt zu erstellen, müssen Sie zunächst eine „Klasse“ schreiben, um es zu definieren. Klassen können fast alles repräsentieren: Konten, Mitarbeiter, Navigationsmenüs, Autos, Pflanzen, Anzeigen, Getränke usw. Und jedes Mal, wenn Sie ein Objekt erstellen möchten, instanziieren Sie ein Objekt aus der Klasse. Mit anderen Worten: Eine Instanz einer Klasse wird als Objekt erstellt. Tatsächlich werden Objekte normalerweise verwendet, wenn es um mehr als eine Sache desselben Typs geht. Außerdem können einfache funktionale Programme einen tollen Job machen. Objekte sind im Wesentlichen Container für Daten. Daher möchten Sie möglicherweise in einem Mitarbeiterobjekt die Mitarbeiternummer, den Namen, das Eintrittsdatum, den Titel, das Gehalt, das Dienstalter usw. speichern.


Objekte enthalten auch Funktionen (auch „Methoden“ genannt), die Daten verarbeiten. Methoden werden als Vermittler eingesetzt, um die Datenintegrität sicherzustellen und Daten vor der Speicherung zu transformieren. Beispielsweise kann eine Methode ein Datum in einem beliebigen Format akzeptieren und es vor dem Speichern in ein standardisiertes Format konvertieren. Schließlich können Klassen auch von anderen Klassen erben. Durch Vererbung können Sie denselben Code in verschiedenen Klassen wiederverwenden. Beispielsweise können sowohl Bankkonten als auch Videothekenkonten eine grundlegende Kontoklasse erben, die persönliche Informationen, Kontoeröffnungsdatum, Filialinformationen usw. umfasst. Anschließend kann jeder seine eigenen Datenstrukturen und Methoden für die Transaktions- oder Kreditabwicklung definieren.


Warnung: JavaScript objektorientiert ist nicht dasselbe


Im vorherigen Abschnitt ein Überblick über den Klassiker Objektorientiert Grundkenntnisse der Objektprogrammierung. Ich sage klassisch, weil JavaScript diesen Regeln nicht folgt. Im Gegensatz dazu werden JavaScript-Klassen als Funktionen geschrieben und die Vererbung wird durch Prototypen implementiert. Bei der prototypischen Vererbung handelt es sich grundsätzlich um die Verwendung von Prototypeigenschaften zur Implementierung der Vererbung von Objekten, anstatt Klassen von Klassen zu erben.


Instanziierung von Objekten


Das Folgende ist ein Beispiel für die Objektinstanziierung in JavaScript:

// 定义Employee类
  function Employee(num, fname, lname) {
    this.getFullName = function () {
  return fname + " " + lname;
  }
  };
// 实例化Employee对象
  var john = new Employee("4815162342", "John", "Doe");
  alert("The employee's full name is " + john.getFullName());

Hier sind drei wichtige Punkte zu beachten:


1 Der erste Buchstabe des Funktionsnamens „Klasse“ muss großgeschrieben werden. Dies weist darauf hin, dass die Funktion instanziiert werden soll und nicht wie eine normale Funktion aufgerufen werden soll.


2 verwendet den neuen Operator während der Instanziierung. Wenn Sie new weglassen und nur die Funktion aufrufen, treten viele Probleme auf.


3 Da getFullName diesem Operator zugewiesen ist, ist es öffentlich verfügbar, fname und lname jedoch nicht. Der von der Employee-Funktion generierte Abschluss gewährt getFullName Zugriff auf fname und lname, bleibt jedoch für andere Klassen privat.


Prototypische Vererbung


Hier ist ein Beispiel für prototypische Vererbung in JavaScript:

// 定义Human类
function Human() {
this.setName = function (fname, lname) {
this.fname = fname;
this.lname = lname;
}
this.getFullName = function () {
return this.fname + " " + this.lname;
}
}
// 定义Employee类
function Employee(num) {
this.getNum = function () {
return num;
}
};
//让Employee继承Human类
Employee.prototype = new Human();
// 实例化Employee对象
var john = new Employee("4815162342");
john.setName("John", "Doe");
alert(john.getFullName() + "'s employee number is " + john.getNum());

Dieses Mal habe ich eine Human-Klasse erstellt, die alle gemeinsamen Attribute von Menschen enthält – ich habe auch fname und lname darin eingefügt, denn nicht nur Mitarbeiter haben Namen, sondern jeder hat Namen. Weisen Sie dann das Human-Objekt seiner Prototypeigenschaft zu.


Code-Wiederverwendung durch Vererbung


Im vorherigen Beispiel wurde die ursprüngliche Employee-Klasse in zwei Teile aufgeteilt. Alle allgemeinen menschlichen Attribute wurden in die Klasse „Mensch“ verschoben, und dann wurde „Mitarbeiter“ von „Mensch“ übernommen. In diesem Fall können die Eigenschaften in Human von anderen Objekten wie Student, Client, Citizen, Visitor usw. verwendet werden. Jetzt haben Sie vielleicht bemerkt, dass dies eine großartige Möglichkeit ist, Code aufzuteilen und wiederzuverwenden. Wenn Sie mit Human-Objekten arbeiten, müssen Sie Human nur erben, um die vorhandenen Eigenschaften zu verwenden, anstatt jedes einzelne Objekt einzeln neu zu erstellen. Wenn Sie außerdem ein Attribut „zweiter Vorname“ hinzufügen möchten, müssen Sie es nur einmal hinzufügen, und diejenigen, die die Human-Klasse erben, können es sofort verwenden. Im Gegenteil, wenn wir nur das Attribut „zweiter Vorname“ zu einem Objekt hinzufügen möchten, können wir es direkt zu diesem Objekt hinzufügen, ohne es der Human-Klasse hinzuzufügen.


1. Öffentlich und privat


Das nächste Thema möchte ich über öffentliche und private Variablen in Klassen sprechen . Abhängig davon, wie die Daten im Objekt behandelt werden, werden die Daten als privat oder öffentlich behandelt. Privateigentümer bedeuten nicht zwangsläufig, dass andere keinen Zugriff darauf haben. Möglicherweise muss nur eine bestimmte Methode verwendet werden.


Schreibgeschützt


Manchmal möchten Sie einfach nur einen Wert, wenn Sie ein Objekt erstellen. Nach der Erstellung möchten Sie nicht, dass andere diesen Wert ändern. Erstellen Sie dazu eine private Variable und weisen Sie ihr bei der Instanziierung einen Wert zu.

function Animal(type) {
var data = [];
data['type'] = type;
this.getType = function () {
return data['type'];
}
}
var fluffy = new Animal('dog');
fluffy.getType();
// 返回 'dog'

在这个例子中,Animal类中创建了一个本地数组data。当 Animal对象被实例化时,传递了一个type的值并将该值放置在data数组中。因为它是私有的,所以该值无法被覆盖(Animal函数定义了它的范围)。一旦对象被实例化了,读取type值的唯一方式是调用getType方法。因为getType是在Animal中定义的,因此凭借Animal产生的闭包,getType可以进到data中。这样的话,虽可以读到对象的类型却无法改变。


有一点非常重要,就是当对象被继承时,“只读”技术就无法运用。在执行继承后,每个实例化的对象都会共享那些只读变量并覆盖其值。最简单的解决办法是将类中的只读变量转换成公共变量。但是你必须保持它们是私有的,你可以使用Philippe在评论中提到的技术。


Public(公有)


当然也有些时候你想要任意读写某个属性的值。要实现这一点,需要使用this操作符。

function Animal() {
this.mood = '';
}
var fluffy = new Animal();
fluffy.mood = 'happy';
fluffy.mood;
// 返回 'happy'


这次Animal类公开了一个叫mood的属性,可以被随意读写。同样地,你还可以将函数指定给公有的属性,例如之前例子中的getType函数。只是要注意不要给getType赋值,不然的话你会毁了它的。


完全私有


最后,可能你发现你需要一个完全私有化的本地变量。这样的话,你可以使用与第一个例子中一样的模式而不需要创建公有方法。

function Animal() {
var secret = "You'll never know!"
}
var fluffy = new Animal();


2. 写灵活的API


既然我们已经谈到类的创建,为了保持与产品需求变化同步,我们需要保持代码不过时。如果你已经做过某些项目或者是长期维护过某个产品,那么你就应该知道需求是变化的。这是一个不争的事实。如果你不是这么想的话,那么你的代码在还没有写之前就将注定荒废。可能你突然就需要将选项卡中的内容弄成动画形式,或是需要通过Ajax调用来获取数据。尽管准确预测未来是不大可能,但是却完全可以将代码写灵活以备将来不时之需。


Saner参数列表


在设计参数列表的时候可以让代码有前瞻性。参数列表是让别人实现你代码的主要接触点,如果没有设计好的话,是会很有问题的。你应该避免下面这样的参数列表:

function Person(employeeId, fname, lname, tel, fax, email, email2, dob) {
};


这个类十分脆弱。如果在你发布代码后想要添加一个中间名参数,因为顺序问题,你不得不在列表的最后往上加。这让工作变得尴尬。如果你没有为每个参数赋值的话,将会十分困难。例如:

var ara = new Person(1234, "Ara", "Pehlivanian", "514-555-1234", null, null, null, "1976-05-17");

操作参数列表更整洁也更灵活的方式是使用这个模式:

function Person(employeeId, data) {
};

有第一个参数因为这是必需的。剩下的就混在对象的里面,这样才可以灵活运用。

var ara = new Person(1234, {
fname: "Ara",
lname: "Pehlivanian",
tel: "514-555-1234",
dob: "1976-05-17"
});

这个模式的漂亮之处在于它即方便阅读又高度灵活。注意到fax, email和email2完全被忽略了。不仅如此,对象是没有特定顺序的,因此哪里方便就在哪里添加一个中间名参数是非常容易的:

var ara = new Person(1234, {
fname: "Ara",
mname: "Chris",
lname: "Pehlivanian",
tel: "514-555-1234",
dob: "1976-05-17"
});

类里面的代码不重要,因为里面的值可以通过索引来访问:

function Person(employeeId, data) {
this.fname = data['fname'];
};

如果data['fname'] 返回一个值,那么他就被设定好了。否则的话,没被设定好,也没有什么损失。

让代码可嵌入

随着时间流逝,产品需求可能对你类的行为有更多的要求。而该行为却与你类的核心功能没有半毛钱关系。也有可能是类的唯一一种实现,好比在一个选项卡的面板获取另一个选项卡的外部数据时,将这个选项卡面板中的内容变灰。你可能想把这些功能放在类的里面,但是它们不属于那里。选项卡条的责任在于管理选项卡。动画和获取数据是完全不同的两码事,也必须与选项卡条的代码分开。唯一一个让你的选项卡条不过时而又将那些额外的功能排除在外的方法是,允许将行为嵌入到代码当中。换句话说,通过创建事件,让它们在你的代码中与关键时刻挂钩,例如onTabChange, afterTabChange, onShowPanel, afterShowPanel等等。那样的话,他们可以轻易地与你的onShowPanel事件挂钩,写一个将面板内容变灰的处理器,这样就皆大欢喜了。JavaScript库让你可以足够容易地做到这一点,但是你自己写也不那么难。下面是使用YUI 3的一个例子。

<script type="text/javascript" src="http://yui.yahooapis.com/combo?3.2.0/build/yui/yui-min.js"></script>
<script type="text/javascript">
YUI().use(&#39;event&#39;, function (Y) {
function TabStrip() {
this.showPanel = function () {
this.fire(&#39;onShowPanel&#39;);
// 展现面板的代码
this.fire(&#39;afterShowPanel&#39;);
};
};
// 让TabStrip有能力激发常用事件
Y.augment(TabStrip, Y.EventTarget);
var ts = new TabStrip();
// 给TabStrip的这个实例创建常用时间处理器
ts.on(&#39;onShowPanel&#39;, function () {
//在展示面板之前要做的事
});
ts.on(&#39;onShowPanel&#39;, function () {
//在展示面板之前要做的其他事
});
ts.on(&#39;afterShowPanel&#39;, function () {
//在展示面板之后要做的事
});
ts.showPanel();
});
</script>

这个例子有一个简单的 TabStrip 类,其中有个showPanel方法。这个方法激发两个事件,onShowPanel和afterShowPanel。这个能力是通过用Y.EventTarget扩大类来实现的。一旦做成,我们就实例化了一个TabStrip对象,并将一堆处理器都分配给它。这是用来处理实例的唯一行为而又能避免混乱当前类的常用代码。


总结


如果你打算重用代码,无论是在同一网页,同一网站还是跨项目操作,考虑一下在类里面将其打包和组织起来。面向对象JavaScript很自然地帮助实现更好的代码组织以及代码重用。除此以外,有点远见的你可以确保代码具有足够的灵活性,可以在你写完代码后持续使用很长时间。编写可重用的不过时JavaScript代码可以节省你,你的团队还有你公司的时间和金钱。这绝对能让你大受欢迎。


Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn