设计模式是软件开发中的经典解决方案,指导我们如何设计更优雅、灵活和可维护的代码。为了理解设计模式的核心思想,让我们从这六大原则出发,看看它们如何应用于日常编程中。在这篇文章中,我将为你揭示这些原则的精髓,并通过幽默、有趣的例子和Java代码来展示它们的应用。
1、单一职责原则 (Single Responsibility Principle, SRP)
想象一下,你在办公室里有一个多功能打印机,它可以打印、复印、扫描、发送传真等。然而,当其中一个功能出现问题时,整个设备都无法使用。同样,一个类也应该只负责一项任务,而不是承担过多的职责。这就是单一职责原则的核心。 Java代码示例:
// Bad example
class Employee {
public void calculateSalary() {
// Calculate salary
}
public void generateReport() {
// Generate report
}
}
// Good example
class Employee {
public void calculateSalary() {
// Calculate salary
}
}
class ReportGenerator {
public void generateReport(Employee employee) {
// Generate report
}
}
在错误的例子中,Employee类负责计算薪水和生成报告,这违反了单一职责原则。正确的例子将这两个功能分离到不同的类中,使得每个类都只负责一项任务。
2、开放封闭原则 (Open/Closed Principle, OCP)
开放封闭原则告诉我们,软件实体(类、模块、函数等)应该对扩展开放,对修改封闭。换句话说,我们应该能够在不修改原有代码的基础上,为其添加新的功能。 Java代码示例:
// Bad example
class Rectangle {
public int width;
public int height;
}
class AreaCalculator {
public int calculateArea(Rectangle[] rectangles) {
int area = 0;
for (Rectangle rectangle : rectangles) {
area += rectangle.width * rectangle.height;
}
return area;
}
}
// Good example
abstract class Shape {
public abstract int calculateArea();
}
class Rectangle extends Shape {
public int width;
public int height;
@Override
public int calculateArea() {
return width * height;
}
}
class Circle extends Shape {
public int radius;
@Override
public int calculateArea() {
return (int) (Math.PI * radius * radius);
}
}
class AreaCalculator {
public int calculateArea(Shape[] shapes) {
int area = 0;
for (Shape shape : shapes) {
area += shape.calculateArea();
}
return area;
}
}
在错误的例子中,如果我们要计算其他形状的面积,就需要修改AreaCalculator类。而在正确的例子中,我们通过引入抽象类Shape来实现开放封闭原则。现在,我们可以添加新的形状类,如Triangle,而无需修改AreaCalculator类。
3、里氏替换原则 (Liskov Substitution Principle, LSP)
里氏替换原则讲的是子类应该能够替换其基类,并且不会改变程序的正确性。换句话说,如果一个类在某个场景下可以使用,那么它的子类也应该适用于同样的场景。 Java代码示例:
class Bird {
public void fly() {
// Fly
}
}
class Penguin extends Bird {
@Override
public void fly() {
throw new UnsupportedOperationException("Penguins can't fly!");
}
}
// Good example
class Bird {
public void fly() {
// Fly
}
}
class FlightlessBird extends Bird {
@Override
public void fly() {
throw new UnsupportedOperationException("Flightless birds can't fly!");
}
}
class Penguin extends FlightlessBird {
}
在错误的例子中,Penguin类违反了里氏替换原则,因为它不能像其他鸟类那样飞行。正确的例子引入了一个新的中间类FlightlessBird,使得Penguin类不再违反LSP。
4、接口隔离原则 (Interface Segregation Principle, ISP)
接口隔离原则告诉我们,不应该强迫一个类实现它不需要的接口。换句话说,我们应该将接口划分为更小的、更具体的接口,以满足不同的需求。 Java代码示例:
// Bad example
interface Animal {
void eat();
void swim();
void fly();
}
class Shark implements Animal {
@Override
public void eat() {
// Eat
}
@Override
public void swim() {
// Swim
}
@Override
public void fly() {
throw new UnsupportedOperationException("Sharks can't fly!");
}
}
// Good example
interface Swimmer {
void swim();
}
interface Eater {
void eat();
}
interface Flyer {
void fly();
}
class Shark implements Swimmer, Eater {
@Override
public void eat() {
// Eat
}
@Override
public void swim() {
// Swim
}
}
在错误的例子中,Animal接口包含了不相关的方法,迫使Shark实现不需要的fly方法。正确的例子将接口划分为更小的接口,遵循了接口隔离原则。
5、依赖倒置原则 (Dependency Inversion Principle, DIP)
依赖倒置原则要求我们依赖抽象而不是具体的实现。这样可以使得系统更加灵活,易于扩展和修改。 Java代码示例:
// Bad example
class MySQLDatabase {
public void saveData(String data) {
// Save data
// Save data to MySQL database
}
}
class DataManager {
private MySQLDatabase database;
public DataManager(MySQLDatabase database) {
this.database = database;
}
public void saveData(String data) {
database.saveData(data);
}
}
// Good example
interface Database {
void saveData(String data);
}
class MySQLDatabase implements Database {
@Override
public void saveData(String data) {
// Save data to MySQL database
}
}
class MongoDBDatabase implements Database {
@Override
public void saveData(String data) {
// Save data to MongoDB database
}
}
class DataManager {
private Database database;
public DataManager(Database database) {
this.database = database;
}
public void saveData(String data) {
database.saveData(data);
}
}
在错误的例子中,DataManager类依赖于具体的MySQLDatabase实现,这使得代码难以修改和扩展。正确的例子引入了一个Database接口,使得DataManager依赖于抽象而不是具体实现。这样,我们可以轻松地切换到其他数据库,例如MongoDBDatabase。
6、合成复用原则 (Composite Reuse Principle, CRP)
合成复用原则强调通过组合和聚合来实现代码的复用,而不是通过继承。这样可以使得系统更加灵活,降低了类之间的耦合。 Java代码示例:
// Bad example
class Engine {
public void start() {
// Start engine
}
}
class Car extends Engine {
public void drive() {
start();
// Drive
}
}
// Good example
class Engine {
public void start() {
// Start engine
}
}
class Car {
private Engine engine;
public Car(Engine engine) {
this.engine = engine;
}
public void drive() {
engine.start();
// Drive
}
}
在错误的例子中,Car类通过继承Engine来实现复用。然而,这使得代码难以修改和扩展。正确的例子通过组合的方式,将Engine作为一个成员变量,从而遵循了合成复用原则。 通过遵循这些设计模式哲学中的六大原则,我们可以编写出更优雅、灵活和可维护的代码。在实际项目中,我们可能会根据具体情况灵活运用这些原则,以实现最佳的设计。希望这篇文章能帮助你更好地理解设计模式,并在实际编程过程中运用得心应手。
标签:六大,database,int,void,public,哲学,设计模式,data,class From: https://blog.51cto.com/u_13616584/6237836