spring boot2 整合(三)JOOQ工具

先来介绍下jooq

jOOQ是一个基于Java编写SQL的工具包,具有:简单、轻量、函数式编程写SQL等独特优势,非常适合敏捷快速迭代开发。

SQL语句:

SELECT AUTHOR.FIRST_NAME, AUTHOR.LAST_NAME, COUNT(*)
    FROM AUTHOR
    JOIN BOOK ON AUTHOR.ID = BOOK.AUTHOR_ID
   WHERE BOOK.LANGUAGE = 'DE'
     AND BOOK.PUBLISHED > DATE '2008-01-01'
GROUP BY AUTHOR.FIRST_NAME, AUTHOR.LAST_NAME
  HAVING COUNT(*) > 5
ORDER BY AUTHOR.LAST_NAME ASC NULLS FIRST
   LIMIT 2
  OFFSET 1  

Java代码:

create.select(AUTHOR.FIRST_NAME, AUTHOR.LAST_NAME, count())
      .from(AUTHOR)
      .join(BOOK).on(AUTHOR.ID.equal(BOOK.AUTHOR_ID))
      .where(BOOK.LANGUAGE.eq("DE"))
      .and(BOOK.PUBLISHED.gt(date("2008-01-01")))
      .groupBy(AUTHOR.FIRST_NAME, AUTHOR.LAST_NAME)
      .having(count().gt(5))
      .orderBy(AUTHOR.LAST_NAME.asc().nullsFirst())
      .limit(2)
      .offset(1)

那从这里开始正式开始整合

1. pom修改


    4.0.0

    springboot-jooq
    springboot-jooq
    0.0.1-SNAPSHOT
    jar

    springboot-jooq
    Demo project for Spring Boot

    
        org.springframework.boot
        spring-boot-starter-parent
        2.0.0.RELEASE
         
    

    
        UTF-8
        UTF-8
        1.8
    

    
        
            org.springframework.boot
            spring-boot-starter
        

        
            org.springframework.boot
            spring-boot-starter-web
        
        
            mysql
            mysql-connector-java
        
        
            org.springframework.boot
            spring-boot-starter-jooq
        

        
            org.projectlombok
            lombok
            1.16.18
        

        
        
            com.alibaba
            fastjson
            1.2.15
        
        
            com.alibaba
            druid
            1.1.3
        
        
            org.jooq
            jooq-meta
        
        
            org.jooq
            jooq-codegen
        
    

    
        
            
                org.apache.maven.plugins
                maven-compiler-plugin
            

            
                org.jooq
                jooq-codegen-maven
                ${jooq.version}
                
                    
                        
                            generate
                        
                    
                
                
                    
                        mysql
                        mysql-connector-java
                        ${mysql.version}
                    
                
                
                    src/main/resources/JooqConfig.xml
                
            
        
    




2. 逆向工程
2.1 逆向配置文件

JooqConfig.xml



    
        com.mysql.jdbc.Driver
        jdbc:mysql://localhost:3306/user
        root
        root
    
    
        
        org.jooq.util.JavaGenerator
        
            
            
            public\..*\.id
            
            override_primmary_key

            org.jooq.util.mysql.MySQLDatabase

            
            .*
            

            
            user
        

        
            
            true
            true
            
            true
            
            
            false
        

        
            
            com.generator
            src/main/java
        
    

2.2 mvn执行逆向工程
spring boot2 整合(三)JOOQ工具_第1张图片

先进行clean 再compile,然后我们就会发现,生成了很多个类


spring boot2 整合(三)JOOQ工具_第2张图片

如果你看到了这些,那么恭喜。你的jooq环境搭建成功了!

三层架构

service接口

UserService .java

package com.fantj.service;

import com.fantj.pojos.User;

import java.util.Iterator;

/**
 * Created by Fant.J.
 */
public interface UserService {
    /** 删除 */
    public void delete(int id);
    /** 增加*/
    public void insert(User user);
    /** 更新*/
    public int update(User user);
    /** 查询单个*/
    public User selectById(int id);
    /** 查询全部列表*/
    public Iterator selectAll(int pageNum, int pageSize);
}

serviceImpl

UserServiceImpl .java

package com.fantj.service.impl;

/**
 * Created by Fant.J.
 */
@Service
public class UserServiceImpl implements UserService {

    @Autowired
    DSLContext dsl;
/*
    @Autowired
    private UserDao userDao;*/

    com.generator.tables.User u =  User.USER_.as("u");
    /**
     * 删除
     *
     * @param id
     */
    @Override
    public void delete(int id) {
        dsl.delete(u).where(u.ID.eq(id));
    }

    /**
     * 增加
     *
     * @param user
     */
    @Override
    public void insert(com.fantj.pojos.User user) {
        dsl.insertInto(u).
                columns(u.ADDRESS,u.BIRTHDAY,u.SEX,u.USERNAME).
                values(user.getAddress(),user.getBirthday(),user.getSex(),user.getUsername())
                .execute();
    }

    /**
     * 更新
     *
     * @param user
     */
    @Override
    public int update(com.fantj.pojos.User user) {
        dsl.update(u).set((Record) user);
        return 0;
    }

    /**
     * 查询单个
     *
     * @param id
     */
    @Override
    public com.fantj.pojos.User selectById(int id) {
        Result result =  dsl.select(u.ADDRESS,u.BIRTHDAY,u.ID,u.SEX,u.USERNAME)
                .from(u)
                .where(u.ID.eq(id)).fetch();
        System.out.println(result.get(0).toString());
        String className = result.get(0).getClass().getName();
        System.out.println(className);
        com.fantj.pojos.User user = new com.fantj.pojos.User();
        return null;
        /*com.fantj.pojos.User user1 = userDao.findById(id);
        return user1;*/
    }

    /**
     * 查询全部列表
     *  @param pageNum
     * @param pageSize
     */
    @Override
    public Iterator selectAll(int pageNum, int pageSize) {
        Result result = dsl.select().from(u).fetch();

        return result.iterator();
    }
}

我对几段代码做点解释

  @Autowired
    DSLContext dsl;

这里是注入DSL上下文对象,DSLContextl里面有connect对象,大概猜测的话应该是与数据库连接交互的一个对象。

com.generator.tables.User u =  User.USER_.as("u");

这段代码的意思是给User表 重命名 u 。(类似sql语句中的 user as u)。
但是注意一点,这个User类是逆向生成的tables包下的,不是pojos包下的User实体类。
(逆向工程它会生成两个User类。一个在pojos下,一个再tables下)。

还有,我在这里只测试了一个方法selectById(),别的我没有测试,大概应该差不多。
好了,让大家看看方法selectById()的运行结果。

spring boot2 整合(三)JOOQ工具_第3张图片

控制台打印分别对应上面代码中的这两个语句

System.out.println(result.get(0).toString());  //获取result对象中的第一个对象并打印toString
String className = result.get(0).getClass().getName();  //获取Result第一个对象的类类型
System.out.println(className);

我目前还没有把org.jooq.impl.RecordImpl这个对象转换成我们想要的pojos包下的User实体类。
但是查询的功能是实现了,希望有能力大佬再研究和试试。

我也试过用它逆向生成的UserDao(我再上面代码中注释掉的)。结果报错:
org.jooq.exception.DetachedException: Cannot execute query. No Connection configured
意思是没有获取到连接配置信息。这块也没能搞懂。希望大佬也能在下面评论。

controller

UserController .java

package com.fantj.controller;

/**
 * Created by Fant.J.
 */
@RestController
@RequestMapping("/user")
public class UserController {

    @Autowired
    private UserService userService;

    @RequestMapping(method = RequestMethod.GET,value = "/delete/{id}")
    public void delete(@PathVariable("id")int id){
        userService.delete(id);
    }

    @RequestMapping(method = RequestMethod.POST,value = "/insert")
    public void insert(User user){
        userService.insert(user);
    }
    @RequestMapping(method = RequestMethod.POST,value = "/update/{id}")
    public void update(@RequestParam User user){
        userService.update(user);
    }

    @RequestMapping(method = RequestMethod.GET,value = "/{id}/select")
    public User select(@PathVariable("id")int id){
        return userService.selectById(id);
    }

    @RequestMapping(method = RequestMethod.GET,value = "/selectAll/{pageNum}/{pageSize}")
    public List selectAll(@PathVariable("pageNum") int pageNum, @PathVariable("pageSize") int pageSize){
        Iterator userIterator = userService.selectAll(pageNum, pageSize);
        List list = new ArrayList<>();
        while(userIterator.hasNext()){
            list.add(userIterator.next());
        }
        return list;
    }

}

好了,这是我8个小时琢磨的结果,因为ssm整合jooq资料真的很少。springboot更是少。但是可以说完成了一大半。再总结一下遗留的两个问题:

  1. UserDao.java 是干什么的
  2. Result 对象怎么 转换成 javabean 实体类
问题2已得道解决

谢谢QQ号为523309375的朋友的提示。

自己也是耐心耗到极限,没有仔细再研究Result这个接口,它里面有info方法。可以转换成很多格式,我在这里选择 List into(Class var1) throws MappingException;方法来返回一个List

修改之后的两个ServiceImpl里的方法。

    /**
     * 查询单个
     *
     * @param id
     */
    @Override
    public com.fantj.pojos.User selectById(int id) {
        List result =  dsl.select(u.ADDRESS,u.BIRTHDAY,u.ID,u.SEX,u.USERNAME)
                .from(u)
                .where(u.ID.eq(id))
                .fetch()
                .into(com.fantj.pojos.User.class);

        return result.get(0);
        /*com.fantj.pojos.User user1 = userDao.findById(id);
        return user1;*/
    }

    /**
     * 查询全部列表
     * @param pageNum
     * @param pageSize
     */
    @Override
    public List selectAll(int pageNum, int pageSize) {
        List list = dsl.select().from(u)
                .orderBy(u.ID.desc())   //id倒序
                .limit(0)   //分页
                .offset(10)   //分页
                .fetch()
                .into(com.fantj.pojos.User.class);  //数据类型格式转化

        return list;
    }
spring boot2 整合(三)JOOQ工具_第4张图片

spring boot2 整合(三)JOOQ工具_第5张图片

再说说我现在对jooq的理解。我个人觉得,如果再对jooq进行一些与ssm整合上的优化。我们可以在serviceImpl里写sql伪代码,达到快速开发。仔细想想,JPA和Mybatis对复杂sql的支持还是挺不方便的。如果能再serviceImpl层里直接写sql语句岂不更好。个人认为jooq很接近。
哈哈,不得不说这种代码格式看上去很舒服。但是有点打破经典三层架构的意思,当然自然的也失去了代码耦合性(对于大型项目而言),希望大家能提出自己好的想法。

谢谢大家赏脸!

源码地址:https://github.com/jiaofanting/springboot-integration/tree/master/springboot-jooq

你可能感兴趣的:(spring boot2 整合(三)JOOQ工具)