引子

从“把大象装进冰箱”来谈谈面向对象程序设计_java

把大象装进冰箱需要3步:打开冰箱门,把大象装入冰箱,关闭冰箱门。

 

扩展一下,我们考虑把动物装进冰箱的场景。比如,把猪🐷装进冰箱,把狗🐶装进冰箱,等等。

 

怎么利用面向对象的思想来进行程序设计呢?

 

talk is cheap, show me the code.

 

把大象装进冰箱的程序设计及实现

把大象装进冰箱,这个比较简单,简单到只有某些初级程序员写不出来。

如下,估计都是这个玩法。

从“把大象装进冰箱”来谈谈面向对象程序设计_ide_02

 代码就不贴了。

 

把动物装进冰箱的程序设计及实现

推而广之,要把不同的动物装进冰箱,并且假定每种动物装进冰箱的方式不同。那么,该怎么进行程序设计来实现这个需求呢?

程序结构图

从“把大象装进冰箱”来谈谈面向对象程序设计_ide_03

类结构图

从“把大象装进冰箱”来谈谈面向对象程序设计_java_04

 

 

 

model-模型类

Animal是抽象类(你可以new一头大象,一条狗,但你总不能new一个动物吧):

package com.animal2icebox.model;

import lombok.Data;

@Data
public abstract class Animal {
String name;
}

大象模型类-Elephant:

package com.animal2icebox.model;

import lombok.Data;

@Data
public class Elephant extends Animal {
int weight;

}

 

狗狗模型类-Dog:

package com.animal2icebox.model;

import lombok.Data;

@Data
public class Dog extends Animal {
/**
* 狗狗品种
*/
String breed;

}

 

重点是把动物装进冰箱的服务类

AnimalIntoIceboxService是抽象基类,提供了把动物装进冰箱的门面方法。但是由于每一种动物装入冰箱的方式各不相同,所以,要暴露接口(抽象方法)给派生类来实现。

package com.animal2icebox.service;

import com.animal2icebox.model.Animal;

public abstract class AnimalIntoIceboxService {
IceboxService iceboxService=new IceboxService();

public void putIntoIcebox(Animal animal) {
System.out.println("=============================");
iceboxService.openTheDoor();
input(animal);
iceboxService.closeTheDoor();
}

protected abstract void input(Animal animal);
}

 

把大象装进冰箱-ElephantIntoIceboxService:

从“把大象装进冰箱”来谈谈面向对象程序设计_程序设计思想_05

从“把大象装进冰箱”来谈谈面向对象程序设计_ide_06

package com.animal2icebox.service;

import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Elephant;

public class ElephantIntoIceboxService extends AnimalIntoIceboxService {
@Override
protected void input(Animal animal) {
Elephant elephant = (Elephant) animal;
System.out.println(String.format("大象信息:name=%s,重量=%s吨", elephant.getName(), elephant.getWeight()));
int threshold = 5;
int i = 0;
for (i = threshold; i <= elephant.getWeight(); i += threshold) {
System.out.println(String.format("这头大象重%s吨,已放入%s吨", elephant.getWeight(), threshold));
}
i -= threshold;
if (elephant.getWeight() > i) {
System.out.println(String.format("这头大象重%s吨,已放入%s吨", elephant.getWeight(), elephant.getWeight() - i));
}
System.out.println("已成功把大象装进冰箱!");
}

}

View Code

 

把狗狗装进冰箱-DogIntoIceboxService:

从“把大象装进冰箱”来谈谈面向对象程序设计_程序设计思想_05

从“把大象装进冰箱”来谈谈面向对象程序设计_ide_06

package com.animal2icebox.service;

import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;

public class DogIntoIceboxService extends AnimalIntoIceboxService {
@Override
protected void input(Animal animal) {
System.out.println("撒狗粮 here");
Dog dog=(Dog)animal;
String words = String.format("%s(品种=%s)", dog.getName(), dog.getBreed());
System.out.println("Hi " + words + ", welcome into the icebox...");
}
}

View Code

 

AnimalIntoIceboxFactory-工厂-生产的产品是上面的装进冰箱的动物service类对象

package com.animal2icebox.factory;

import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;
import com.animal2icebox.model.Elephant;
import com.animal2icebox.service.AnimalIntoIceboxService;
import com.animal2icebox.service.DogIntoIceboxService;
import com.animal2icebox.service.ElephantIntoIceboxService;
import lombok.Getter;

public class AnimalIntoIceboxFactory {
public static AnimalIntoIceboxService getServiceInstance(Animal animal) {
Class<AnimalIntoIceboxService> serviceClass = AnimalIntoBoxEnum.getServiceClass(animal);
assert serviceClass != null;

try {
return serviceClass.newInstance();
} catch (InstantiationException | IllegalAccessException e) {
e.printStackTrace();
}
return null;
}

@Getter
enum AnimalIntoBoxEnum {
Elephants(Elephant.class, ElephantIntoIceboxService.class),
Dogs(Dog.class, DogIntoIceboxService.class),
;

Class<Animal> animalClass;
Class<AnimalIntoIceboxService> serviceClass;

<E extends Animal, S extends AnimalIntoIceboxService> AnimalIntoBoxEnum(Class<E> animalClass, Class<S> serviceType) {
this.animalClass = (Class<Animal>) animalClass;
this.serviceClass = (Class<AnimalIntoIceboxService>) serviceType;
}

public static Class<AnimalIntoIceboxService> getServiceClass(Animal animal) {
AnimalIntoBoxEnum[] values = AnimalIntoBoxEnum.values();
for (AnimalIntoBoxEnum anEnum : values) {
if (anEnum.animalClass.equals(animal.getClass())) {
return anEnum.getServiceClass();
}
}
return null;
}
}
}

 

AnimalIntoIceboxContext-上下文容器,供外部调用

package com.animal2icebox.context;

import com.animal2icebox.factory.AnimalIntoIceboxFactory;
import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;
import com.animal2icebox.model.Elephant;
import com.animal2icebox.service.AnimalIntoIceboxService;

import java.util.ArrayList;
import java.util.List;

public class AnimalIntoIceboxContext {

public void putAnimalIntoIcebox(List<Animal> animals) {
for (Animal animal : animals) {
AnimalIntoIceboxService elephantIntoIceboxService = AnimalIntoIceboxFactory.getServiceInstance(animal);
assert elephantIntoIceboxService != null;
elephantIntoIceboxService.putIntoIcebox(animal);
}
}
}

 

来吧,开启调用之旅

package com.animal2icebox;

import com.animal2icebox.context.AnimalIntoIceboxContext;
import com.animal2icebox.model.Animal;
import com.animal2icebox.model.Dog;
import com.animal2icebox.model.Elephant;

import java.util.ArrayList;
import java.util.List;

public class AnimalIntoIceboxTest {
public static void main(String[] args) {
List<Animal> animals = new ArrayList<>();

Elephant elephant1 = new Elephant();
elephant1.setName("Emily");
elephant1.setWeight(13);
animals.add(elephant1);

Dog dog = new Dog();
dog.setName("Lucky");
dog.setBreed("dhole");
animals.add(dog);

Elephant elephant2 = new Elephant();
elephant2.setName("包包大人");
elephant2.setWeight(2);
animals.add(elephant2);

AnimalIntoIceboxContext context=new AnimalIntoIceboxContext();
context.putAnimalIntoIcebox(animals);
}
}

 

程序运行结果:

=============================
the door is open.
大象信息:name=Emily,重量=13吨
这头大象重13吨,已放入5吨
这头大象重13吨,已放入5吨
这头大象重13吨,已放入3吨
已成功把大象装进冰箱!
the door is closed.
=============================
the door is open.
撒狗粮 here
Hi Lucky(品种=dhole), welcome into the icebox...
the door is closed.
=============================
the door is open.
大象信息:name=包包大人,重量=2吨
这头大象重2吨,已放入2吨
已成功把大象装进冰箱!
the door is closed.

 

so,本文在“把动物装进冰箱”这个需求场景里,使用了哪些设计模式或设计思想呢?