面向对象编程(OOP)是一种编程范式,它使用“对象”来设计软件。在 Java 中,面向对象编程的四个基本原则是封装、继承、多态和抽象。每个原则都有其特定的目标,帮助开发者构建更加模块化、可维护和可扩展的代码。
封装
封装是指将数据(属性)和行为(方法)捆绑在一起,并隐藏对象的具体实现细节。封装的好处是可以保护对象内部状态的完整性,防止外部代码随意修改对象的状态,从而提高了代码的安全性和可靠性。
代码示例
假设我们有一个 BankAccount
类,我们需要保护账户余额不被外部直接修改。
1public class BankAccount {
2 private double balance;
3
4 public BankAccount(double initialBalance) {
5 if (initialBalance > 0.0) {
6 balance = initialBalance;
7 } else {
8 throw new IllegalArgumentException("Initial balance must be greater than zero.");
9 }
10 }
11
12 public double getBalance() {
13 return balance;
14 }
15
16 public void deposit(double amount) {
17 if (amount > 0.0) {
18 balance += amount;
19 } else {
20 throw new IllegalArgumentException("Deposit amount must be positive.");
21 }
22 }
23
24 public void withdraw(double amount) {
25 if (amount > 0.0 && amount <= balance) {
26 balance -= amount;
27 } else {
28 throw new IllegalArgumentException("Withdrawal amount must be positive and less than or equal to the balance.");
29 }
30 }
31}
这里我们通过将 balance
属性声明为 private
来隐藏其内部实现,并提供了公共的方法(getBalance()
、deposit()
和 withdraw()
)来访问和修改余额。
继承
继承是指一个类(子类)可以继承另一个类(父类)的属性和方法。继承使得子类可以在复用父类功能的基础上添加或覆盖特定的行为,从而实现代码的重用。
代码示例
继续使用 BankAccount
类,我们可以创建一个 SavingsAccount
子类来继承 BankAccount
的功能,并添加一些特定的功能,比如利息计算。
1public class SavingsAccount extends BankAccount {
2 private double interestRate;
3
4 public SavingsAccount(double initialBalance, double interestRate) {
5 super(initialBalance);
6 this.interestRate = interestRate;
7 }
8
9 public void addInterest() {
10 double interest = getBalance() * interestRate;
11 deposit(interest);
12 }
13}
在这里,SavingsAccount
继承了 BankAccount
的所有属性和方法,并添加了一个 addInterest()
方法来计算并添加利息。
多态
多态是指一个类的对象可以被当作其父类或其他接口类型来使用。多态允许我们编写更通用的代码,可以处理多种类型的对象。
代码示例
我们可以在 BankAccount
类中添加一个 calculateInterest()
方法,并让 SavingsAccount
类重写这个方法。
1public abstract class BankAccount {
2 private double balance;
3
4 public BankAccount(double initialBalance) {
5 if (initialBalance > 0.0) {
6 balance = initialBalance;
7 } else {
8 throw new IllegalArgumentException("Initial balance must be greater than zero.");
9 }
10 }
11
12 public double getBalance() {
13 return balance;
14 }
15
16 public void deposit(double amount) {
17 if (amount > 0.0) {
18 balance += amount;
19 } else {
20 throw new IllegalArgumentException("Deposit amount must be positive.");
21 }
22 }
23
24 public void withdraw(double amount) {
25 if (amount > 0.0 && amount <= balance) {
26 balance -= amount;
27 } else {
28 throw new IllegalArgumentException("Withdrawal amount must be positive and less than or equal to the balance.");
29 }
30 }
31
32 public abstract double calculateInterest();
33}
34
35public class SavingsAccount extends BankAccount {
36 private double interestRate;
37
38 public SavingsAccount(double initialBalance, double interestRate) {
39 super(initialBalance);
40 this.interestRate = interestRate;
41 }
42
43 @Override
44 public double calculateInterest() {
45 return getBalance() * interestRate;
46 }
47}
现在,我们可以在父类中定义一个抽象方法 calculateInterest()
,然后在子类中实现它。这样,我们就可以编写一个通用的方法来处理不同的账户类型。
1public void printInterest(BankAccount account) {
2 System.out.println("Interest: " + account.calculateInterest());
3}
4
5public static void main(String[] args) {
6 BankAccount savings = new SavingsAccount(1000, 0.05);
7 printInterest(savings); // 输出 "Interest: 50.0"
8}
抽象
抽象是指通过抽象类或接口来定义一组通用的行为,而不关心具体实现细节。抽象类或接口可以被其他类继承或实现,从而达到代码复用的目的。
代码示例
我们可以创建一个 Account
接口来定义所有账户都应该有的基本行为,然后让 BankAccount
和 SavingsAccount
实现这个接口。
1public interface Account {
2 void deposit(double amount);
3 void withdraw(double amount);
4 double getBalance();
5}
6
7public class BankAccount implements Account {
8 private double balance;
9
10 public BankAccount(double initialBalance) {
11 if (initialBalance > 0.0) {
12 balance = initialBalance;
13 } else {
14 throw new IllegalArgumentException("Initial balance must be greater than zero.");
15 }
16 }
17
18 @Override
19 public void deposit(double amount) {
20 if (amount > 0.0) {
21 balance += amount;
22 } else {
23 throw new IllegalArgumentException("Deposit amount must be positive.");
24 }
25 }
26
27 @Override
28 public void withdraw(double amount) {
29 if (amount > 0.0 && amount <= balance) {
30 balance -= amount;
31 } else {
32 throw new IllegalArgumentException("Withdrawal amount must be positive and less than or equal to the balance.");
33 }
34 }
35
36 @Override
37 public double getBalance() {
38 return balance;
39 }
40}
41
42public class SavingsAccount extends BankAccount implements Account {
43 private double interestRate;
44
45 public SavingsAccount(double initialBalance, double interestRate) {
46 super(initialBalance);
47 this.interestRate = interestRate;
48 }
49
50 public void addInterest() {
51 double interest = getBalance() * interestRate;
52 deposit(interest);
53 }
54
55 @Override
56 public double calculateInterest() {
57 return getBalance() * interestRate;
58 }
59}
通过这种方式,我们可以确保所有的账户类型都遵循相同的接口,从而更容易地交换和组合它们。
合理化的使用建议
-
封装
- 尽可能将类的属性设为
private
,并通过公共方法暴露必要的访问途径。 - 使用 getter 和 setter 方法来增加数据访问的安全性,比如添加校验逻辑。
- 尽可能将类的属性设为
-
继承
- 继承应仅用于共享行为和状态的类之间。
- 避免过度继承,因为这可能会导致代码难以维护。
-
多态
- 利用多态可以让代码更具通用性和灵活性。
- 通过抽象类或接口定义通用行为,并在具体实现中进行定制。
-
抽象
- 使用抽象类或接口来定义通用的行为和规范。
- 抽象类可以提供部分实现,而接口则只定义方法签名。
实际开发过程中的注意点:
-
过度封装
过度封装可能会导致代码变得过于复杂。适当暴露一些属性,如通过final
修饰符,可以让代码更简洁。 -
过度继承
过度继承可能会导致类层次结构过于复杂,从而影响代码的可读性和可维护性。 -
接口污染
在设计接口时,要确保接口的功能单一,避免接口过于庞大,导致实现类需要实现大量无关的方法。 -
滥用抽象
抽象类和接口应该用来定义行为,而不是具体实现。避免为了抽象而抽象,导致设计复杂度上升。
通过合理地应用面向对象编程的原则,我们可以编写出更加健壮、易于维护和扩展的代码。在实际项目中,我们应该根据具体的需求灵活运用这些原则,以达到最佳的设计效果。
标签:BankAccount,Java,double,void,多态,amount,面向对象编程,balance,public From: https://blog.csdn.net/liangzai215/article/details/141831481