本章我们将讨论 C# 中的面向对象编程。
共有三种广泛使用的编程范例:过程编程,函数编程和面向对象的编程。 C# 支持过程式编程和面向对象的编程。
OOP 定义
面向对象编程(OOP)是一种使用对象及其相互作用设计应用和计算机程序的编程范例。
OOP 中有一些基本的编程概念:
- 抽象化
- 多态性
- 封装形式
- 继承
抽象通过建模适合该问题的类来简化复杂的现实。 多态性是将运算符或函数以不同方式用于不同数据输入的过程。 封装对其他对象隐藏了类的实现细节。 继承是一种使用已经定义的类形成新类的方法。
C# 对象
对象是 C# OOP 程序的基本构建块。 对象是数据和方法的组合。 数据和方法称为对象的成员。 在 OOP 程序中,我们创建对象。 这些对象通过方法进行通信。 每个对象都可以接收消息,发送消息和处理数据。
创建对象有两个步骤。 首先,我们定义一个类。 类是对象的模板。 它是一个蓝图,描述了类对象共享的状态和行为。 一个类可以用来创建许多对象。 在运行时从类创建的对象称为该特定类的实例。
Program.cs
在第一个示例中,我们创建一个简单的对象。
这是一个简单的类定义。 模板的主体为空。 它没有任何数据或方法。
我们创建Being
类的新实例。 为此,我们使用了new
关键字。 b
变量是创建对象的句柄。
我们将对象打印到控制台以获取该对象的一些基本描述。 打印对象是什么意思? 实际上,当我们打印对象时,我们将其称为ToString()
方法。 但是我们还没有定义任何方法。 这是因为创建的每个对象都继承自基本object
。 它具有一些基本功能,可以在所有创建的对象之间共享。 其中之一是ToString()
方法。
我们得到对象类名。
C# 对象属性
对象属性是捆绑在类实例中的数据。 对象属性称为实例变量或成员字段。 实例变量是在类中定义的变量,该类中的每个对象都有一个单独的副本。
Program.cs
在上面的 C# 代码中,我们有一个带有一个成员字段的Person
类。
我们声明一个名称成员字段。 public
关键字指定可以在类块之外访问成员字段。
我们创建Person
类的实例,并将名称变量设置为“ Jane”。 我们使用点运算符来访问对象的属性。
我们创建Person
类的另一个实例。 在这里,我们将变量设置为“ Beky”。
我们将变量的内容打印到控制台。
我们看到了程序的输出。 Person
类的每个实例都有一个单独的名称成员字段副本。
C# 方法
方法是在类主体内定义的函数。 它们用于通过对象的属性执行操作。 方法将模块化带入我们的程序。
在 OOP 范式的封装概念中,方法至关重要。 例如,我们的AccessDatabase
类中可能有一个Connect()
方法。 我们无需知道方法Connect()
如何精确地连接到数据库。 我们只需要知道它用于连接数据库。 这对于划分编程中的职责至关重要,尤其是在大型应用中。
对象将状态和行为分组,方法代表对象的行为部分。
Program.cs
在代码示例中,我们有一个 Circle 类。 我们定义了两种方法。
我们只有一个成员字段。 它是圆的半径。 private
关键字是访问说明符。 它表明变量仅限于外部世界。 如果要从外部修改此变量,则必须使用公共可用的SetRadius()
方法。 这样我们可以保护我们的数据。
这是SetRadius()
方法。 this
变量是一个特殊变量,我们用它来访问方法中的成员字段。 this.radius
是实例变量,而半径是局部变量,仅在SetRadius()
方法内部有效。
我们创建Circle
类的实例,并通过在圆对象上调用SetRadius()
方法来设置其半径。 我们使用点运算符来调用该方法。
Area()
方法返回圆的面积。 Math.PI
是内置常数。
运行该示例可得出此结果。
C# 访问修饰符
访问修饰符设置方法和成员字段的可见性。 C# 具有四个基本访问修饰符:public
,protected
,private
和internal
。 可以从任何地方访问public
成员。 protected
成员只能在类本身内部以及继承的和父类访问。 private
成员仅限于包含类型,例如 仅在其类或接口内。 可以从同一程序集(exe 或 DLL)中访问internal
成员。
修饰符还有两种组合:protected internal
和private protected
。 protected internal
类型或成员可以由声明它的程序集中的任何代码访问,也可以从另一个程序集中的派生类中访问。 private protected
类型或成员只能在其声明程序集中通过同一个类或从该类派生的类型的代码进行访问。
访问修饰符可防止意外修改数据。 它们使程序更强大。
类 | 当前程序集 | 派生类 | 当前程序集中的派生类 | 整个程序 | |
---|---|---|---|---|---|
public |
+ | + | + | + | + |
`protected | + | o | + | + | o |
internal |
+ | + | o | o | o |
private |
+ | o | o | o | o |
protected internal |
+ | + | + | + | o |
private protected |
+ | o | o | + | o |
上表总结了 C# 访问修饰符(+是可访问的,o 是不可访问的)。
Program.cs
在上面的程序中,我们有两个成员字段。 一个被宣布为公开,另一个被宣布为私有。
如果成员字段是private
,则访问它的唯一方法是通过方法。 如果要在类外部修改属性,则必须将方法声明为public
。 这是数据保护的重要方面。
SetAge()
方法使我们能够从类定义之外更改private
年龄变量。
我们创建Person
类的新实例。 因为名称属性是public
,所以我们可以直接访问它。 但是,不建议这样做。
SetAge()
方法修改年龄成员字段。 由于已声明private
,因此无法直接访问或修改。
最后,我们访问两个成员以构建一个字符串。
运行示例将给出此输出。
具有private
访问修饰符的成员字段不被派生类继承。
Program.cs
在前面的程序中,我们有一个Derived
类,该类继承自Base
类。 Base
类具有三个成员字段。 全部具有不同的访问修饰符。 isDefined 成员不继承。 private
修饰符可以防止这种情况。
类Derived
继承自Base
类。 要从另一个类继承,我们使用冒号(:)运算符。
public
和protected
成员由Derived
类继承。 可以访问它们。 private
成员未继承。 访问成员字段的行被注释。 如果我们取消注释该行,则代码将无法编译。
运行程序,我们收到此输出。
C# 构造函数
构造函数是一种特殊的方法。 创建对象时会自动调用它。 构造函数不返回值。 构造函数的目的是初始化对象的状态。 构造函数与类具有相同的名称。 构造函数是方法,因此它们也可以重载。
构造函数不能被继承。 它们按继承顺序被调用。 如果我们不为类编写任何构造函数,则 C# 提供一个隐式默认构造函数。 如果提供任何类型的构造函数,则不提供默认值。
Program.cs
我们有一个Being
类。 此类具有两个构造函数。 第一个不带参数; 第二个采用一个参数。
此构造函数采用一个字符串参数。
创建Being
类的实例。 这次,在创建对象时调用没有参数的构造函数。
这是程序的输出。
在下一个示例中,我们初始化类的数据成员。 变量的初始化是构造函数的典型工作。
Program.cs
我们有一个带有数据成员和方法的MyFriend
类。
类定义中有两个私有变量。
在构造函数中,我们启动两个数据成员。 this
变量是用于引用对象变量的处理程序。
我们创建带有两个参数的MyFriend
对象。 然后我们调用对象的Info()
方法。
这是输出。
C# 构造函数链接
构造函数链接是类从构造函数调用另一个构造函数的能力。 要从同一类调用另一个构造函数,我们使用this
关键字。
Program.cs
我们有一个Circle
类。 该类具有两个构造函数。 一种采用一个参数,一种不采用任何参数。
此构造函数采用一个参数-radius
。
这是没有参数的构造函数。 它只是简单地调用另一个构造函数,并为其提供默认半径 1。
C# ToString
方法
每个对象都有一个ToString()
方法。 它返回人类可读的对象表示形式。 默认实现返回Object
类型的标准名称。 请注意,当我们使用对象作为参数调用Console.WriteLine()
方法时,将调用ToString()
。
Program.cs
我们有一个Being
类,其中我们重写了ToString()
方法的默认实现。
创建的每个类都从基object
继承。 ToString()
方法属于此对象类。 我们使用override
关键字来通知我们正在覆盖方法。
我们创建一个自定义对象和一个内置对象。
我们在这两个对象上调用ToString()
方法。
正如我们之前指定的,将对象作为Console.WriteLine()
的参数将调用其ToString()
方法。 这次,我们隐式调用了该方法。
这是我们运行示例时得到的。
C# 对象初始化器
对象初始化程序让我们在创建时将值分配给对象的任何可访问字段或属性,而无需调用构造函数。 属性或字段在{}
括号内分配。 另外,我们可以为构造函数指定参数,也可以省略参数。
Program.cs
在示例中,我们使用对象初始化程序语法创建一个新用户。
我们定义一个空的构造函数。
我们有两个属性:Name
和Occupation
。
我们将值分配给{}
括号中的属性。
C# 类常量
C# 可以创建类常量。 这些常量不属于具体对象。 他们属于阶级。 按照约定,常量用大写字母表示。
Program.cs
我们有一个带有PI
常量的Math
类。
const
关键字用于定义常数。 public
关键字使它可以在类的主体之外访问。
运行示例,我们看到此输出。
C# 继承
继承是使用已经定义的类形成新类的方法。 新形成的类称为派生的类,我们派生的类称为基类。 继承的重要好处是代码重用和降低程序的复杂性。 派生类(后代)将覆盖或扩展基类(祖先)的功能。
Program.cs
在此程序中,我们有两个类。 基类Being
和派生的Human
类。 派生类继承自基类。
在 C# 中,我们使用冒号(:)运算符创建继承关系。
我们实例化派生的Human
类。
我们可以看到两个构造函数都被调用了。 首先,调用基类的构造函数,然后调用派生类的构造函数。
接下来是一个更复杂的示例。
Program.cs
我们有四个班。 继承层次更加复杂。 Human
和Animal
类继承自Being
类。 Dog 类直接继承自Animal
类,并间接继承自Being
类。 我们还介绍了static
变量的概念。
我们定义一个static
变量。 静态成员是类的所有实例共享的成员。
每次实例化Being
类时,我们将 count 变量增加一。 这样,我们就可以跟踪创建的实例数。
Animal
继承自Being
,Dog
继承自Animal
。 Dog
也间接继承自Being
。
我们从Human
和Dog
类创建实例。 我们称为 Dog 对象的GetCount()
方法。
Human
调用两个构造函数。 Dog
调用三个构造函数。 有两个实例化的存在。
我们使用base
关键字显式调用父级的构造函数。
Program.cs
我们有两个类:Shape
类和Circle
类。 Shape
类是几何形状的基类。 我们可以在此类中加入一些常见形状的共同点,例如x
和y
坐标。
Shape
类具有两个构造函数。 第一个是默认构造函数。 第二个参数有两个参数:x,y 坐标。
这是Circle
类的构造函数。 此构造函数启动r
成员并调用父级的第二个构造函数,并向其传递x
和y
坐标。 如果不使用base
关键字显式调用构造函数,则将调用Shape
类的默认构造函数。
这是示例的输出。
C# 抽象类和方法
抽象类无法实例化。 如果一个类至少包含一个抽象方法,则也必须将其声明为抽象方法。 抽象方法无法实现; 他们只是声明方法的签名。 当我们从抽象类继承时,所有抽象方法都必须由派生类实现。 此外,必须以较少受限制的可见性声明这些方法。
与接口不同,抽象类可能具有完全实现的方法,并且可能具有定义的成员字段。 因此,抽象类可以提供部分实现。 程序员经常将一些通用功能放入抽象类中。 这些抽象类随后会被子类化以提供更具体的实现。 例如,Qt 图形库具有QAbstractButton
,它是按钮小部件的抽象基类,提供按钮所共有的功能。 按钮Q3Button
,QCheckBox
,QPushButton
,QRadioButton
和QToolButton
都从此基本抽象类继承。
正式地说,抽象类用于强制执行协议。 协议是所有实现对象都必须支持的一组操作。
Program.cs
我们有一个抽象基类Drawing
。 该类定义两个成员字段,定义一个方法并声明一个方法。 一种方法是抽象的,另一种是完全实现的。 Drawing
类是抽象的,因为我们无法绘制它。 我们可以画一个圆,一个点或一个正方形。 Drawing
类对我们可以绘制的对象具有一些通用功能。
我们使用abstract
关键字定义一个抽象类。
抽象方法之前还带有abstract
关键字。
圆是Drawing
类的子类。 它必须实现抽象的Area()
方法。
当我们实现Area()
方法时,必须使用override
关键字。 这样,我们通知编译器我们将覆盖现有的(继承的)方法。
C# 部分类
使用partial
关键字,可以将类的定义拆分到同一名称空间中的几个部分中。 该类也可以在多个文件中定义。
当使用非常大的代码库时可以使用部分类,这些代码库可以拆分为较小的单元。 局部类也与自动代码生成器一起使用。
Program.cs
在示例中,我们将Worker
类定义为两部分。 这些部分由编译器连接在一起以形成最终类。
这是输出。