SSM Spring MVC集成mybatis

2024-09-08 01:20
文章标签 java ssm spring 集成 mvc mybatis

本文主要是介绍SSM Spring MVC集成mybatis,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、练习目标

1、需求

完成部门基本的 CRUD 和分页查询,完成员工基本的 CRUD、分页查询和过滤查询(根据姓名和邮箱模糊查询,根据部门查询)。

2、技术架构

使用 Spring MVC + Spring + MyBatis,数据库选用 MySQL,视图选用 JSP。

3、SSM 集成作用及本质

作用:在框架上基础上开发,发挥各个框架在各层的好处,提高开发效率。

本质:

  • Spring 去集成 Spring MVC 和 MyBatis,即控制器对象、业务对象、Mapper 对象等都交由 Spring 容器管理,使用 Spring IoC 和 DI 来完成对象创建及其属性注入;

  • 使用 AOP 来配置事务;

  • 使用 Spring MVC 解决 MVC 的问题,处理请求和响应。

4、集成步骤

  • 先用 Spring 集成 MyBatis

    • 搭建项目,添加依赖配置插件。

    • 把 Spring 和 MyBatis 的等配置文件拷贝进项目 resources 目录下

    • 配置数据库链接池

    • 配置 SqlSessionFactory

    • 配置 Mapper 对象

    • 配置业务对象

    • 配置事务相关

  • 再加入 Spring MVC

    • 在 web.xml 配置端控制器和编码过滤器

    • 在项目 resources 目录下添加 Spring MVC 的配置文件,并配置 MVC 注解解析器,扫描控制器,静态资源处理,视图解析器等等

    • 在 Spring MVC 的配置文件中引入 Spring 配置文件

5、项目准备

5.1 在之前项目上改

这里所谓之前的项目就是 ssm 项目。

5.2、检查依赖和插件

确定依赖和插件都没有配错,或者少配。

二、MyBatis 逆向工程

1、MyBatis 逆向工程

一个 Maven 插件,根据数据的表生成实体类和 Mapper 接口和 Mapper XML。

2、插件使用

2.1、新建表

参考文档,在 ssm 库新建如下表:

CREATE TABLE `department` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`name` varchar(255) DEFAULT NULL,`sn` varchar(255) DEFAULT NULL,PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

2.2、配置插件

在 pom.xml,配置 MyBatis 逆向工程插件:

<!-- MyBatis 逆向工程插件 -->
<plugin><groupId>org.mybatis.generator</groupId><artifactId>mybatis-generator-maven-plugin</artifactId><version>1.3.2</version><configuration><verbose>true</verbose><overwrite>false</overwrite></configuration><dependencies><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.45</version></dependency></dependencies>
</plugin>

2.3、插件设置

在项目中 resources 目录中新建配置文件 generatorConfig.xml,在 generatorConfig.xml 配置数据库连接、配置表等等。注意: 直接从课件目录下 practise/MyBatis 逆向工程目录中找到此文件拷贝过来修改即可。

2.4、运行插件

在 IDEA 的 Maven 窗口下中双击 mybatis-generator:generate 即可运行,运行注意观察控制台打印。

3、编写部门业务接口及实现

里面提供最基本 CRUD 方法。

package cn.service;
​
public interface IDepartmentService {void delete(Long id);void save(Department department);Department get(Long id);List<Department> listAll();void update(Department department);
}
package cn.service.impl;
​
public class DepartmentServiceImpl implements IDepartmentService {private DepartmentMapper departmentMapper;
​@Overridepublic void delete(Long id) {departmentMapper.deleteByPrimaryKey(id);}
​@Overridepublic void save(Department department) {departmentMapper.insert(department);}
​@Overridepublic Department get(Long id) {return departmentMapper.selectByPrimaryKey(id);}
​@Overridepublic List<Department> listAll() {return departmentMapper.selectAll();}
​@Overridepublic void update(Department department) {departmentMapper.updateByPrimaryKey(department);}
}

三、Spring 集成 MyBatis

1、配置数据库连接池

在 resources 目录下新建 applicationContext.xml,配置如下:

<!-- 关联 db.properties 文件 -->
<context:property-placeholder location="classpath:db.properties"/>
​
<!-- 配置 DataSource bean -->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close"><property name="driverClassName" value="${jdbc.driverClassName}"/><property name="url" value="${jdbc.url}"/><property name="username" value="${jdbc.username}"/><property name="password" value="${jdbc.password}"/>
</bean>

2、配置 SqlSessionFactory bean

在 applicationContext.xml,配置如下:

<!-- 配置 SqlSessionFactory bean -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"><!-- 关联主配置文件 目前可以不配置--><property name="configLocation" value="classpath:mybatis-config.xml"/><!-- 配置别名 若不用别名,可以不配置 --><property name="typeAliasesPackage" value="cn.domain"/><!-- 配置数据源--><property name="dataSource" ref="dataSource"/><!-- 关联 Mapper XML 可以不配置, 前提编译 Mapper 接口字节码文件与 Mapper XML 文件在一起,文件名也一样 -->
</bean>

3、配置 Mapper bean

在 applicationContext.xml,配置如下:

<!-- 配置 Mapper bean -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"><!-- 指定 Mapper 接口所在包 --><property name="basePackage" value="cn.mapper"/>
</bean>

4、配置 Service bean

修改 DepartmentServiceImpl,贴 IoC 和 DI 注解。

package cn.service.impl;
​
@Service
public class DepartmentServiceImpl implements IDepartmentService {
​@Autowiredprivate DepartmentMapper departmentMapper;
​@Overridepublic void delete(Long id) {departmentMapper.deleteByPrimaryKey(id);}
​@Overridepublic void save(Department department) {departmentMapper.insert(department);}
​@Overridepublic Department get(Long id) {return departmentMapper.selectByPrimaryKey(id);}
​@Overridepublic List<Department> listAll() {return departmentMapper.selectAll();}
​@Overridepublic void update(Department department) {departmentMapper.updateByPrimaryKey(department);}
}

在 applicationContext.xml,配置如下:

<!-- 配置 IoC DI 注解解析器 , 让 Spring 帮我们创建业务接口的实现类对象, 完成属性或者字段注入 -->
<context:component-scan base-package="cn.service.impl"/>

5、配置事务相关

在 applicationContext.xml,配置如下:

<!-- 配置事务管理器 WHAT-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="dataSource"/>
</bean>
​
<!-- 配置增强,包含 WHEN,并关联上面 WHAT-->
<tx:advice id="txAdvice" transaction-manager="transactionManager"><tx:attributes><tx:method name="get*" read-only="true"/><tx:method name="select*" read-only="true"/><tx:method name="query*" read-only="true"/><tx:method name="count*" read-only="true"/><tx:method name="list*" read-only="true"/><tx:method name="*"/></tx:attributes>
</tx:advice>
​
<!-- 配置 AOP -->
<aop:config><!-- WHERE --><aop:pointcut id="txPointcut" expression="execution(* cn.service.impl.*ServiceImpl.*(..))"/><!-- 关联 WHERE WHEN--><aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
</aop:config>

6、编写单元测试类

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class DepartmentServiceTest {@Autowiredprivate IDepartmentService departmentService;
​@Testpublic void testSave() {Department department = new Department();department.setName("公关部");department.setSn("DC");departmentService.save(department);}
}

四、集成 Spring MVC

1、在 web.xml 配置端控制器和编码过滤器

<!-- 配置前端控制器 -->
<servlet><servlet-name>dispatcherServlet</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:mvc.xml</param-value></init-param><load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping><servlet-name>dispatcherServlet</servlet-name><url-pattern>/</url-pattern>
</servlet-mapping>
​
<!-- 配置编码过滤器, 针对 POST -->
<filter><filter-name>characterEncodingFilter</filter-name><filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class><init-param><param-name>encoding</param-name><param-value>utf-8</param-value></init-param>
</filter>
<filter-mapping><filter-name>characterEncodingFilter</filter-name><url-pattern>/*</url-pattern>
</filter-mapping>

2、添加 Spring MVC 的配置文件

在 resources 目录新建 mvc.xml,配置如下:

<!-- IoC DI 注解解析器,配置扫描控制器。说人话:让 Spring 帮我们创建控制器对象,并注入 -->
<context:component-scan base-package="cn.web.controller"/>
​
<!-- 配置 MVC 注解解析器,时间注解,JSON 注解 -->
<mvc:annotation-driven/>
​
<!-- 处理静态资源 -->
<mvc:default-servlet-handler/> 
​
<!-- 配置试图解析器 -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="prefix" value="/WEB-INF/views/"/><property name="suffix" value=".jsp"/>
</bean>

3、在 Spring MVC 的配置文件中引入 Spring 配置文件

在 mvc.xml,配置如下

<!-- 引入 applicationContext.xml -->
<import resource="classpath:applicationContext.xml"/>

4、编写 DepartmentController

package cn.web.controller;
​
@Controller
@RequestMapping("/department")
public class DepartmentController {@Autowiredprivate IDepartmentService departmentService;
​// 处理查询所有部门请求@RequestMapping("/list")public String list(Model model){List<Department> departments = departmentService.listAll(qo);model.addAttribute("departments", departments);return "department/list"; // 找视图 /WEB-INF/views/department/list.jsp}
​// 处理删除请求@RequestMapping("/delete")public String delete(Long id){if (id != null) {departmentService.delete(id);}return "redirect:/department/list"; // 让浏览器重新发一次请求}
​// 去新增或去修改页面@RequestMapping("/input")public String input(Long id, Model model){if (id != null) { // 表示去修改Department department = departmentService.get(id);model.addAttribute("department", department);}return "department/input"; // 让浏览器重新发一次请求}
​// 新增或修改@RequestMapping("/saveOrUpdate")public String saveOrUpdate(Department department){if(department.getId()  == null){ // 新增departmentService.save(department);}else {departmentService.update(department); // 修改}return "redirect:/department/list"; // 让浏览器重新发一次请求}
}
​

5、拷贝部门 JSP 修改

到课件的 practise/views 目录拷贝 JSP 文件到项目的 webapp/views/department 目录下修改。

五、部门分页查询

1、编写封装分页参数类

package cn.qo;
​
@Setter
@Getter
public class QueryObject {// 当前页private int currentPage = 1;// 每页条数private int pageSize = 5;public int getStart(){return (currentPage - 1) * pageSize;}
}

2、编写封装查询结果类

package cn.qo;
​
@Getter
public class PageResult<T> {
​private int currentPage; // 页面传的private int pageSize; // 页面传的private List<T> data; // 数据库查询private int totalCount; // 数据库查询
​private int totalPage; // 总页数private int prevPage; // 上一页private int nextPage; // 下一页
​public PageResult(int currentPage, int pageSize, int totalCount, List<T> data) {this.currentPage = currentPage;this.pageSize = pageSize;this.data = data;this.totalCount = totalCount;
​// 加入判断,节省计算性能if(totalCount <= pageSize) {this.totalPage = 1;this.prevPage = 1;this.nextPage = 1;return;}
​this.totalPage = this.totalCount % this.pageSize == 0? this.totalCount / this.pageSize : this.totalCount / this.pageSize + 1;this.nextPage = this.currentPage + 1 <= this.totalPage ? this.currentPage + 1 : this.totalPage;this.prevPage = this.currentPage - 1 >= 1 ? this.currentPage - 1 : 1;}
}

3、给部门业务接口及实现添加分页功能

package cn.service;
​
public interface IDepartmentService {// ......PageResult<Department> query(QueryObject qo);
}
package cn.service.impl;
​
@Service
public class DepartmentServiceImpl implements IDepartmentService {
​@Autowiredprivate DepartmentMapper departmentMapper;
​// ......
​@Overridepublic PageResult<Department> query(QueryObject qo) {int count = departmentMapper.selectForCount(qo);if(count == 0){return  new PageResult<>(qo.getCurrentPage(), qo.getPageSize(),count, Collections.emptyList());}List<Department> departments = departmentMapper.selectForList(qo);return  new PageResult<>(qo.getCurrentPage(), qo.getPageSize(),count, departments);}
}

4、给部门 Mapper 接口及 Mapper XMl 添加分页功能

package cn.mapper;
​
public interface DepartmentMapper {// ......
​int selectForCount(QueryObject qo);List<Department> selectForList(QueryObject qo);
}
<select id="selectForCount" resultType="java.lang.Integer">SELECT count(*)FROM department
</select>
<select id="selectForList" resultType="cn.domain.Department">SELECT id, name, snFROM departmentLIMIT #{start}, #{pageSize}
</select>

5、修改部门控制器查询方法

package cn.web.controller;
​
@Controller
@RequestMapping("/department")
public class DepartmentController {@Autowiredprivate IDepartmentService departmentService;
​// 处理查询所有部门请求@RequestMapping("/list")public String list(Model model, QueryObject qo){PageResult<Department> pageResult = departmentService.query(qo);model.addAttribute("pageResult", pageResult);return "department/list"; // 找视图 /WEB-INF/views/department/list.jsp}
​// ......
}

6、修改部门 list.jsp

回显分页查询的部门数据。

六、员工增删改查

1、使用逆向工程生成 Employee 实体类等

根据权限系统业务对象属性参照表文档建员工表,再使用逆向工程生成 Employee.java 、EmployeeMapper.java 、EmployeeMapper.xml。

CREATE TABLE `employee` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`username` varchar(255) DEFAULT NULL,`name` varchar(255) DEFAULT NULL,`password` varchar(255) DEFAULT NULL,`email` varchar(255) DEFAULT NULL,`age` int(11) DEFAULT NULL,`admin` bit(1) DEFAULT NULL,`dept_id` bigint(20) DEFAULT NULL,PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

生成的 Employee 类中 admin 字段注意改成 boolean 类型。

2、编写封装查询参数类

package cn.qo;@Setter
@Getter
public class EmployeeQueryObject extends QueryObject {private String keyword;private Long deptId; // 若这个值为 null 情况,就不根据部门过滤数据
}

3、给员工 Mapper 接口及 Mapper XMl 添加分页和过滤功能

package cn.mapper;
​
public interface EmployeeMapper {// ......int selectForCount(QueryObject qo);List<Employee> selectForList(QueryObject qo);
}
​
<sql id="where_sql"><where><if test="keyword != null">AND (name LIKE concat('%', #{keyword}, '%') OR email LIKE concat('%', #{keyword}, '%'))</if><if test="deptId != null">AND dept_id = #{deptId}</if></where>
</sql>
<select id="selectForCount" resultType="java.lang.Integer">SELECT count(id)FROM employee <include refid="where_sql"/>
</select>
<select id="selectForList" resultMap="BaseResultMap">SELECT id, username, name, password, age, email, admin, dept_idFROM employee<include refid="where_sql"/>LIMIT #{start}, #{pageSize}
</select>

4、拷贝部门业务类与控制器类修改

5、拷贝员工 JSP 修改

到课件的 practise/views 目录拷贝 JSP 文件到项目的 webapp/views/employee 目录下修改。

七、员工查询完善

1、完善超管显示

超管显示为是或否,而不显示 true 或 false。修改员工 list.jsp,如下:

<td>${employee.admin ? '是' : '否'}</td>

2、完事部门显示

部门显示部门名称,而不是显示部门 id 值。

2.1、修改员工实体类

package cn.domain;
​
@Setter
@Getter
public class Employee {private Long id;private String username; // 登录用户名private String name; // 姓名private String password;private Integer age;private String email;private boolean admin;// 关联属性, 不仅可以封装部门 id 可以封装部门名称private Department dept;
}

2.2、修改员工 Mapper XML

修改查询员工 SQL 及结果集处理的 resultMap。

 <resultMap id="BaseResultMap" type="cn.domain.Employee"><id column="id" property="id"/><result column="username" property="username"/><result column="name" property="name"/><result column="password" property="password"/><result column="age" property="age"/><result column="email" property="email"/><result column="admin" property="admin"/><result column="d_id" property="dept.id"/><result column="d_name" property="dept.name"/><result column="d_sn" property="dept.sn"/>
</resultMap>
​
<sql id="where_sql"><where><if test="keyword != null">AND (e.name LIKE concat('%', #{keyword}, '%') OR e.email LIKE concat('%', #{keyword}, '%'))</if><if test="deptId != null">AND e.dept_id = #{deptId}</if></where>
</sql>
​
<select id="selectForCount" resultType="java.lang.Integer">SELECT count(e.id)FROM employee e<include refid="where_sql"/>
</select>
<select id="selectForList" resultMap="BaseResultMap">SELECT e.id, e.name, e.password, e.age, e.email, e.admin, e.dept_id, d.id AS d_id, d.name AS d_name, d.sn AS d_snFROM employee eLEFT JOIN department d ON e.dept_id = d.id<include refid="where_sql"/>LIMIT #{start}, #{pageSize}
</select>

2.3、修改员工 list.jsp

<td>${employee.dept.name}</td>

3、完善过滤查询

页面提供关键字输入栏及部门下来框,让用户输入关键字与选择部门,以便过滤查询员工数据,并回显查询条件。

3.1、修改员工控制器

package cn.web.controller;
​
@Controller
@RequestMapping("/employee")
public class EmployeeController {@Autowiredprivate IEmployeeService employeeService;@Autowiredprivate IDepartmentService departmentService;
​// 处理查询所有员工请求@RequestMapping("/list")// 把查询的方法的封装参数的形参改成 EmployeeQueryObject 类型,并贴上 @ModelAttribute("qo")public String list(Model model, @ModelAttribute("qo")EmployeeQueryObject qo){PageResult<Employee> pageResult = employeeService.query(qo);model.addAttribute("pageResult", pageResult);
​List<Department> departments = departmentService.listAll();// 查询所有部门往模型存入,提供数据,让员工列表页面显示出来以供用户选择model.addAttribute("departments", departments);
​return "employee/list";}
​// ......
}

3.2、修改员工 list.jsp

只要表单那块即可,提供搜索的关键字输入栏和部门选择下拉框,并使用 EL 表达式回显查询条件,如下:

<!-- 查询的表单 -->
<form action="/employee/list" method="post"><input type="hidden" name="currentPage" id="currentPage" value="1"/>关键字:<input type="text" name="keyword" value="${qo.keyword}"/>部门:<select name="deptId"><option value="">全部</option><c:forEach items="${departments}" var="department"><option value="${department.id}" ${department.id == qo.deptId ? 'selected' : ''}></option></c:forEach></select><input type="submit" value="查询"/>
</form>

八、员工新增修改完善

1、员工新增完善

新增员工时,部门不要输入而是选择。

1.1、修改员工控制器

package cn.web.controller;
​
@Controller
@RequestMapping("/employee")
public class EmployeeController {@Autowiredprivate IEmployeeService employeeService;@Autowiredprivate IDepartmentService departmentService;
​@RequestMapping("/input")public String input(Long id, Model model){
​List<Department> departments = departmentService.listAll();// 查询所有部门往模型存入,提供数据,让员工新增修改时显示出来以供用户选择model.addAttribute("departments", departments);
​if (id != null) { // 表示去修改Employee employee = employeeService.get(id);model.addAttribute("employee", employee);}return "employee/input";}
​// ......
}

1.2、修改员工 input.jsp

把原来 input 改成 select,并使用 JSTL 遍历部门数据。

部门:<select name="dept.id"><c:forEach items="${departments}" var="department"><option value="${department.id}">${department.name}</option></c:forEach>
</select>

1.3、修改员工 Mapper XML

因为之前我们把员工实体类的 Long 类型的 deptId 属性改成了 Department 类型的 department 属性,所以要修改员工保存的代码,不然报找不到属性的错误。

<insert id="insert" parameterType="cn.domain.Employee" useGeneratedKeys="true" keyProperty="id">INSERT INTO employee (username, name, password, age, email, admin, dept_id)VALUES (#{username}, #{name}, #{password}, #{age}, #{email}, #{admin}, #{dept.id})
</insert>

2、员工修改完善

2.1 修改的时候部门得回显

2.1.1、修改根据 id 查询的 SQL
<select id="selectByPrimaryKey" resultMap="BaseResultMap">SELECT id, username, name, password, age, email, admin, dept_id AS d_idFROM employee eWHERE e.id = #{id}
</select>
2.1.2、修改员工 input.jsp
部门:<select name="dept.id"><c:forEach items="${departments}" var="department"><option value="${department.id}" ${department.id == employee.dept.id ? 'selected' : ''}>${department.name}</option></c:forEach>
</select>

2.2、修改员工 Mapper XML

  • 修改部门 id 值的获取。

  • 去除密码的修改。

<update id="updateByPrimaryKey" parameterType="cn.domain.Employee">UPDATE employeeSET username = #{username}name = #{name},age = #{age},email = #{email},admin = #{admin},dept_id = #{dept.id}WHERE id = #{id}
</update>

这篇关于SSM Spring MVC集成mybatis的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1146711

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

mybatis的整体架构

mybatis的整体架构分为三层: 1.基础支持层 该层包括:数据源模块、事务管理模块、缓存模块、Binding模块、反射模块、类型转换模块、日志模块、资源加载模块、解析器模块 2.核心处理层 该层包括:配置解析、参数映射、SQL解析、SQL执行、结果集映射、插件 3.接口层 该层包括:SqlSession 基础支持层 该层保护mybatis的基础模块,它们为核心处理层提供了良好的支撑。

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听