用枚举实现工厂方法模式更简洁?

《编写高质量代码java》建议88--用枚举实现工厂方法模式更简洁<质疑>

建议先看下附录的代码,然后再看文字说明,附录代码展示了:原始工厂方法模式枚举实现的工厂方法模式

作者(秦小波)认为用枚举实现工厂方法模式更简洁,主要原因如下三点:

1.避免错误调用的发生,比如

Car car = CarFactory.createCar(Car.class);

这样的代码在编译时候不会报错,运行的时会报InstantiationException,我觉得用枚举调用实例化的时候,程序员也要指定你调用的是BuickCar还是FordCar,只是枚举做了下限定而已。

我认为良好的异常控制就能比较好的解决问题。而且一般动态加载类的时候,也是用Enum.valueof()方法定位枚举,也会因此抛出异常!

2.性能好,使用便捷

枚举实现,还需要switch,虽然用ordinal()实现了int排序,不过我感觉CarFactory.createCar(FordCar.class)连这个开销都没有吧?!

3.降低类间耦合

用原始工厂模式作者认为,调用者需要传递一个FordCar.class参数才能生产一两福特汽车,但是用枚举方法的时候也是要指定枚举的值才可以调用,这个并无太大区别。

主要我认为,如果新定义一种汽车,枚举的EnumCarFactory里就需要对应新加一个代码块,用来实现新加汽车的实例化,原始的工厂模式不需要变,只要新增的汽车类是extends Car!

大家也留言说说自己的想法吧。

附录:原始工厂方法模式和枚举实现的工厂方法模式

interface Car {
    public void myName();
};

class FordCar implements Car {
    public void myName() {
        System.out.println("it's ford");
    }
};

class BuickCar implements Car {
    public void myName() {
        System.out.println("it's buick");
    }
};

/**
 * 原始的工厂模式,新增一个Car子类,完全不管
 */
class OriginCarFactory {
    public static Car createCar(Class<? extends Car> c) {
        try {
            return (Car) c.newInstance();
        } catch (Exception e) {
            System.out.println("无效参数,无法初始化");
        }
        return null;
    }
}

/**
 * 枚举实现工厂模式,一旦新加一个类,还需要在enum中新增对应实例化方法
 */
enum EnumCarFactory {
    FordCar {
        @Override
        public Car create() {
            return new FordCar();
        }        
    },
    BuickCar {
        @Override
        public Car create() {
            return new BuickCar();
        }        
    };

    //abstract修饰方法,强制每个枚举实现该方法
    public abstract Car create();
}
public class Main {
    public static void main(String[] args) {
        
        //最常用的工厂模式,实例化class
        Car car = OriginCarFactory.createCar(BuickCar.class);
        if(car != null) {
            car.myName();
        }        
        
        //用枚举模式实例化class
        try {
            car = EnumCarFactory.valueOf("FordCar").create();
            car.myName();
        } catch (Exception e) {
            System.out.println("无效参数,无法初始化");
        }
        
    }
}
原文地址:https://www.cnblogs.com/yanghuahui/p/3308487.html