如今假设咱们有两种类型的订单,汽车服务订单和商城配件订单安全
咱们的抽象订单接口为app
public interface Order { public void makeOrder(Order order); }
抽象订单工厂接口为dom
public interface OrderFactory { public Order createOrder(); public Order getOrder(); }
现有具体的汽车服务类型订单ide
@Data @AllArgsConstructor @NoArgsConstructor @ServiceOrderVersion(value = 1) @RequiredArgsConstructor public class ServiceOrder implements Order { private Long id; @NonNull private String code; @NonNull private Store store; @NonNull private ProviderService service; @NonNull private Car car; @NonNull private Date serviceDate; @NonNull private String contact; @NonNull private String contactTel; private AppUser user; @NonNull private String content; private int status; private Date createDate; @Override public void makeOrder(Order order) { ServiceOrderDao serviceOrderDao = SpringBootUtil.getBean(ServiceOrderDao.class); IdService idService = SpringBootUtil.getBean(IdService.class); ((ServiceOrder)order).setId(idService.genId()); AppUser loginAppUser = AppUserUtil.getLoginAppUser(); AppUser user = new AppUser(); user.setId(loginAppUser.getId()); user.setUsername(loginAppUser.getUsername()); ((ServiceOrder)order).setUser(user); ((ServiceOrder)order).setStatus(1); ((ServiceOrder)order).setCreateDate(new Date()); serviceOrderDao.save((ServiceOrder) order); } }
@ServiceOrderVersion版本号以下ui
@Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface ServiceOrderVersion { int value(); }
商城配件订单略this
具体汽车服务工厂,咱们将其注入Spring容器,不须要每次下单的时候都去扫描包,过滤规则spa
@Component public class ServiceOrderFactory implements OrderFactory { private Set<Class<?>> classes = ClassUtil.getClassSet("com.cloud.ownercar.domain"); private Order createdOrder; @PostConstruct private void initOrder() { this.createdOrder = createOrder(); } @Override public Order createOrder() { Object instance = null; try { //过滤有@OrderVersion标签的类 instance = classes.stream().filter(clazz -> clazz.isAnnotationPresent(ServiceOrderVersion.class)) //过滤实现了Order接口的类 .filter(clazz -> Order.class.isAssignableFrom(clazz)) //找出版本号大的类,并实例化为对象 .max(Comparator.comparingInt(clazz -> clazz.getAnnotation(ServiceOrderVersion.class).value())) .get().newInstance(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } return (Order) instance; } @Override public Order getOrder() { return createdOrder; } }
具体配件工厂略线程
另外为了知足开闭原则,咱们须要加载全部类型的订单和订单工厂,之后在新增订单类型的时候,只须要在指定包下追加新的类型就好。code
@Component public class OrderBean { @Getter private Map<String,Class<?>> orderFactoryMap = new HashMap<>(); @Getter private Map<String,Class<?>> orderMap = new HashMap<>(); @PostConstruct private void init() { Set<Class<?>> classes = ClassUtil.getClassSet("com.cloud.ownercar.domain"); classes.stream().filter(clazz -> OrderFactory.class.isAssignableFrom(clazz)) .forEach(clazz -> orderFactoryMap.put(clazz.getSimpleName(),clazz)); classes.stream().filter(clazz -> Order.class.isAssignableFrom(clazz)) .forEach(clazz -> orderMap.put(clazz.getSimpleName(),clazz)); } }
Controller以下,用传递的内容来判断是哪一种类型的订单,并给抽象订单工厂来获取具体的订单工厂,经过具体的订单工厂来生成订单服务,完成下单功能。考虑到线程安全问题,因此要加ThreadLocal进行保护。这个type是在下单的时候须要提供的订单类型,好比"ServiceOrder"或者"ProductOrder"或者之后新增的订单类型。对象
@Slf4j @RestController public class OrderController { private ThreadLocal<OrderFactory> orderFactory = new ThreadLocal<>(); private ThreadLocal<Order> orderService = new ThreadLocal<>(); @Autowired private OrderBean orderBean; @Transactional @SuppressWarnings("unchecked") @PostMapping("/makeeorder") public Result<String> makeOrder(@RequestBody String orderStr, @RequestParam("type") String type) { log.info(orderStr); try { Order order = setOrderFactory(orderStr,type); orderService.get().makeOrder(order); return Result.success("下单成功"); } finally { orderFactory.remove(); orderService.remove(); } } /** * 判断是哪种类型的订单来获取哪种类型的具体订单工厂 * @param orderStr * @return */ private Order setOrderFactory(String orderStr,String type) { Class<?> classType = orderBean.getOrderMap().get(type); Object order = JSONObject.parseObject(orderStr, classType); // if (orderStr.contains("service")) { // order = JSON.parseObject(orderStr, ServiceOrder.class); // }else if (orderStr.contains("product")) { // order = JSON.parseObject(orderStr, ProductOrder.class); // } Class<?> classFactoryType = orderBean.getOrderFactoryMap().get(type + "Factory"); this.orderFactory.set((OrderFactory) SpringBootUtil.getBean(classFactoryType)); // if (order instanceof ServiceOrder) { // this.orderFactory.set(SpringBootUtil.getBean(ServiceOrderFactory.class)); // }else if (order instanceof ProductOrder) { // this.orderFactory.set(SpringBootUtil.getBean(ProductOrderFactory.class)); // } orderService.set(orderFactory.get().getOrder()); return (Order) order; } }
dao,mapper略