前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >架构框架搭建(一)《单体应用服务之SSM整合:Spring4 + SpringMvc + Mybatis》

架构框架搭建(一)《单体应用服务之SSM整合:Spring4 + SpringMvc + Mybatis》

作者头像
小傅哥
发布2020-07-14 15:19:31
6760
发布2020-07-14 15:19:31
举报
文章被收录于专栏:CodeGuide | 程序员编码指南

作者:付政委

烛火羸光映觥筹,踏入红尘何所求

微信公众号:bugstack虫洞栈 | 关注获取源码 沉淀、分享、成长,专注于原创专题案例,以最易学习编程的方式分享知识,让自己和他人都能有所收获。目前已完成的专题有;Netty4.x实战专题案例、用Java实现JVM、基于JavaAgent的全链路监控、手写RPC框架、架构设计专题案例[Ing]等。 你用剑?、我用刀?,好的代码都很烧?,望你不吝出招?!

前言介绍

在实际的业务开发中按照不同的场景需要,会有不同的业务架构也就同时会有不同的技术框架来支撑。那么这个专题想把一些常用的框架整理下,方便平时使用的同时也做一些技术沉淀。那么本章节会先搭建一个比较适合个人项目或者一些小公司开发项目的单体架构模型。服务功能展示页面如下;

微信公众号:bugstack虫洞栈 & 展示页面

工程环境

  1. JDK1.8
  2. Maven 3.2.3
  3. Spring 4.3.24.RELEASE + SpringMvc + Mybatis 3.3.0
  4. Mysql 5.6 + dbcp2
  5. layui 2.5.4

工程模型

整体的工程模型采用DDD四层架构,相对于MVC模式来讲。嗯!相当于家里三居换四居了!

代码语言:javascript
复制
 1itstack-demo-frame-ssm
 2└── src
 3    ├── main
 4    │   ├── java
 5    │   │   └── org.itstack.demo
 6    │   │       ├── application    
 7    │   │       │    └── UserService.java    
 8    │   │       ├── domain
 9    │   │       │    ├── model
10    │   │       │    │   ├── aggregates
11    │   │       │    │   │   └── UserInfoCollect.java
12    │   │       │    │   ├── req
13    │   │       │    │   │   └── UserReq.java        
14    │   │       │    │   └── vo
15    │   │       │    │       └── UserInfo.java   
16    │   │       │    ├── repository
17    │   │       │    │   └── IUserRepository.java    
18    │   │       │    └── service 
19    │   │       │        └── UserServiceImpl.java    
20    │   │       ├── infrastructure
21    │   │       │    ├── common
22    │   │       │    │   ├── EasyResult.java
23    │   │       │    │   └── PageRequest.java
24    │   │       │    ├── dao
25    │   │       │    │   └── IUserDao.java   
26    │   │       │    ├── po
27    │   │       │    │   └── User.java       
28    │   │       │    └── repository
29    │   │       │        └── UserRepository.java 
30    │   │       └── interfaces
31    │   │            └── UserController.java
32    │   ├── resources    
33    │   │   ├── mapper
34    │   │   ├── props    
35    │   │   ├── spring
36    │   │   ├── logback.xml
37    │   │   ├── mybatis-config.xml
38    │   │   └── spring-config.xml
39    │   └── webapp
40    │       ├── page
41    │       ├── res
42    │       ├── WEB-INF
43    │       ├── index.html
44    │       └── res_layui.html
45    └── test
46         └── java
47             └── org.itstack.demo.test
48                 └── ApiTest.java

以下对工程模块进行介绍,整体源码获取,可以关注公众号:bugstack虫洞栈,回复:框架搭建

application应用层

应用层是比较薄的一层,不做具体逻辑开发。本工程里只包括服务的定义,具体逻辑有领域层实现。如果需要扩展可以做一些应用服务编排。

application/UserService.java & 定义接口

代码语言:javascript
复制
 1/**
 2 * 微信公众号:bugstack虫洞栈 | 欢迎关注学习专题案例
 3 * 论坛:http://bugstack.cn
 4 * Create by 付政委 on @2019
 5 */
 6public interface UserService {
 7
 8    UserInfoCollect queryUserInfoList(UserReq req);
 9
10}

domain领域层

领域层是整个工程的核心服务层,这里负责处理具体的核心功能,完成领域服务。domain下可以有多个领域,每个领域里包括;聚合、请求对象、业务对象、仓储、服务。

domain/model/aggregates/UserInfoCollect.java & 定义聚合查询结果

代码语言:javascript
复制
 1/**
 2 * 微信公众号:bugstack虫洞栈 | 欢迎关注学习专题案例
 3 * 论坛:http://bugstack.cn
 4 * Create by 付政委 on @2019
 5 */
 6public class UserInfoCollect {
 7
 8    private Long count;
 9    private List<UserInfo> userInfoList;
10
11    public UserInfoCollect() {
12    }
13
14    public UserInfoCollect(Long count, List<UserInfo> userInfoList) {
15        this.count = count;
16        this.userInfoList = userInfoList;
17    }
18
19    public Long getCount() {
20        return count;
21    }
22
23    public void setCount(Long count) {
24        this.count = count;
25    }
26
27    public List<UserInfo> getUserInfoList() {
28        return userInfoList;
29    }
30
31    public void setUserInfoList(List<UserInfo> userInfoList) {
32        this.userInfoList = userInfoList;
33    }
34}

domain/repository/IUserRepository.java & 定义仓储服务

代码语言:javascript
复制
 1/**
 2 * 微信公众号:bugstack虫洞栈 | 欢迎关注学习专题案例
 3 * 论坛:http://bugstack.cn
 4 * Create by 付政委 on @2019
 5 */
 6public interface IUserRepository {
 7
 8    UserInfoCollect queryUserInfoList(UserReq req);
 9
10}

domain/service/UserServiceImpl.java & 对业务层功能进行实现

代码语言:javascript
复制
 1/**
 2 * 微信公众号:bugstack虫洞栈 | 欢迎关注学习专题案例
 3 * 论坛:http://bugstack.cn
 4 * Create by 付政委 on @2019
 5 */
 6@Service("userService")
 7public class UserServiceImpl implements UserService {
 8
 9    @Resource(name = "userRepository")
10    private IUserRepository userRepository;
11
12    @Override
13    public UserInfoCollect queryUserInfoList(UserReq req) {
14        return userRepository.queryUserInfoList(req);
15    }
16
17}

infrastructure基础层

  1. 实现领域层仓储定义
  2. 数据库操作为非业务属性的功能操作
  3. 在仓储实现层进行组合装配DAO&Redis&Cache等

infrastructure/dao/IUserDao.java & 数据库操作

代码语言:javascript
复制
 1/**
 2 * 微信公众号:bugstack虫洞栈 | 欢迎关注学习专题案例
 3 * 论坛:http://bugstack.cn
 4 * Create by 付政委 on @2019
 5 */
 6public interface IUserDao {
 7
 8    List<User> queryUserInfoList(UserReq req);
 9
10    Long queryUserInfoCount(UserReq req);
11
12}

infrastructure/repository/UserRepository.java & 仓储功能实现如果有redis可以进行包装使用

代码语言:javascript
复制
 1/**
 2 * 微信公众号:bugstack虫洞栈 | 欢迎关注学习专题案例
 3 * 论坛:http://bugstack.cn
 4 * Create by 付政委 on @2019
 5 */
 6@Repository("userRepository")
 7public class UserRepository implements IUserRepository {
 8
 9    @Resource
10    private IUserDao userDao;
11
12    @Override
13    public UserInfoCollect queryUserInfoList(UserReq req) {
14        Long count = userDao.queryUserInfoCount(req);
15        List<User> userList = userDao.queryUserInfoList(req);
16        List<UserInfo> userInfoList = new ArrayList<>();
17        userList.forEach(user -> {
18            UserInfo userInfo = new UserInfo();
19            userInfo.setUserId(user.getId());
20            userInfo.setName(user.getName());
21            userInfo.setAge(user.getAge());
22            userInfo.setAddress(user.getAddress());
23            userInfo.setEntryTime(user.getEntryTime());
24            userInfo.setStatus(user.getStatus());
25            userInfoList.add(userInfo);
26        });
27        return new UserInfoCollect(count, userInfoList);
28    }
29
30}

interfaces接口层

  1. 包装应用接口对外提供api,目前这一层比较简单只需要进行接口使用即可
  2. 如果是对外部提供服务接口,那么可以使用DTO方式进行转换,避免污染到业务类

interfaces/UserController.java & 提供接口服务

代码语言:javascript
复制
 1@Controller
 2@RequestMapping("/api/user/")
 3public class UserController {
 4
 5    private Logger logger = LoggerFactory.getLogger(UserController.class);
 6
 7    @Resource
 8    private UserService userService;
 9
10    @RequestMapping(path = "queryUserInfoList", method = RequestMethod.GET)
11    @ResponseBody
12    public EasyResult queryUserInfoList(String json, String page, String limit) {
13        try {
14            logger.info("查询用户信息列表开始。json:{}", json);
15            UserReq req = JSON.parseObject(json, UserReq.class);
16            if (null == req) req = new UserReq();
17            req.setPage(page, limit);
18            UserInfoCollect userInfoCollect = userService.queryUserInfoList(req);
19            logger.info("查询用户信息列表完成。userInfoCollect:{}", JSON.toJSONString(userInfoCollect));
20            return EasyResult.buildEasyResultSuccess(userInfoCollect.getCount(), userInfoCollect.getUserInfoList());
21        } catch (Exception e) {
22            logger.error("查询用户信息列表失败。json:{}", json, e);
23            return EasyResult.buildEasyResultError(e);
24        }
25    }
26
27}

resource配置

这里包括了Spring、SpringMvc、mybatis、以及日志信息的配置;

mapper/User_Mapper.xml

代码语言:javascript
复制
 1<?xml version="1.0" encoding="UTF-8"?>
 2<!DOCTYPE mapper
 3        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
 4        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
 5
 6<mapper namespace="org.itstack.demo.infrastructure.dao.IUserDao">
 7
 8    <select id="queryUserInfoCount" resultType="java.lang.Long">
 9        select count(id)
10        from user
11        <where>
12            <if test="name != null">
13                and name = #{name}
14            </if>
15            <if test="status != null">
16                and status = #{status}
17            </if>
18        </where>
19    </select>
20
21    <select id="queryUserInfoList" resultType="org.itstack.demo.infrastructure.po.User">
22      SELECT id, name, age, address, entryTime, status, remark, createTime, updateTime
23      FROM user
24      <where>
25          <if test="name != null">
26              and name = #{name}
27          </if>
28          <if test="status != null">
29              and status = #{status}
30          </if>
31      </where>
32      limit #{pageStart},#{pageEnd}
33    </select>
34
35</mapper>

props/jdbc.properties & 数据库链接信息

代码语言:javascript
复制
1db.jdbc.driverClassName=com.mysql.jdbc.Driver
2db.jdbc.url=jdbc:mysql://127.0.0.1:3306/itstack?createDatabaseIfNotExist=true&amp;characterEncoding=utf-8&amp;useUnicode=true
3db.jdbc.username=root
4db.jdbc.password=123456

spring/spring-config-datasource.xml & dbcp2数据源配置以及扫描Mapper等

代码语言:javascript
复制
 1<?xml version="1.0" encoding="UTF-8"?>
 2<beans xmlns="http://www.springframework.org/schema/beans"
 3       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4       xsi:schemaLocation="http://www.springframework.org/schema/beans
 5    http://www.springframework.org/schema/beans/spring-beans.xsd">
 6
 7    <!-- 1.数据库连接池 -->
 8    <bean id="dataSource" class="org.apache.commons.dbcp2.BasicDataSource">
 9        <property name="driverClassName" value="${db.jdbc.driverClassName}" />
10        <property name="url" value="${db.jdbc.url}" />
11        <property name="username" value="${db.jdbc.username}" />
12        <property name="password" value="${db.jdbc.password}" />
13        <property name="maxTotal" value="20" />
14        <property name="maxIdle" value="3" />
15        <property name="maxWaitMillis" value="15000" />
16        <property name="timeBetweenEvictionRunsMillis" value="60000" />
17        <property name="minEvictableIdleTimeMillis" value="180000" />
18    </bean>
19
20    <!-- 2.配置SqlSessionFactory对象 -->
21    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
22        <!-- 注入数据库连接池 -->
23        <property name="dataSource" ref="dataSource" />
24        <!-- 配置MyBaties全局配置文件:mybatis-config.xml -->
25        <property name="configLocation" value="classpath:mybatis-config.xml" />
26        <!-- 扫描entity包 使用别名 -->
27        <property name="typeAliasesPackage" value="com.soecode.lyf.entity" />
28        <!-- 扫描sql配置文件:mapper需要的xml文件 -->
29        <property name="mapperLocations" value="classpath:mapper/*.xml" />
30    </bean>
31
32    <!-- 3.配置扫描Dao接口包,动态实现Dao接口,注入到spring容器中 -->
33    <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
34        <!-- 注入sqlSessionFactory -->
35        <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory" />
36        <!-- 给出需要扫描Dao接口包 -->
37        <property name="basePackage" value="org.itstack.demo.infrastructure.dao" />
38    </bean>
39
40</beans>

resources/mybatis-config.xml

代码语言:javascript
复制
 1<?xml version="1.0" encoding="UTF-8" ?>
 2<!DOCTYPE configuration
 3  PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
 4  "http://mybatis.org/dtd/mybatis-3-config.dtd">
 5<configuration>
 6    <!-- 配置全局属性 -->
 7    <settings>
 8        <!-- 使用jdbc的getGeneratedKeys获取数据库自增主键值 -->
 9        <setting name="useGeneratedKeys" value="true" />
10        <!-- 使用列别名替换列名 默认:true -->
11        <setting name="useColumnLabel" value="true" />
12        <!-- 开启驼峰命名转换:Table{create_time} -> Entity{createTime} -->
13        <setting name="mapUnderscoreToCamelCase" value="true" />
14    </settings>
15</configuration>

resources/spring-config.xml

代码语言:javascript
复制
 1<?xml version="1.0" encoding="UTF-8"?>
 2<beans xmlns="http://www.springframework.org/schema/beans"
 3       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4       xmlns:context="http://www.springframework.org/schema/context"
 5       xmlns:aop="http://www.springframework.org/schema/aop"
 6       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd     http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd"
 7       default-autowire="byName">
 8
 9    <context:component-scan base-package="org.itstack"/>
10    <aop:aspectj-autoproxy/>
11
12    <!-- 属性文件读入 -->
13    <bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
14        <property name="locations">
15            <list>
16                <value>classpath:props/*.properties</value>
17            </list>
18        </property>
19    </bean>
20
21    <import resource="classpath:spring/spring-*.xml"/>
22
23</beans>
24

itstack.sql

代码语言:javascript
复制
1DROP TABLE user;
2CREATE TABLE user ( id bigint(11) NOT NULL AUTO_INCREMENT, name varchar(32), age int(4), address varchar(128), entryTime datetime, remark varchar(64), createTime datetime, updateTime datetime, status int(4) DEFAULT '0', PRIMARY KEY (id), INDEX idx_name (name) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
3insert into user (id, name, age, address, entryTime, remark, createTime, updateTime, status) values (1, '水水', 18, '吉林省榆树市黑林镇尹家村5组', '2019-12-22 00:00:00', '无', '2019-12-22 00:00:00', '2019-12-22 00:00:00', 0);
4insert into user (id, name, age, address, entryTime, remark, createTime, updateTime, status) values (2, '豆豆', 18, '辽宁省大连市清河湾司马道407路', '2019-12-22 00:00:00', '无', '2019-12-22 00:00:00', '2019-12-22 00:00:00', 1);
5insert into user (id, name, age, address, entryTime, remark, createTime, updateTime, status) values (3, '花花', 19, '辽宁省大连市清河湾司马道407路', '2019-12-22 00:00:00', '无', '2019-12-22 00:00:00', '2019-12-22 00:00:00', 0);

综上总结

  • 此工程模型基于SSM比较适合开发ERP服务,ERP使用layui页面清新,功能完善
  • 工程框架采用了DDD架构模式,在此架构模式下可以更容易的开发系统,适应后比MVC更加方便
  • 后续将继续拓展架构服务搭建,包括一些Dubbo、Redis、mq等使用,方便自己也方便他人
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-12-23,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 bugstack虫洞栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言介绍
  • 工程环境
  • 工程模型
    • application应用层
      • domain领域层
        • infrastructure基础层
          • interfaces接口层
            • resource配置
              • itstack.sql
              • 综上总结
              相关产品与服务
              数据库
              云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
              领券
              问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档