flea-db使用之JPA接入

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS PostgreSQL,集群系列 2核4GB
简介: 【4月更文挑战第4天】本篇 Huazie 介绍 Flea 框架下的 flea-db模块中如何接入JPA

jpa.jpeg

引言

本节内容需要了解JPA封装内容,请参见笔者上篇博文《JPA封装介绍》

1. 准备工作

为了演示JPA接入,需要先准备如下:

  • MySQL 数据库 (客户端可以使用 navicat for mysql)
  • 新建测试数据库 fleajpatest
    image.png

  • 新建测试表 student

    image.png

    建表语句如下:

      DROP TABLE IF EXISTS `student`;
      CREATE TABLE `student` (
        `stu_id` int(10) NOT NULL AUTO_INCREMENT COMMENT '学生编号',
        `stu_name` varchar(255) NOT NULL COMMENT '学生姓名',
        `stu_age` tinyint(2) NOT NULL COMMENT '学生年龄',
        `stu_sex` tinyint(1) NOT NULL COMMENT '学生性别(1:男 2:女)',
        `stu_state` tinyint(2) NOT NULL COMMENT '学生状态(0:删除 1:在用)',
        PRIMARY KEY (`stu_id`)
      ) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8;
    
  • 依赖
    MySQLJDBC 驱动 mysql-connector-java-5.1.25.jar

    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>5.1.25</version>
    </dependency>
    

    FLEA DB ECLIPSELINK【这里为 flea-framework 项目下的包,目前版本 2.0.0,暂未发布】

    <dependency>
        <groupId>com.huazie.fleaframework</groupId>
        <artifactId>flea-db-eclipselink</artifactId>
        <version>2.0.0</version>
    </dependency>
    

2. 接入讲解

2.1 实体类

新建如下学生表对应的实体类 Student,对应测试表 student

@Entity
@Table(name = "student")
public class Student implements FleaEntity {
   

    private static final long serialVersionUID = 1267943552214677159L;

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY, generator = "STUDENT_SEQ")
    @SequenceGenerator(name = "STUDENT_SEQ")
    @Column(name = "stu_id", unique = true, nullable = false)
    private Long stuId;

    @Column(name = "stu_name", nullable = false)
    private String stuName;

    @Column(name = "stu_age", nullable = false)
    private Integer stuAge; 

    @Column(name = "stu_sex", nullable = false)
    private Integer stuSex;

    @Column(name = "stu_state", nullable = false)
    private Integer stuState;

    // ... 省略get和set方法

    @Override
    public String toString() {
   
        return ToStringBuilder.reflectionToString(this);
    }
}
  • Long stuId : 学生编号【主键】。在笔者的《JPA主键生成策略介绍》 中,介绍了 GenerationType.IDENTITY【适用于支持 主键自增长 的数据库系统,比如 MySQL】,详细内容可自行查看。
  • String stuName : 学生姓名【非空】
  • Integer stuAge : 学生年龄【非空】
  • Integer stuSex : 学生性别(1:男 2:女)【非空】
  • Integer stuState : 学生状态(0:删除 1:在用)【非空】

2.2 持久化单元DAO层实现

上篇博文说到,增加一个持久化单元配置,便需要增加一个持久化单元 DAO 层实现。针对本次演示新增持久化单元 fleajpa,持久化配置文件 fleajpa-persistence.xml, Spring 配置中新增数据库事务管理者配置,相关内容可参考上一篇博文。下面贴出本次演示的持久化单元 DAO 层实现代码:

FleaJpa数据源DAO层父类

public class FleaJpaDAOImpl<T> extends AbstractFleaJPADAOImpl<T> {
   
    @PersistenceContext(unitName="fleajpa")
    protected EntityManager entityManager;

    @Override
    @Transactional("fleaJpaTransactionManager")
    public boolean remove(long entityId) throws Exception {
   
        return super.remove(entityId);
    }

    // ...其他实现省略

    @Override
    protected EntityManager getEntityManager() {
   
        return entityManager;
    }

}

我们来看上面两个注解,分别是:

  • @PersistenceContext(unitName="fleajpa") :持久化上下文注解,其值为持久化单元 unitName ,在持久化配置文件中定义,spring 配置中 JPA 实体管理器工厂初始化该参数。
  • @Transactional("fleaJpaTransactionManager") :事务注解,其值为持久化事务管理器, 在 spring 配置文件中定义。

2.3 配置介绍

详细配置信息,可以参考笔者上篇博文,这里不再赘述。
涉及文件 fleajpa-persistence.xmlapplicationContext.xml,文章最后会给出示例工程,可自行查看。

2.4 学生DAO层接口

IStudentDAO 继承了抽象 Flea JPA DAO 层接口,并定义了两个方法,分别获取学生信息列表(分页)和学生总数。

学生DAO层接口

public interface IStudentDAO extends IAbstractFleaJPADAO<Student> {
   
    List<Student> getStudentList(String name, Integer sex, Integer minAge, Integer maxAge, int pageNum, int pageCount) throws DaoException;

    int getStudentCount(String name, Integer sex, Integer minAge, Integer maxAge) throws DaoException;

}
  • getStudentList(String name, Integer sex, Integer minAge, Integer maxAge, int pageNum, int pageCount) :获取学生信息列表 (分页),其中参数如下:
    • name : 学生姓名,可以模糊查询
    • sex : 性别
    • minAge : 最小年龄
    • maxAge : 最大年龄
    • pageNum : 查询页
    • pageCount : 每页总数
  • getStudentCount(String name, Integer sex, Integer minAge, Integer maxAge) :获取学生总数
    • name : 学生姓名,可以模糊查询
    • sex : 性别
    • minAge : 最小年龄
    • maxAge : 最大年龄

2.5 学生DAO层实现

StudentDAOImpl 是学生信息的数据操作层实现,继承持久化单元DAO层实现类,并实现了上述学生DAO层接口自定义的两个方法。 具体如何使用 FleaJPAQuery 可以参见下面代码 :

学生DAO层实现类

@Repository("studentDAO")
public class StudentDAOImpl extends FleaJpaDAOImpl<Student> implements IStudentDAO {
   

    @Override
    @SuppressWarnings(value = "unchecked")
    public List<Student> getStudentList(String name, Integer sex, Integer minAge, Integer maxAge, int pageNum, int pageCount) throws DaoException {
   
        FleaJPAQuery query = initQuery(name, sex, minAge, maxAge, null);
        List<Student> studentList;
        if (pageNum > 0 && pageCount > 0) {
   
            // 分页查询
            studentList = query.getResultList4Page(pageNum, pageCount);
        } else {
   
            // 全量查询
            studentList = query.getResultList();
        }
        return studentList;
    }

    @Override
    @SuppressWarnings(value = "unchecked")
    public long getStudentCount(String name, Integer sex, Integer minAge, Integer maxAge) throws DaoException {
   
        Object result = initQuery(name, sex, minAge, maxAge, Long.class).count().getSingleResult();
        return Long.parseLong(StringUtils.valueOf(result));
    }

    private FleaJPAQuery initQuery(String name, Integer sex, Integer minAge, Integer maxAge, Class<?> result) throws DaoException{
   
        return getQuery(result)
                .like("stuName", name)
                .equal("stuSex", sex)
                .ge("stuAge", minAge)
                .le("stuAge", maxAge);
    }
}

上述代码,我们需要重点关注 initQuery 私有方法,它实际上用于返回一个已经组装好查询条件的 FleaJPAQuery 对象:

  • getQuery(result) :在《flea-db使用之JPA封装介绍》 中的抽象 Flea JPA DAO 层实现可以看到,通过 Flea JPA 查询对象池来获取 FleaJPAQuery
  • like("stuName", name) :根据姓名模糊查询, attrName 为 实体类对应的成员变量名,并非表字段名
  • equal("stuSex", sex) :查询性别(等于)
  • ge("stuAge", minAge) :查询年龄范围 (大于等于)
  • le("stuAge", maxAge) :查询年龄范围 (小于等于)

另外,我们看到 学生DAO层实现类 的上面还有一个注解:

  • @Repository("studentDAO") :在 Spring 框架中,它是用来标注数据访问层(DAO层)的类。Spring 会将 StudentDAOImpl 实例化为一个名为 studentDAO 的Bean;然后,在其他Bean 中通过@Autowired 或者 @Resource 注解来注入这个 Bean

2.6 学生SV层接口

IStudentSV 继承抽象Flea JPA SV层接口,并定义两个方法,分别获取学生信息列表(分页)和学生总数。

学生SV层接口


public interface IStudentSV extends IAbstractFleaJPASV<Student> {
   
    List<Student> getStudentList(String name, Integer sex, Integer minAge, Integer maxAge, int pageNum, int pageCount) throws DaoException;

    long getStudentCount(String name, Integer sex, Integer minAge, Integer maxAge) throws DaoException;
}

2.7 学生SV层实现

StudentSVImpl 继承抽象Flea JPA SV层实现类,并实现了上述学生SV层接口的两个自定义方法。具体实现参见如下代码:

学生SV层实现类

@Service("studentSV")
public class StudentSVImpl extends AbstractFleaJPASVImpl<Student> implements IStudentSV {
   
    // 注入学生DAO层实现类
    @Autowired
    @Qualifier("studentDAO") 
    private IStudentDAO studentDao; 

    @Override
    public List<Student> getStudentList(String name, Integer sex, Integer minAge, Integer maxAge, int pageNum, int pageCount) throws DaoException {
   
        return studentDao.getStudentList(name, sex, minAge, maxAge, pageNum, pageCount);
    }

    @Override
    public long getStudentCount(String name, Integer sex, Integer minAge, Integer maxAge) throws DaoException {
   
        return studentDao.getStudentCount(name, sex, minAge, maxAge);
    }

    @Override
    protected IAbstractFleaJPADAO<Student> getDAO() {
   
        return studentDao;
    }
}

上述逻辑不复杂,学生SV层实现类中的两个方法均调用 IStudentDAO 接口来查询数据库。

上述代码需要关注三个注解 和 一个方法:

  • @Service("studentSV") :在 Spring 框架中,它通常用于将一个服务层的类标记为 Spring 管理的 BeanstudentSV 是这个 Bean 的名称,可以在其他地方通过这个名称来获取这个 Bean 的实例。当 Spring 容器启动并扫描到带有 @Service 注解的类时,它会创建这个类的实例,并将其注册到Spring应用程序上下文中,使得这个 Bean 可以被依赖注入(DI)到其他组件中。
  • @Autowired :在 Spring 框架中,它用于自动装配 Bean
  • @Qualifier("studentDAO") :它通常与 @Autowired 一起使用,用于指定 @Autowired 应该注入的特定 Bean。这里就指定了一个名为 studentDAOBean。如果你不使用 @Qualifier 注解,而是仅仅依赖 @Autowired 来注入 Bean,那么在存在多个相同类型 Bean 的情况下,Spring 容器将无法确定应该注入哪一个,从而导致 NoUniqueBeanDefinitionException 异常。
  • getDAO() :在《flea-db使用之JPA封装介绍》 中的抽象Flea JPA SV层实现,可以看到 getDAO() 用于通过的一些增删改查操作,实际的实现需要子类来返回对应的 DAO层 实现。

2.8 JPA接入自测

首先添加自测类,并注入学生服务类

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {
   "classpath:applicationContext.xml"})
public class StudentTest {
   

    private static final Logger LOGGER = FleaLoggerProxy.getProxyInstance(StudentTest.class);

    @Resource(name = "studentSV")
    private IStudentSV studentSV;

    // 自测代码详见下面

}

2.8.1 新增学生信息

    @Test
    public void testInsertStudent() {
   
        try {
   
            Student student = new Student();
            student.setStuName("张三");
            student.setStuAge(18);
            student.setStuSex(1);
            student.setStuState(1);
            studentSV.save(student);

            student = new Student();
            student.setStuName("李四");
            student.setStuAge(19);
            student.setStuSex(1);
            student.setStuState(1);
            studentSV.save(student);

            student = new Student();
            student.setStuName("王二麻子");
            student.setStuAge(20);
            student.setStuSex(1);
            student.setStuState(1);
            studentSV.save(student);

        } catch (Exception e) {
   
            LOGGER.error("Exception : ", e);
        }
    }

执行结果:

image.png

2.8.2 更新学生信息

    @Test
    public void testStudentUpdate() {
   
        try {
   
            Student student = studentSV.query(3L);
            LOGGER.debug("Before : {}", student);
            student.setStuName("王三麻子");
            student.setStuAge(19);
            studentSV.update(student);
            student = studentSV.query(3L);
            LOGGER.debug("After : {}", student);
        } catch (Exception e) {
   
            LOGGER.error("Exception : ", e);
        }
    }

上述演示代码逻辑:

  • 首先根据指定主键,调用 query 方法查询学生信息,并打印 Before :XXX
  • 然后调用 update 方法更新该学生信息;
  • 最后再根据指定主键,调用 query 方法查询学生信息,并打印After : XXX

运行结果:

image.png

2.8.3 删除学生信息

    @Test
    public void testStudentDelete() {
   
        try {
   
            Student student = studentSV.query(3L);
            LOGGER.debug("Before : {}", student);
            // 
            studentSV.remove(3L);
            // 最后再根据主键查询学生信息
            student = studentSV.query(3L);
            LOGGER.debug("After : {}", student);
        } catch (Exception e) {
   
            LOGGER.error("Exception : ", e);
        }
    }

上述演示代码逻辑:

  • 首先根据指定主键,调用 query 方法查询学生信息,并打印 Before :XXX
  • 然后调用 remove 方法删除指定主键的学生信息(里面会先去将学生实体信息查出来,然后再删除);
  • 最后再根据指定主键,调用 query 方法查询学生信息,并打印After : XXX.

运行结果:

image.png

2.8.4 查询学生信息(按条件分页查询)

表里自行再插入些数据,用于测试查询,查询结果因各自表数据而异;

目前我表中数据如下:

image.png

      @Test
    public void testStudentQueryPage() {
   
        try {
   

            IStudentSV studentSV = (IStudentSV) applicationContext.getBean("studentSV");
            List<Student> studentList = studentSV.getStudentList("张三", 1, 18, 20, 1, 5);
            LOGGER.debug("Student List = {}", studentList);
        } catch (Exception e) {
   
            LOGGER.error("Exception : ", e);
        }
    }

运行结果:

image.png

2.8.5 查询学生总数(按条件查询)

    @Test
    public void testStudentQueryCount() {
   
        try {
   
            IStudentSV studentSV = (IStudentSV) applicationContext.getBean("studentSV");
            long count = studentSV.getStudentCount("张三", 1, 18, 20);
            LOGGER.debug("Student Count = {}", count);
        } catch (Exception e) {
   
            LOGGER.error("Exception : ", e);
        }
    }

运行结果:

image.png

总结

看到这里,我们的 JPA接入 工作已经成功完成,本篇演示示例可以移步到 GitHub 查看 flea-jpa-test

JPA 封装介绍博文中,针对 Flea JPA 查询对象还存在的一个并发问题,将在后续的博文 《flea-db使用之基于对象池的FleaJPAQuery》 中介绍。

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
8月前
|
Java 关系型数据库 MySQL
flea-db使用之JPA封装介绍
【4月更文挑战第3天】本篇 Huazie 介绍 Flea 框架下的 flea-db模块中的 JPA 封装代码
104 6
flea-db使用之JPA封装介绍
|
8月前
|
Java 关系型数据库 MySQL
flea-db使用之封装JDBC接入
【4月更文挑战第6天】本篇 Huazie 介绍 Flea 框架下的 flea-db 模块中封装JDBC的逻辑
75 1
flea-db使用之封装JDBC接入
|
8月前
|
SQL Java 数据库
flea-db使用之JPA分库分表实现
【4月更文挑战第7天】本篇 Huazie 介绍 Flea 框架下的 flea-db 模块中JPA分库分表实现
106 1
flea-db使用之JPA分库分表实现
|
8月前
|
存储 Java 数据库
flea-db使用之基于对象池的FleaJPAQuery
【4月更文挑战第5天】本篇 Huazie 介绍 Flea 框架下的 flea-db 模块中基于对象池的 FleaJPAQuery
91 2
flea-db使用之基于对象池的FleaJPAQuery
|
消息中间件 存储 数据库
消息代理对比DB
有些消息代理甚至可使用 XA 或 JTA 参与两阶段提交协议。这和DB在本质相似,尽管消息代理和DB存在实践上很重要的差异
80 0
|
SQL IDE NoSQL
tp5源码解析--Db操作
在TP5的框架使用过程中,Db类是一定会接触到的,上手不难,但若想随心所欲的用,还是需要了解一番。用了千次,却没看过一次源码,学习源码,起码对TP5这个框架使用更加得心应手,毕竟技术服务于业务,能够写出更简介、更方便、更有效的业务代码,本身就是一件身心愉悦的事儿;
161 0
|
安全 数据库
数据库——报错:无法创建链接服务器 “(null)“ 的 OLE DB 访问接口 “Microsoft.Ace.OLEDB.12.0“ 的实例。
数据库——报错:无法创建链接服务器 “(null)“ 的 OLE DB 访问接口 “Microsoft.Ace.OLEDB.12.0“ 的实例。
数据库——报错:无法创建链接服务器 “(null)“ 的 OLE DB 访问接口 “Microsoft.Ace.OLEDB.12.0“ 的实例。
|
SQL 关系型数据库 数据库
【玩转DB2】二、DB2联邦详细操作和踩坑
建立联邦(建立数据库与数据库联系与问题) 有问题可以找小可玩,评论留言。什么是联邦??建立和另外一个库的关系,换句话说就是我可以在这个库查到另外一个库的表。 本地库连目标数据库1、登陆本地服务器数据库 切换到对应的用户:su - 本地库用户名 连接本地库:db2 connect to 本地库 user 本地库用户名 using 本地用户密码 2、开启联邦支持开启联邦支持,查看数据库管理配置文件,FEDERATED属性 db2 get dbm cfg 如联邦属性为NO, 则需开启属性,重启数据库。
4084 0
|
关系型数据库 Java 数据库连接
Hibernate连接DB2的问题(已解决)
折腾半个月了,这个问题一直没有得到解决。 我有两个web应用,它们使用同一个数据库,通过hibernate连接。应用A向数据库里插入、修改和删除数据,应用B读取数据展现给用户。在Mysql里使用一切正常,现在决定使用DB2数据库,理论上讲只要在DB2里建一个空数据库,然后修改hibernate.properties就可以了。
1412 0