简单工厂(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;
    }

在后半段中可以看出其根据参数通过switchif-else创建了相应的具体子类对象,与之前的应用场景十分类似。在这里,Calendar既作为抽象产品类,也作为一个工厂类。

相关文章:

  • 2021-07-21
猜你喜欢
  • 2021-06-03
相关资源
相似解决方案