简单工厂(Simple Factory)
简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例,而不向客户暴露内部细节。
简单工厂不属于23种设计模式,但是之后的工厂方法模式、抽象工厂模式都是由其演化而来,并且在实际场景中也有应用,因此有必要了解。
适用场景:工厂类负责创建的对象比较少。
优缺点
优点:只需要传入一个正确的参数,就可以获取所需要的对象而无须知道其创建细节。
缺点:工厂类的职责相对过重,增加新的产品需要修改工厂类的判断逻辑,违背开闭原则。
应用场景
创建五个类:Video、JavaVideo、PythonVideo、VideoFactory、Test:
抽象产品类Video:
public abstract class Video {
public abstract void produce();
}
具体产品类JavaVideo、PythonVideo:
public class JavaVideo extends Video {
@Override
public void produce() {
System.out.println("录制Java课程视频");
}
}
public class PythonVideo extends Video {
@Override
public void produce() {
System.out.println("录制Python课程视频");
}
}
客户端类Test,这里可以传入字符串参数或者Class类参数:
public class Test {
public static void main(String[] args) {
VideoFactory videoFactory = new VideoFactory();
Video video = videoFactory.getVideo("java");
if(video == null){
return;
}
video.produce();
VideoFactory videoFactory2 = new VideoFactory();
Video video2 = videoFactory2.getVideo(JavaVideo.class);
if(video2 == null){
return;
}
video2.produce();
}
}
在简单工厂中,客户端不应该直接创建出具体的产品类,而应交给工厂类去创建,下面看看工厂类VideoFactory,使用了if-else判断参数或者使用使用反射技术从而决定创建哪个具体子类:
public class VideoFactory {
public Video getVideo(Class c){
Video video = null;
try {
video = (Video) Class.forName(c.getName()).newInstance();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
return video;
}
public Video getVideo(String type){
if("java".equalsIgnoreCase(type)){
return new JavaVideo();
} else if("python".equalsIgnoreCase(type)){
return new PythonVideo();
}
return null;
}
}
通过简单工厂,客户端类就不需要自己去实例化具体的产品类,做到了客户端类和产品类的解耦。
Calendar类的应用
java.util下的Calendar类是一个抽象类,我们看看其中的getInstance方法:
public static Calendar getInstance(TimeZone zone,
Locale aLocale)
{
return createCalendar(zone, aLocale);
}
private static Calendar createCalendar(TimeZone zone,
Locale aLocale)
{
CalendarProvider provider =
LocaleProviderAdapter.getAdapter(CalendarProvider.class, aLocale)
.getCalendarProvider();
if (provider != null) {
try {
return provider.getInstance(zone, aLocale);
} catch (IllegalArgumentException iae) {
// fall back to the default instantiation
}
}
Calendar cal = null;
if (aLocale.hasExtensions()) {
String caltype = aLocale.getUnicodeLocaleType("ca");
if (caltype != null) {
switch (caltype) {
case "buddhist":
cal = new BuddhistCalendar(zone, aLocale);
break;
case "japanese":
cal = new JapaneseImperialCalendar(zone, aLocale);
break;
case "gregory":
cal = new GregorianCalendar(zone, aLocale);
break;
}
}
}
if (cal == null) {
if (aLocale.getLanguage() == "th" && aLocale.getCountry() == "TH") {
cal = new BuddhistCalendar(zone, aLocale);
} else if (aLocale.getVariant() == "JP" && aLocale.getLanguage() == "ja"
&& aLocale.getCountry() == "JP") {
cal = new JapaneseImperialCalendar(zone, aLocale);
} else {
cal = new GregorianCalendar(zone, aLocale);
}
}
return cal;
}
在后半段中可以看出其根据参数通过switch和if-else创建了相应的具体子类对象,与之前的应用场景十分类似。在这里,Calendar既作为抽象产品类,也作为一个工厂类。