注意:创建父模块选用Spring Initializr的方式创建,让父模块继承springboot的maven配置,之后的所有子模块都通过maven的方式创建,子模块继承父模块就会间接继承springboot相关的配置,web,api模块再另外添加spring-boot-starter-web,spring-boot-starter-tomcat相关的依赖。网上有很多教程父模块都是通过maven的方式创建的,然后子模块是通过Spring Initalizr方式创建,这种方式父模块无法管理子模块的依赖仲裁,需要每个子模块自行管理,就失去了父模块的用处了。
idea -> file -> new -> project,选择Spring Initializr,填写相关的Group,Artifact,Package name 等信息,点击Next;
JDK版本可以根据需要选择高一些的版本,我是由于需要兼容线上的老版本系统,所以沿用JDK1.8的版本
根据需要选取初始化需要加载的依赖,我添加了Cloud Bootstrap,可以自动生成Spring Cloud依赖管理包,直接点解Create
删除生成后的src目录,pom.xml中添加pom属性
刚才创建时选择的Cloud Bootstrap 会在pom文件中自动生成spring-cloud 相关的依赖配置,spring-cloud-dependencies 是一个依赖管理器的pom文件,是对spring cloud版本的依赖管理
使用maven方式创建api,biz,core,dao,integration,common,model 相关模块,创建内部依赖模块是使用maven的方式创建,便于总模块管理子模块的依赖仲裁;
点击项目根目录,右键 new -> module,选用maven的方式新建模块,Archetype可以选择quickstart,分别添加api,biz,core,dao,integration,common,model 这些模块。直接点击Create
web模块可以采用Spring Initializr的方式创建,可以自动初始化web相关的pom包和springboot的启动类,之后再修改web模块的pom文件的标签为当前父工程的配置即可;
点击项目根目录,右键 new -> module,选用Spring Initializr的方式新建web模块,点击Next
根据自己的需要勾选所需的依赖,我选择了Spring Boot DevTools 和 Spring Web,点击Create
创建完所有模块之后,自行在各个模块的pom 文件中添加模块依赖配置,推荐模块依赖关系如下图:
模块职责&关系说明
包名命名规范
web
|- com.xxx.web.interceptor
|- com.xxx.web.controller
|- com.xxx.web.websocket
|- com.xxx.web.vo
biz
|- com.xxx.biz.manager
|- com.xxx.biz.processor
|- com.xxx.biz.job
core
|- com.xxx.core.service
|- com.xxx.core.service.impl
|- com.xxx.core.handler
model
|- com.xxx.model.biz
|- com.xxx.model.core
|- com.xxx.model.common
|- com.xxx.model.integration
|- com.xxx.model.dto
|- com.xxx.model.vo
|- com.xxx.model.exception
|- com.xxx.model.constants
|- com.xxx.model.enums
dao
|- com.xxx.dao.dao
|- com.xxx.dao.dataobject
|- com.xxx.dao.mapper
integration
|- com.xxx.integration.client
|- com.xxx.integration.client.impl
common
|- com.xxx.common.utils
方法命名
领域模型命名
spring事务规范
声明式事务和编程式事务的比较:
总结:不推荐使用声明式事务,建议统一使用编程式事务。并且事务中只能包含数据库数据操作相关的代码,不能包含较多的业务处理逻辑,外部系统调用等耗时较长的代码;
事务模板代码实现:
Templates
import com.insigma.traffic.road.detection.model.exception.BizSystemException;
import com.insigma.traffic.road.detection.model.exception.ExceptionEnum;
import com.insigma.traffic.road.detection.model.result.BizResult;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
import org.springframework.transaction.support.TransactionTemplate;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* 服务模版,可执行事务和非事务处理。
* 该模板会吃掉所有异常,不会往外抛异常,需要调用方根据返回结果作相应的处理
* 事务处理需要在参数中指定事务模版类型 {@link TxType}
*
* 事务模板使用样例:
* BizResult result = Templates.execute(TxType.CVP, () -> {
* // do checker
* }, () -> {
* // do execute
* return "return result";
* }, LOGGER);
*
* @author wangzhongxing
* @version $Id: Templates.java, v 0.1 2019年08月27日 13:28 PM wangzhongxing Exp $
*/
@Component
public class Templates implements ApplicationContextAware {
/**
* 默认日志
*/
private static final Logger DEFAULT_LOGGER = LoggerFactory.getLogger(Templates.class);
/**
* 事务模板
*/
private static final Map<String, TransactionTemplate> TEMPLATES = new ConcurrentHashMap<>();
/**
* Disable construct.
*/
private Templates() {
}
/**
* @see ApplicationContextAware#setApplicationContext(ApplicationContext)
*/
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
// init transaction templates
String name;
for (TxType txType : TxType.values()) {
name = txType.getBeanName();
TEMPLATES.put(name, applicationContext.getBean(name, TransactionTemplate.class));
}
}
/**
* 执行模板操作,不带数据库事务
*
* @param logger 自定义日志
* @param executor 业务执行器
* @param 返回结果类型
* @return biz result
*/
public static <T> BizResult<T> execute(SupplyFunction<T> executor, Logger logger) {
return execute(null, null, executor, logger);
}
/**
* 执行模板操作,不带数据库事务
*
* @param logger 自定义日志
* @param checker 前置业务检查器
* @param executor 业务执行器
* @param 返回结果类型
* @return biz result
*/
public static <T> BizResult<T> execute(ActionFunction checker, SupplyFunction<T> executor,
Logger logger) {
return execute(null, checker, executor, logger);
}
/**
* 执行模板操作,带数据库事务
*
* @param txType 事务类型
* @param logger 自定义日志
* @param executor 业务执行器
* @param 返回结果类型
* @return biz result
*/
public static <T> BizResult<T> execute(TxType txType, SupplyFunction<T> executor, Logger logger) {
return execute(txType, null, executor, logger);
}
/**
* 执行模板操作,带数据库事务
*
* @param txType 事务类型
* @param logger 自定义日志
* @param checker 前置业务检查器
* @param executor 业务执行器
* @param 返回结果类型
* @return biz result
*/
public static <T> BizResult<T> execute(TxType txType, ActionFunction checker,
SupplyFunction<T> executor, Logger logger) {
BizResult<T> result;
try {
if (logger == null) {
logger = DEFAULT_LOGGER;
}
// decide run mode
TransactionTemplate transactionTemplate = null;
if (null != txType) {
transactionTemplate = TEMPLATES.get(txType.getBeanName());
logger.debug("Run with transaction mode, TEMPLATES={}", TEMPLATES);
if (transactionTemplate == null) {
logger.error("Get transaction template by name[{}] error.", txType.getBeanName());
}
}
// biz check
if (checker != null) {
checker.apply();
}
// biz execute
if (executor == null) {
throw new BizSystemException(ExceptionEnum.SYSTEM_ERROR);
}
T data;
if (transactionTemplate != null) {
data = transactionTemplate.execute(status -> executor.apply());
} else {
data = executor.apply();
}
result = new BizResult<>(data);
} catch (BizSystemException e) {
result = new BizResult<>(e.getErrorCode(), e.getMessage());
logger.error("Tamplates execute BizSystemException!errorCode={},errorMsg={}",
e.getErrorCode(), e.getErrorMsg());
} catch (Exception e) {
result = new BizResult<>(ExceptionEnum.SYSTEM_ERROR);
logger.error("Tamplates execute Exception!", e);
}
return result;
}
public static void main(String[] args) {
ActionFunction checker = () -> {
};
SupplyFunction<String> executor = () -> {
return "";
};
BizResult<String> result = Templates.execute(TxType.CVP, () -> {
// do checker
// ...
}, () -> {
// do execute
// ...
return "return result";
}, DEFAULT_LOGGER);
}
}
TxType
/**
* 事务类型枚举
*
*/
public enum TxType {
/**
* 默认事务模版:
* 如果上下文中已经存在事务,那么就加入到事务中执行,如果当前上下文中不存在事务,则新建事务。
* 简而言之,只会存在最外层的父事务模版,因此,任意嵌套层次事务模版的回滚,都会导致所有事务模版的回滚。
* {@link TransactiontemplateConfig}
*/
CVP("transactionTemplateRequired"),
/**
* 事务模版 PROPAGATION_REQUIRES_NEW:
* 每次都会新建一个事务,并且同时将上下文中的事务挂起,当新建事务执行完成以后,上下文事务再恢复执行
* 父事务模版和子事务模版相互隔离,回滚互不影响
* {@link TransactiontemplateConfig}
*/
CVP_NEW("transactionTemplateNew"),
;
/** 事务模版名称 */
private String beanName;
/**
* Constructor
*
* @param beanName 事务模版名称
*/
TxType(String beanName) {
this.beanName = beanName;
}
public String getBeanName() {
return beanName;
}
}
ActionFunction
/**
* 没有入参和出参, 只执行业务动作的函数。
*
* @author wangzhongxing
* @version $Id: ActionFunction.java, v 0.1 2019年08月27日 13:38 PM wangzhongxing Exp $
*/
@FunctionalInterface
public interface ActionFunction {
/**
* 执行方法
*/
void apply();
}
SupplyFunction
/**
* 没有入参, 执行业务动作并返回处理结果的函数。
*
* @author wangzhongxing
* @version $Id: ActionFunction.java, v 0.1 2019年08月27日 13:38 PM wangzhongxing Exp $
*/
@FunctionalInterface
public interface SupplyFunction<T> {
/**
* 执行方法.
*
* @return 返回结果。
*/
T apply();
}
多模块化后,想让每个模块的配置文件独立管理,整合的时候再合并各个不同的模块的配置,需要先在每个模块下面创建一个配置文件,文件的命名格式必须是application-xxx.yml,启动层模块想引用下层模块的配置时需要在配置文件中添加spring.profiles.include: xxx 或者spring.profiles.active: xxx (xxx为子模块配置文件的后缀标识名)配置项显示引用所需要的配置文件。注意启动层模块的配置文件名必须是application.yml或者application.properties,不能带上-xxx的后缀,不然导致spring.profiles.active的配置失效,从而无法加载子模块的配置。例如启动层是web层,web层的配置文件命名为application.yml,如果命名是application-web.yml会导致子模块的配置无法加载。
ConfigFileApplicationListener 子模块配置加载类源码说明如下,默认是从加载application.properties或application.yml这两个配置文件中加载其它配置文件的配置,所以启动层模块的配置文件名必须是application.properties或application.yml。