【发布时间】:2016-12-14 13:09:55
【问题描述】:
我想知道 Man 和 Child 的共同点和区别。 p>
class Person {
name: string;
age: number;
}
class Child extends Person {}
class Man implements Person {}
【问题讨论】:
标签: typescript extends implements
我想知道 Man 和 Child 的共同点和区别。 p>
class Person {
name: string;
age: number;
}
class Child extends Person {}
class Man implements Person {}
【问题讨论】:
标签: typescript extends implements
extends 表示:新班级是一个孩子。它得到继承带来的好处。它具有作为其父级的所有属性、方法。它可以覆盖其中的一些并实现新的,但已经包含了父级的东西。
implements 表示:新类可以被视为相同的“形状”,而它不是子类 .它可以传递给任何需要Person 的方法,无论其父对象是否与Person 不同
在OOP (C#、Java 等语言)我们会使用
extends 从继承中获利(参见wiki)。小引:
... 在大多数基于类的面向对象语言中,继承是一种机制,其中一个对象获取父对象的所有属性和行为。继承允许程序员:创建基于现有类的类...
implements 将更多地用于多态性(参见wiki)。小引:
...多态性是为不同类型的实体提供单一接口...
所以,我们的class Man 可以拥有完全不同的继承树。
class Man extends Human ...
但如果我们也声明我们可以伪装成不同的类型 - Person:
class Man extends Human
implements Person ...
.. 那么我们可以在任何需要Person 的地方使用它。我们只需要完成 Persons 的 "interface"(即实现其所有公共内容)。
implement其他班级?这真是很酷的东西Javascript 的漂亮外观(其中一个好处) 是对 Duck 类型 (see wiki) 的内置支持。小引:
“如果它走路像鸭子,叫起来像鸭子,那它一定是鸭子。”
因此,在 Javascript 中,如果两个不同的对象...会有一个相似的方法(例如 render()),它们可以传递给期望它的函数:
function(engine){
engine.render() // any type implementing render() can be passed
}
为了不失去这一点——我们可以在 Typescript 中做同样的事情——提供更多类型的支持。这就是
class implements class
有它的作用,它是有意义的
在 OOP 语言中,C# ... 没办法做到这一点...
Interfaces Extending Classes
当一个接口类型扩展一个类类型时,它会继承其成员 类,但不是他们的实现。好像界面有 声明了类的所有成员而不提供 执行。接口甚至继承私有和受保护的 基类的成员。这意味着当你创建一个接口时 扩展具有私有或受保护成员的类,该接口 type 只能由该类或其子类实现。
当你有一个大的继承层次结构时,这很有用,但是想要 指定您的代码仅适用于具有特定 特性。除了继承之外,子类不必相关 从基类。例如:
class Control { private state: any; } interface SelectableControl extends Control { select(): void; } class Button extends Control implements SelectableControl { select() { } } class TextBox extends Control { select() { } } // Error: Property 'state' is missing in type 'Image'. class Image implements SelectableControl { private state: any; select() { } } class Location { }
所以,虽然
extends 表示 - 它从其父级获取所有内容implements 在这种情况下几乎就像实现一个接口。子对象可以假装它是父对象......但它没有得到任何实现【讨论】:
extends-it get all from its parent”时,它是否适用于私人成员?例如class Person {private name: string} class man extends Person{gender: string;} man 有属性名称吗?
在 typescript(和其他一些 OO 语言)中,您有类和接口。
接口没有实现,它只是该类型所具有的成员/方法的“契约”。
例如:
interface Point {
x: number;
y: number;
distance(other: Point): number;
}
实现这个Point接口的实例必须有两个类型编号的成员:x和y,一个方法distance接收另一个Point实例并返回一个number。
该接口没有实现任何这些。
类是实现:
class PointImplementation implements Point {
public x: number;
public y: number;
constructor(x: number, y: number) {
this.x = x;
this.y = y;
}
public distance(other: Point): number {
return Math.sqrt(Math.pow(this.x - other.x, 2) + Math.pow(this.y - other.y, 2));
}
}
在您的示例中,您在扩展 Person 类时将其视为类,在实现时将其视为接口。
你的代码:
class Person {
name: string;
age: number;
}
class Child extends Person {}
class Man implements Person {}
有一个编译错误说:
“Man”类错误地实现了“Person”接口。
“人”类型中缺少属性“名称”。
那是因为接口缺乏实现。
所以如果你 implement 一个类,那么你只接受它的“合同”而没有实现,所以你需要这样做:
class NoErrorMan implements Person {
name: string;
age: number;
}
底线是,在大多数情况下,您想要extend 另一个类而不是implement 它。
【讨论】:
extends: 子类(被扩展)将继承类的所有属性和方法被扩展implements:使用implements 关键字的类需要实现它implements 的类的所有属性和方法
简单来说:
extends: 在这里你可以从父类中获取所有这些方法/属性,所以你不必自己实现这个implements:这是班级必须遵守的合同。该类必须实现至少以下方法/属性class Person {
name: string;
age: number;
walk(): void {
console.log('Walking (person Class)')
}
constructor(name: string, age: number) {
this.name = name;
this.age = age;
}
}
class child extends Person { }
// Man has to implements at least all the properties
// and methods of the Person class
class man implements Person {
name: string;
age: number
constructor(name: string, age: number) {
this.name = name;
this.age = age;
}
walk(): void {
console.log('Walking (man class)')
}
}
(new child('Mike', 12)).walk();
// logs: Walking(person Class)
(new man('Tom', 12)).walk();
// logs: Walking(man class)
在示例中,我们可以观察到子类继承了 Person 的所有内容,而 man 类必须实现 Person 本身的所有内容。
如果我们要从 man 类中删除一些东西,例如 walk 方法,我们会得到以下 compile time 错误:
“man”类错误地实现了“Person”类。你的意思是 扩展“人”并将其成员作为子类继承?财产 “man”类型中缺少“walk”,但在“Person”类型中是必需的。(2720)
【讨论】:
来自@nitzan-tomer 的好答案!帮了我很多...我扩展了他的演示:
IPoint interface;
Point implements IPoint;
Point3D extends Point;
以及它们在期望 IPoint 类型的函数中的行为。
到目前为止,我所学到的并被用作经验法则:如果您正在使用期望泛型类型的类和方法,请使用接口作为期望的类型。并确保父类或基类使用该接口。这样你就可以使用那些实现接口的所有子类。
【讨论】:
extends 专注于继承,implements 专注于约束,无论是接口还是类。
【讨论】:
基本上:
extends 将获取父类的所有属性和方法。implements 将要求我们实现接口中定义的所有属性和方法。【讨论】: