C#中级教程(2)——走进 C# 面向对象编程:从基础到进阶的深度探索
一、为什么选择面向对象编程
在软件开发的演进过程中,随着程序规模和复杂度的不断增加,传统的编程方式逐渐暴露出局限性。面向对象编程应运而生,它就像是一位智慧的组织者,将程序中的功能进行模块化划分。每个模块各司其职,提供特定的功能,而且这些模块相互独立。这种模块化编程极大地提升了代码的可维护性和可扩展性,为开发者带来了诸多便利。例如,在一个电商系统中,用户管理、订单处理、商品展示等功能模块可以各自独立开发、测试和维护,当需要对某个模块进行修改或升级时,不会影响其他模块的正常运行。同时,它还大大增加了代码的重用机会,提高了开发效率,减少了重复劳动。
二、什么是面向对象编程
面向对象编程(OOP)是一种编程范式,它将现实世界中的事物抽象成程序中的对象。简单来说,OOP 把程序看作是由多个相互协作的对象组成,每个对象都有自己的状态(数据)和行为(方法)。这就好比在一个游戏中,每个角色都有自己的生命值、等级等状态,同时具备攻击、防御、移动等行为。与结构化编程相比,OOP 更符合人类对现实世界的认知方式,使得代码的结构更加清晰、易于理解和维护。
三、类与对象:面向对象的基石
(一)类的定义与结构
类是创建对象的模板,它定义了对象的属性和行为。在 C# 中,类的定义包含访问修饰符、类名、成员变量(数据成员)和成员方法(函数成员)等部分。例如:
public class Employee
{// 数据成员(字段)private string name;private int age;private decimal salary;// 函数成员(方法)public void DisplayInfo(){Console.WriteLine($"姓名:{name},年龄:{age},薪资:{salary}");}// 构造函数public Employee(string empName, int empAge, decimal empSalary){name = empName;age = empAge;salary = empSalary;}
}
在这个例子中,Employee
类定义了员工的基本信息和显示信息的方法。name
、age
和salary
是数据成员,用于存储员工的具体信息;DisplayInfo
方法用于展示员工信息;构造函数Employee
用于在创建对象时初始化员工的属性。
(二)对象的创建与使用
对象是类的实例,通过new
关键字创建。
访问权限 | 类 | 包 | 子类 | 其他包 |
---|---|---|---|---|
public | √ | √ | √ | √ |
protect | √ | √ | √ | |
default | √ | √ | ||
private | √ |
例如:
Employee emp1 = new Employee("张三", 25, 5000m);
emp1.DisplayInfo();
上述代码创建了一个Employee
类的对象emp1
,并调用DisplayInfo
方法输出员工信息。创建对象的过程就像是根据图纸(类)制造出一个具体的产品(对象),每个对象都有自己独立的属性值。
四、构造函数:对象的初始化器
构造函数是一种特殊的方法,用于在创建对象时初始化对象的状态。它与类名相同,没有返回类型。构造函数可以重载,以满足不同的初始化需求。例如:
public class Circle
{private double radius;// 无参构造函数public Circle(){radius = 1.0;}// 有参构造函数public Circle(double r){if (r > 0){radius = r;}else{radius = 1.0;}}public double CalculateArea(){return Math.PI * radius * radius;}
}
在Circle
类中,定义了无参构造函数和有参构造函数。无参构造函数将圆的半径初始化为 1.0,有参构造函数则根据传入的半径值进行初始化,并对半径进行合法性检查。
五、属性:灵活的数据访问方式
属性是一种特殊的成员,它提供了对类的字段进行访问的方式,同时可以在访问过程中进行数据验证和其他逻辑处理。属性包含get
访问器和set
访问器。例如:
public class Person
{private int age;public int Age{get{return age;}set{if (value >= 0 && value <= 120){age = value;}}}
}
在Person
类中,Age
属性对age
字段进行了封装。通过get
访问器获取age
的值,通过set
访问器设置age
的值,并在设置时进行了年龄范围的验证。
属性还可以设置为只读或只写,以及使用自动实现属性。例如:
// 只读属性
public string Name { get; private set; }// 自动实现属性
public int ID { get; set; }
六、匿名类型:便捷的数据定义
在 C# 中,可以使用var
关键字声明匿名类型。匿名类型允许在不定义显式类的情况下创建对象。例如:
var student = new { Name = "李四", Age = 20, Grade = "A" };
Console.WriteLine($"姓名:{student.Name},年龄:{student.Age},成绩:{student.Grade}");
匿名类型在需要临时创建一个简单对象,且不需要在其他地方复用该类型时非常方便。
七、内存管理:栈与堆的奥秘
在 C# 中,内存分为栈空间和堆空间。栈空间较小,但读取速度快,主要用于存储值类型数据和对象的引用。堆空间较大,但读取速度相对较慢,用于存储引用类型的数据。
值类型(如整数、bool
、struct
、char
、小数等)在内存中直接存储其数据值,而引用类型(如string
、数组、自定义类等)在内存中需要两段空间,一段存储实际数据(位于堆中),另一段存储指向该数据在堆中位置的引用(位于栈中)。例如:
int num = 10; // num是值类型,存储在栈中
string str = "Hello"; // str是引用类型,"Hello"存储在堆中,str的引用存储在栈中
C# 的垃圾回收器(GC)会自动管理内存,释放不再使用的对象所占用的内存空间,开发者无需手动管理内存,从而减少了内存泄漏等问题的发生。
八、面向对象编程的高级特性
(一)继承:代码复用的利器
继承是面向对象编程的重要特性之一,它允许一个类从另一个类中获取属性和方法。通过继承,子类可以复用父类的代码,同时还可以添加自己特有的属性和方法。例如:
public class Animal
{public string Name { get; set; }public void Eat(){Console.WriteLine($"{Name}正在吃东西。");}
}public class Dog : Animal
{public void Bark(){Console.WriteLine($"{Name}正在汪汪叫。");}
}
在这个例子中,Dog
类继承自Animal
类,Dog
类不仅拥有Animal
类的Name
属性和Eat
方法,还添加了自己的Bark
方法。
(二)多态:同一行为的不同表现
多态是指同一个方法在不同的对象上有不同的表现形式。C# 中实现多态的方式有两种:方法重载和方法重写。
- 方法重载:在同一个类中,定义多个同名但参数列表不同的方法。例如:
public class Calculator
{public int Add(int a, int b){return a + b;}public double Add(double a, double b){return a + b;}
}
在Calculator
类中,定义了两个Add
方法,一个用于整数相加,一个用于双精度浮点数相加。
2. 方法重写:在子类中重新定义父类中已经存在的虚方法。例如:
public class Shape
{public virtual double Area(){return 0;}
}public class Rectangle : Shape
{public double Width { get; set; }public double Height { get; set; }public override double Area(){return Width * Height;}
}
在这个例子中,Rectangle
类继承自Shape
类,并重写了Area
方法,以计算矩形的面积。
(三)接口:定义行为契约
接口是一种特殊的抽象类型,它定义了一组方法签名,但不包含方法的实现。类可以实现一个或多个接口,以表明它具备某些行为。例如:
public interface IPrintable
{void Print();
}public class Book : IPrintable
{public string Title { get; set; }public void Print(){Console.WriteLine($"书名:{Title}");}
}
在这个例子中,Book
类实现了IPrintable
接口,必须实现接口中定义的Print
方法。
九、实践案例:开发一个简单的图书馆管理系统
下面通过一个简单的图书馆管理系统案例,综合运用上述所学的面向对象编程知识。
// 定义书籍类
public class Book
{public string Title { get; set; }public string Author { get; set; }public bool IsAvailable { get; set; }public Book(string title, string author){Title = title;Author = author;IsAvailable = true;}
}// 定义图书馆类
public class Library
{private List<Book> books = new List<Book>();public void AddBook(Book book){books.Add(book);Console.WriteLine($"已添加书籍:{book.Title}");}public void BorrowBook(string title){Book book = books.Find(b => b.Title == title);if (book != null && book.IsAvailable){book.IsAvailable = false;Console.WriteLine($"已借阅书籍:{book.Title}");}else{Console.WriteLine($"抱歉,{title}不可借阅。");}}public void ReturnBook(string title){Book book = books.Find(b => b.Title == title);if (book != null &&!book.IsAvailable){book.IsAvailable = true;Console.WriteLine($"已归还书籍:{book.Title}");}else{Console.WriteLine($"无法归还,{title}可能未借阅或不存在。");}}
}
在上述代码中,Book
类表示书籍,包含书名、作者和是否可借阅的属性;Library
类表示图书馆,包含添加书籍、借阅书籍和归还书籍的方法。通过这些类的协同工作,可以实现一个简单的图书馆管理功能。
十、结语
在今后的学习和实践中,不断积累经验,深入探索 C# 的更多特性和应用场景,你将能够开发出更加复杂和高效的软件系统。希望你在编程的道路上不断前行,创造出更多优秀的作品!