原文: https://zetcode.com/lang/csharp/oopi/

在 C# 教程的这一部分中,我们将讨论 C# 中的面向对象编程。

共有三种广泛使用的编程范例:过程编程,函数编程和面向对象的编程。 C# 支持过程式编程和面向对象的编程。

OOP 定义

面向对象编程(OOP)是一种使用对象及其相互作用设计应用和计算机程序的编程范例。

OOP 中有一些基本的编程概念:

  • 抽象
  • 多态
  • 封装
  • 继承

抽象通过建模适合该问题的类来简化复杂的现实。 多态是将运算符或函数以不同方式用于不同数据输入的过程。 封装对其他对象隐藏了类的实现细节。 继承是一种使用已经定义的类形成新类的方法。

C# 对象

对象是 C# OOP 程序的基本构建块。 对象是数据和方法的组合。 数据和方法称为对象的成员。 在 OOP 程序中,我们创建对象。 这些对象通过方法进行通信。 每个对象都可以接收消息,发送消息和处理数据。

创建对象有两个步骤。 首先,我们定义一个类。 类是对象的模板。 它是一个蓝图,描述了类对象共享的状态和行为。 一个类可以用来创建许多对象。 在运行时从类创建的对象称为该特定类的实例。

Program.cs

  1. using System;
  2. namespace Being
  3. {
  4. class Being {}
  5. class Program
  6. {
  7. static void Main(string[] args)
  8. {
  9. var b = new Being();
  10. Console.WriteLine(b);
  11. }
  12. }
  13. }

在第一个示例中,我们创建一个简单的对象。

  1. class Being {}

这是一个简单的类定义。 模板的主体为空。 它没有任何数据或方法。

  1. var b = new Being();

我们创建Being类的新实例。 为此,我们使用了new关键字。 b变量是创建对象的句柄。

  1. Console.WriteLine(b);

我们将对象打印到控制台以获取该对象的一些基本描述。 打印对象是什么意思? 实际上,当我们打印对象时,我们将其称为ToString()方法。 但是我们还没有定义任何方法。 这是因为创建的每个对象都继承自基本object。 它具有一些基本功能,可以在所有创建的对象之间共享。 其中之一是ToString()方法。

  1. $ dotnet run
  2. Being.Being

我们得到对象类名。

C# 对象属性

对象属性是捆绑在类实例中的数据。 对象属性称为实例变量或成员字段。 实例变量是在类中定义的变量,该类中的每个对象都有一个单独的副本。

Program.cs

  1. using System;
  2. namespace ObjectAttributes
  3. {
  4. class Person
  5. {
  6. public string name;
  7. }
  8. class Program
  9. {
  10. static void Main(string[] args)
  11. {
  12. var p1 = new Person();
  13. p1.name = "Jane";
  14. var p2 = new Person();
  15. p2.name = "Beky";
  16. Console.WriteLine(p1.name);
  17. Console.WriteLine(p2.name);
  18. }
  19. }
  20. }

在上面的 C# 代码中,我们有一个带有一个成员字段的Person类。

  1. class Person
  2. {
  3. public string name;
  4. }

我们声明一个名称成员字段。 public关键字指定可以在类块之外访问成员字段。

  1. var p1 = new Person();
  2. p1.name = "Jane";

我们创建Person类的实例,并将名称变量设置为"Jane"。 我们使用点运算符来访问对象的属性。

  1. var p2 = new Person();
  2. p2.name = "Beky";

我们创建Person类的另一个实例。 在这里,我们将变量设置为"Beky"

  1. Console.WriteLine(p1.name);
  2. Console.WriteLine(p2.name);

我们将变量的内容打印到控制台。

  1. $ dotnet run
  2. Jane
  3. Beky

我们看到了程序的输出。 Person类的每个实例都有一个单独的名称成员字段副本。

C# 方法

方法是在类主体内定义的函数。 它们用于通过对象的属性执行操作。 方法将模块化带入我们的程序。

在 OOP 范式的封装概念中,方法至关重要。 例如,我们的AccessDatabase类中可能有一个Connect()方法。 我们无需知道方法Connect()如何精确地连接到数据库。 我们只需要知道它用于连接数据库。 这对于划分编程中的职责至关重要,尤其是在大型应用中。

对象将状态和行为分组,方法代表对象的行为部分。

Program.cs

  1. using System;
  2. namespace Methods
  3. {
  4. class Circle
  5. {
  6. private int radius;
  7. public void SetRadius(int radius)
  8. {
  9. this.radius = radius;
  10. }
  11. public double Area()
  12. {
  13. return this.radius * this.radius * Math.PI;
  14. }
  15. }
  16. class Program
  17. {
  18. static void Main(string[] args)
  19. {
  20. var c = new Circle();
  21. c.SetRadius(5);
  22. Console.WriteLine(c.Area());
  23. }
  24. }
  25. }

在代码示例中,我们有一个Circle类。 我们定义了两种方法。

  1. private int radius;

我们只有一个成员字段。 它是圆的半径。 private关键字是访问说明符。 它表明变量仅限于外部世界。 如果要从外部修改此变量,则必须使用公共可用的SetRadius()方法。 这样我们可以保护我们的数据。

  1. public void SetRadius(int radius)
  2. {
  3. this.radius = radius;
  4. }

这是SetRadius()方法。 this变量是一个特殊变量,我们用它来访问方法中的成员字段。 this.radius是实例变量,而半径是局部变量,仅在SetRadius()方法内部有效。

  1. var c = new Circle();
  2. c.SetRadius(5);

我们创建Circle类的实例,并通过在圆对象上调用SetRadius()方法来设置其半径。 我们使用点运算符来调用该方法。

  1. public double Area()
  2. {
  3. return this.radius * this.radius * Math.PI;
  4. }

Area()方法返回圆的面积。 Math.PI是内置常数。

  1. $ dotnet run
  2. 78.5398163397448

运行该示例可得出此结果。

C# 访问修饰符

访问修饰符设置方法和成员字段的可见性。 C# 具有四个基本访问修饰符:publicprotectedprivateinternal。 可以从任何地方访问public成员。 protected成员只能在类本身内部以及继承的和父类访问。 private成员仅限于包含类型,例如仅在其类或接口内。 可以从同一程序集(exe 或 DLL)中访问internal成员。

修饰符还有两种组合:protected internalprivate protectedprotected 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

  1. using System;
  2. namespace AccessModifiers
  3. {
  4. class Person
  5. {
  6. public string name;
  7. private int age;
  8. public int GetAge()
  9. {
  10. return this.age;
  11. }
  12. public void SetAge(int age)
  13. {
  14. this.age = age;
  15. }
  16. }
  17. class Program
  18. {
  19. static void Main(string[] args)
  20. {
  21. var p = new Person();
  22. p.name = "Jane";
  23. p.SetAge(17);
  24. Console.WriteLine("{0} is {1} years old",
  25. p.name, p.GetAge());
  26. }
  27. }
  28. }

在上面的程序中,我们有两个成员字段。 一个被宣布为公开,另一个被宣布为私有。

  1. public int GetAge()
  2. {
  3. return this.age;
  4. }

如果成员字段是private,则访问它的唯一方法是通过方法。 如果要在类外部修改属性,则必须将方法声明为public。 这是数据保护的重要方面。

  1. public void SetAge(int age)
  2. {
  3. this.age = age;
  4. }

SetAge()方法使我们能够从类定义之外更改private年龄变量。

  1. var p = new Person();
  2. p.name = "Jane";

我们创建Person类的新实例。 因为名称属性是public,所以我们可以直接访问它。 但是,不建议这样做。

  1. p.SetAge(17);

SetAge()方法修改年龄成员字段。 由于已声明private,因此无法直接访问或修改。

  1. Console.WriteLine("{0} is {1} years old",
  2. p.name, p.GetAge());

最后,我们访问两个成员以构建一个字符串。

  1. $ dotnet run
  2. Jane is 17 years old

运行示例将给出此输出。

具有private访问修饰符的成员字段不被派生类继承。

Program.cs

  1. using System;
  2. namespace Protected
  3. {
  4. class Base
  5. {
  6. public string name = "Base";
  7. protected int id = 5323;
  8. private bool isDefined = true;
  9. }
  10. class Derived : Base
  11. {
  12. public void info()
  13. {
  14. Console.WriteLine("This is Derived class");
  15. Console.WriteLine("Members inherited");
  16. Console.WriteLine(this.name);
  17. Console.WriteLine(this.id);
  18. // Console.WriteLine(this.isDefined);
  19. }
  20. }
  21. class Program
  22. {
  23. static void Main(string[] args)
  24. {
  25. var derived = new Derived();
  26. derived.info();
  27. }
  28. }
  29. }

在前面的程序中,我们有一个Derived类,该类继承自Base类。 Base类具有三个成员字段。 全部具有不同的访问修饰符。 isDefined成员不继承。 private修饰符可以防止这种情况。

  1. class Derived : Base

Derived继承自Base类。 要从另一个类继承,我们使用冒号(:)运算符。

  1. Console.WriteLine(this.name);
  2. Console.WriteLine(this.id);
  3. // Console.WriteLine(this.isDefined);

publicprotected成员由Derived类继承。 可以访问它们。 private成员未继承。 访问成员字段的行被注释。 如果我们取消注释该行,则代码将无法编译。

  1. $ dotnet run
  2. Program.cs(9,22): warning CS0414: The field 'Base.isDefined' is assigned but its value
  3. is never used [C:\Users\Jano\Documents\csharp\tutorial\oop\Protected\Protected.csproj]
  4. This is Derived class
  5. Members inherited
  6. Base
  7. 5323

运行程序,我们收到此输出。

C# 构造器

构造器是一种特殊的方法。 创建对象时会自动调用它。 构造器不返回值。 构造器的目的是初始化对象的状态。 构造器与类具有相同的名称。 构造器是方法,因此它们也可以重载。

构造器不能被继承。 它们按继承顺序被调用。 如果我们不为类编写任何构造器,则 C# 提供一个隐式默认构造器。 如果提供任何类型的构造器,则不提供默认值。

Program.cs

  1. using System;
  2. namespace Constructor
  3. {
  4. class Being
  5. {
  6. public Being()
  7. {
  8. Console.WriteLine("Being is created");
  9. }
  10. public Being(string being)
  11. {
  12. Console.WriteLine("Being {0} is created", being);
  13. }
  14. }
  15. class Program
  16. {
  17. static void Main(string[] args)
  18. {
  19. new Being();
  20. new Being("Tom");
  21. }
  22. }
  23. }

我们有一个Being类。 此类具有两个构造器。 第一个不带参数; 第二个采用一个参数。

  1. public Being(string being)
  2. {
  3. Console.WriteLine("Being {0} is created", being);
  4. }

此构造器采用一个字符串参数。

  1. new Being();

创建Being类的实例。 这次,在创建对象时调用没有参数的构造器。

  1. $ dotnet run
  2. Being is created
  3. Being Tom is created

这是程序的输出。

在下一个示例中,我们初始化类的数据成员。 变量的初始化是构造器的典型工作。

Program.cs

  1. using System;
  2. namespace Constructor2
  3. {
  4. class MyFriend
  5. {
  6. private DateTime born;
  7. private string name;
  8. public MyFriend(string name, DateTime born)
  9. {
  10. this.name = name;
  11. this.born = born;
  12. }
  13. public void Info()
  14. {
  15. Console.WriteLine("{0} was born on {1}",
  16. this.name, this.born.ToShortDateString());
  17. }
  18. }
  19. class Program
  20. {
  21. static void Main(string[] args)
  22. {
  23. var name = "Lenka";
  24. var born = new DateTime(1990, 3, 5);
  25. var friend = new MyFriend(name, born);
  26. friend.Info();
  27. }
  28. }
  29. }

我们有一个带有数据成员和方法的MyFriend类。

  1. private DateTime born;
  2. private string name;

类定义中有两个私有变量。

public MyFriend(string name, DateTime born)
{
    this.name = name;
    this.born = born;
}

在构造器中,我们启动两个数据成员。 this变量是用于引用对象变量的处理器。

var friend = new MyFriend(name, born);
friend.Info();

我们创建带有两个参数的MyFriend对象。 然后我们调用对象的Info()方法。

$ dotnet run
Lenka was born on 3/5/1990

这是输出。

C# 构造器链接

构造器链接是类从构造器调用另一个构造器的能力。 要从同一类调用另一个构造器,我们使用this关键字。

Program.cs

using System;

namespace ConstructorChaining
{
    class Circle
    {
        public Circle(int radius)
        {
            Console.WriteLine("Circle, r={0} is created", radius);
        }

        public Circle() : this(1)
        {

        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            new Circle(5);
            new Circle();
        }
    }
}

我们有一个Circle类。 该类具有两个构造器。 一种采用一个参数,一种不采用任何参数。

public Circle(int radius)
{
    Console.WriteLine("Circle, r={0} is created", radius);
}

此构造器采用一个参数-radius

public Circle() : this(1)
{

}

这是没有参数的构造器。 它只是简单地调用另一个构造器,并为其提供默认半径 1。

$ dotnet run
Circle, r=5 is created
Circle, r=1 is created

This is the output.

C# ToString方法

每个对象都有一个ToString()方法。 它返回人类可读的对象表示形式。 默认实现返回Object类型的标准名称。 请注意,当我们使用对象作为参数调用Console.WriteLine()方法时,将调用ToString()

Program.cs

using System;

namespace ToStringMethod
{
    class Being
    {
        public override string ToString()
        {
            return "This is Being class";
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var b = new Being();
            var o = new Object();

            Console.WriteLine(o.ToString());
            Console.WriteLine(b.ToString());
            Console.WriteLine(b);
        }
    }
}

我们有一个Being类,其中我们重写了ToString()方法的默认实现。

public override string ToString()
{
    return "This is Being class";
}

创建的每个类都从基object继承。 ToString()方法属于此对象类。 我们使用override关键字来通知我们正在覆盖方法。

var b = new Being();
var o = new Object();

我们创建一个自定义对象和一个内置对象。

Console.WriteLine(o.ToString());
Console.WriteLine(b.ToString());

我们在这两个对象上调用ToString()方法。

Console.WriteLine(b);

正如我们之前指定的,将对象作为Console.WriteLine()的参数将调用其ToString()方法。 这次,我们隐式调用了该方法。

$ dotnet run
System.Object
This is Being class
This is Being class

这是我们运行示例时得到的。

C# 对象初始化器

对象初始化器让我们在创建时将值分配给对象的任何可访问字段或属性,而无需调用构造器。 属性或字段在{}括号内分配。 另外,我们可以为构造器指定参数,也可以省略参数。

Program.cs

using System;

namespace ObjectInitializers
{
    class User
    {
        public User() {}

        public string Name { set; get; }
        public string Occupation { set; get; }

        public override string ToString()
        {
            return $"{Name} is a {Occupation}";
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var u = new User { Name = "John Doe", Occupation = "gardener" };
            Console.WriteLine(u);
        }
    }
}

在示例中,我们使用对象初始化器语法创建一个新用户。

public User() {}

我们定义一个空的构造器。

public string Name { set; get; }
public string Occupation { set; get; }

我们有两个属性:NameOccupation

var u = new User { Name = "John Doe", Occupation = "gardener" };

我们将值分配给{}括号中的属性。

$ dotnet run
John Doe is a gardener

This is the output.

C# 类常量

C# 可以创建类常量。 这些常量不属于具体对象。 他们属于类。 按照约定,常量用大写字母表示。

Program.cs

using System;

namespace ClassConstants
{
    class Math
    {
        public const double PI = 3.14159265359;
    }

    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine(Math.PI);
        }
    }
}

我们有一个带有PI常量的Math类。

public const double PI = 3.14159265359;

const关键字用于定义常数。 public关键字使它可以在类的主体之外访问。

$ dotnet run
3.14159265359

运行示例,我们看到此输出。

C# 继承

继承是使用已经定义的类形成新类的方法。 新形成的类称为派生的类,我们派生的类称为基类。 继承的重要好处是代码重用和降低程序的复杂性。 派生类(后代)将覆盖或扩展基类(祖先)的功能。

Program.cs

using System;

namespace Inheritance
{
    class Being
    {
        public Being()
        {
            Console.WriteLine("Being is created");
        }
    }

    class Human : Being
    {
        public Human()
        {
            Console.WriteLine("Human is created");
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            new Human();
        }
    }
}

在此程序中,我们有两个类。 基类Being和派生的Human类。 派生类继承自基类。

class Human : Being

在 C# 中,我们使用冒号(:)运算符创建继承关系。

new Human();

我们实例化派生的Human类。

$ dotnet run
Being is created
Human is created

我们可以看到两个构造器都被调用了。 首先,调用基类的构造器,然后调用派生类的构造器。

接下来是一个更复杂的示例。

Program.cs

using System;

namespace Inheritance2
{
    class Being
    {
        static int count = 0;

        public Being()
        {
            count++;
            Console.WriteLine("Being is created");
        }

        public void GetCount()
        {
            Console.WriteLine("There are {0} Beings", count);
        }
    }

    class Human : Being
    {
        public Human()
        {
            Console.WriteLine("Human is created");
        }
    }

    class Animal : Being
    {
        public Animal()
        {
            Console.WriteLine("Animal is created");
        }
    }

    class Dog : Animal
    {
        public Dog()
        {
            Console.WriteLine("Dog is created");
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            new Human();

            var dog = new Dog();
            dog.GetCount();
        }
    }
}

我们有四个类。 继承层次更加复杂。 HumanAnimal类继承自Being类。 Dog 类直接继承自Animal类,并间接继承自Being类。 我们还介绍了static变量的概念。

static int count = 0;

我们定义一个static变量。 静态成员是类的所有实例共享的成员。

Being()
{
    count++;
    Console.WriteLine("Being is created");
}

每次实例化Being类时,我们将count变量增加一。 这样,我们就可以跟踪创建的实例数。

class Animal : Being
...

class Dog : Animal
...

Animal继承自BeingDog继承自AnimalDog也间接继承自Being

new Human();
var dog = new Dog();
dog.GetCount();

我们从HumanDog类创建实例。 我们称为Dog对象的GetCount()方法。

$ dotnet run
Being is created
Human is created
Being is created
Animal is created
Dog is created
There are 2 Beings

Human调用两个构造器。 Dog调用三个构造器。 有两个实例化的存在。

我们使用base关键字显式调用父级的构造器。

Program.cs

using System;

namespace Shapes
{
    class Shape
    {
        protected int x;
        protected int y;

        public Shape()
        {
            Console.WriteLine("Shape is created");
        }

        public Shape(int x, int y)
        {
            this.x = x;
            this.y = y;
        }
    }

    class Circle : Shape
    {
        private int r;

        public Circle(int r, int x, int y) : base(x, y)
        {
            this.r = r;
        }

        public override string ToString()
        {
            return String.Format("Circle, r:{0}, x:{1}, y:{2}", r, x, y);
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var c = new Circle(2, 5, 6);
            Console.WriteLine(c);
        }
    }
}

我们有两个类:Shape类和Circle类。 Shape类是几何形状的基类。 我们可以在此类中加入一些常见形状的共同点,例如xy坐标。

public Shape()
{
    Console.WriteLine("Shape is created");
}

public Shape(int x, int y)
{
    this.x = x;
    this.y = y;
}

Shape类具有两个构造器。 第一个是默认构造器。 第二个参数有两个参数:xy坐标。

public Circle(int r, int x, int y) : base(x, y)
{
    this.r = r;
}

这是Circle类的构造器。 此构造器启动r成员并调用父级的第二个构造器,并向其传递xy坐标。 如果不使用base关键字显式调用构造器,则将调用Shape类的默认构造器。

$ dotnet run
Circle, r:2, x:5, y:6

这是示例的输出。

C# 抽象类和方法

抽象类无法实例化。 如果一个类至少包含一个抽象方法,则也必须将其声明为抽象方法。 抽象方法无法实现; 他们只是声明方法的签名。 当我们从抽象类继承时,所有抽象方法都必须由派生类实现。 此外,必须以较少受限制的可见性声明这些方法。

与接口不同,抽象类可能具有完全实现的方法,并且可能具有定义的成员字段。 因此,抽象类可以提供部分实现。 程序员经常将一些通用功能放入抽象类中。 这些抽象类随后会被子类化以提供更具体的实现。 例如,Qt 图形库具有QAbstractButton,它是按钮小部件的抽象基类,提供按钮所共有的功能。 按钮Q3ButtonQCheckBoxQPushButtonQRadioButtonQToolButton都从此基本抽象类继承。

正式地说,抽象类用于强制执行协议。 协议是所有实现对象都必须支持的一组操作。

Program.cs

using System;

namespace AbstractClass
{
    abstract class Drawing
    {
        protected int x = 0;
        protected int y = 0;

        public abstract double Area();

        public string GetCoordinates()
        {
            return string.Format("x: {0}, y: {1}", this.x, this.y);
        }
    }

    class Circle : Drawing
    {
        private int r;

        public Circle(int x, int y, int r)
        {
            this.x = x;
            this.y = y;
            this.r = r;
        }

        public override double Area()
        {
            return this.r * this.r * Math.PI;
        }

        public override string ToString()
        {
            return string.Format("Circle at x: {0}, y: {1}, radius: {2}",
                this.x, this.y, this.r);
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var c = new Circle(12, 45, 22);

            Console.WriteLine(c);
            Console.WriteLine("Area of circle: {0}", c.Area());
            Console.WriteLine(c.GetCoordinates());
        }
    }
}

我们有一个抽象基类Drawing。 该类定义两个成员字段,定义一个方法并声明一个方法。 一种方法是抽象的,另一种是完全实现的。 Drawing类是抽象的,因为我们无法绘制它。 我们可以画一个圆,一个点或一个正方形。 Drawing类对我们可以绘制的对象具有一些通用功能。

abstract class Drawing

我们使用abstract关键字定义一个抽象类。

public abstract double Area();

抽象方法之前还带有abstract关键字。

class Circle : Drawing

圆是Drawing类的子类。 它必须实现抽象的Area()方法。

public override double Area()
{
    return this.r * this.r * Math.PI;
}

当我们实现Area()方法时,必须使用override关键字。 这样,我们通知编译器我们将覆盖现有的(继承的)方法。

$ dotnet run
Circle at x: 12, y: 45, radius: 22
Area of circle: 1520.53084433746
x: 12, y: 45

This is the output of the program.

C# 部分类

使用partial关键字,可以将类的定义拆分到同一名称空间中的几个部分中。 该类也可以在多个文件中定义。

当使用非常大的代码库时可以使用部分类,这些代码库可以拆分为较小的单元。 局部类也与自动代码生成器一起使用。

Program.cs

using System;

namespace PartialClass
{
    partial class Worker
    {
        public string DoWork()
        {
            return "Doing work";
        }
    }

    partial class Worker
    {
        public string DoPause()
        {
            return "Pausing";
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            var worker = new Worker();

            Console.WriteLine(worker.DoWork());
            Console.WriteLine(worker.DoWork());
            Console.WriteLine(worker.DoPause());
        }
    }
}

在示例中,我们将Worker类定义为两部分。 这些部分由编译器连接在一起以形成最终类。

$ dotnet run
Doing work
Doing work
Pausing

这是输出。

这是 C# 中 OOP 描述的第一部分。