spring boot2 整合(三)JOOQ工具

先來介紹下jooq

jOOQ是一個基于Java編寫SQL的工具包,具有:簡單、輕量、函數(shù)式編程寫SQL等獨特優(yōu)勢,非常適合敏捷快速迭代開發(fā)。

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修改
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>springboot-jooq</groupId>
    <artifactId>springboot-jooq</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>springboot-jooq</name>
    <description>Demo project for Spring Boot</description>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.0.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-jooq</artifactId>
        </dependency>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.16.18</version>
        </dependency>

        <!-- 阿里巴巴fastjson,解析json視圖 -->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.15</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.1.3</version>
        </dependency>
        <dependency>
            <groupId>org.jooq</groupId>
            <artifactId>jooq-meta</artifactId>
        </dependency>
        <dependency>
            <groupId>org.jooq</groupId>
            <artifactId>jooq-codegen</artifactId>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
            </plugin>

            <plugin>
                <groupId>org.jooq</groupId>
                <artifactId>jooq-codegen-maven</artifactId>
                <version>${jooq.version}</version>
                <executions>
                    <execution>
                        <goals>
                            <goal>generate</goal>
                        </goals>
                    </execution>
                </executions>
                <dependencies>
                    <dependency>
                        <groupId>mysql</groupId>
                        <artifactId>mysql-connector-java</artifactId>
                        <version>${mysql.version}</version>
                    </dependency>
                </dependencies>
                <configuration>
                    <configurationFile>src/main/resources/JooqConfig.xml</configurationFile>
                </configuration>
            </plugin>
        </plugins>
    </build>


</project>

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

JooqConfig.xml

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<configuration>
    <jdbc>
        <driver>com.mysql.jdbc.Driver</driver>
        <url>jdbc:mysql://localhost:3306/user</url>
        <user>root</user>
        <password>root</password>
    </jdbc>
    <generator>
        <!-- 代碼生成器 -->
        <name>org.jooq.util.JavaGenerator</name>
        <database>
            <!--下面這兩行是為view也生成代碼的關(guān)鍵-->
            <!--force generating id'sfor everything in public schema, that has an 'id' field-->
            <syntheticPrimaryKeys>public\..*\.id</syntheticPrimaryKeys>
            <!--name for fake primary key-->
            <overridePrimaryKeys>override_primmary_key</overridePrimaryKeys>

            <name>org.jooq.util.mysql.MySQLDatabase</name>

            <!--include和exclude用于控制為數(shù)據(jù)庫中哪些表生成代碼-->
            <includes>.*</includes>
            <!--<excludes></excludes>-->

            <!--數(shù)據(jù)庫名稱-->
            <inputSchema>user</inputSchema>
        </database>

        <generate>
            <!--生成dao和pojo-->
            <daos>true</daos>
            <pojos>true</pojos>
            <!--把數(shù)據(jù)庫時間類型映射到j(luò)ava 8時間類型-->
            <javaTimeTypes>true</javaTimeTypes>
            <!--<interfaces>true</interfaces>-->
            <!--不在生成的代碼中添加spring注釋,比如@Repository-->
            <springAnnotations>false</springAnnotations>
        </generate>

        <target>
            <!--生成代碼文件的包名及放置目錄-->
            <packageName>com.generator</packageName>
            <directory>src/main/java</directory>
        </target>
    </generator>
</configuration>
2.2 mvn執(zhí)行逆向工程

先進行clean 再compile,然后我們就會發(fā)現(xiàn),生成了很多個類


如果你看到了這些,那么恭喜。你的jooq環(huán)境搭建成功了!

三層架構(gòu)

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);
    /** 查詢?nèi)苛斜?/
    public Iterator<User> 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;*/
    }

    /**
     * 查詢?nèi)苛斜?     *  @param pageNum
     * @param pageSize
     */
    @Override
    public Iterator<com.fantj.pojos.User> selectAll(int pageNum, int pageSize) {
        Result result = dsl.select().from(u).fetch();

        return result.iterator();
    }
}

我對幾段代碼做點解釋

  @Autowired
    DSLContext dsl;

這里是注入DSL上下文對象,DSLContextl里面有connect對象,大概猜測的話應(yīng)該是與數(shù)據(jù)庫連接交互的一個對象。

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

這段代碼的意思是給User表 重命名 u 。(類似sql語句中的 user as u)。
但是注意一點,這個User類是逆向生成的tables包下的,不是pojos包下的User實體類。
(逆向工程它會生成兩個User類。一個在pojos下,一個再tables下)。

還有,我在這里只測試了一個方法selectById(),別的我沒有測試,大概應(yīng)該差不多。
好了,讓大家看看方法selectById()的運行結(jié)果。


控制臺打印分別對應(yīng)上面代碼中的這兩個語句

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這個對象轉(zhuǎn)換成我們想要的pojos包下的User實體類。
但是查詢的功能是實現(xiàn)了,希望有能力大佬再研究和試試。

我也試過用它逆向生成的UserDao(我再上面代碼中注釋掉的)。結(jié)果報錯:
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<User> selectAll(@PathVariable("pageNum") int pageNum, @PathVariable("pageSize") int pageSize){
        Iterator<User> userIterator = userService.selectAll(pageNum, pageSize);
        List<User> list = new ArrayList<>();
        while(userIterator.hasNext()){
            list.add(userIterator.next());
        }
        return list;
    }

}

好了,這是我8個小時琢磨的結(jié)果,因為ssm整合jooq資料真的很少。springboot更是少。但是可以說完成了一大半。再總結(jié)一下遺留的兩個問題:

  1. UserDao.java 是干什么的
  2. Result 對象怎么 轉(zhuǎn)換成 javabean 實體類
問題2已得道解決

謝謝QQ號為523309375的朋友的提示。

自己也是耐心耗到極限,沒有仔細再研究Result這個接口,它里面有info方法??梢赞D(zhuǎn)換成很多格式,我在這里選擇<E> List<E> into(Class<? extends E> var1) throws MappingException;方法來返回一個List<com.fantj.pojos.User>。

修改之后的兩個ServiceImpl里的方法。

    /**
     * 查詢單個
     *
     * @param id
     */
    @Override
    public com.fantj.pojos.User selectById(int id) {
        List<com.fantj.pojos.User> 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;*/
    }

    /**
     * 查詢?nèi)苛斜?     * @param pageNum
     * @param pageSize
     */
    @Override
    public List<com.fantj.pojos.User> selectAll(int pageNum, int pageSize) {
        List<com.fantj.pojos.User> list = dsl.select().from(u)
                .orderBy(u.ID.desc())   //id倒序
                .limit(0)   //分頁
                .offset(10)   //分頁
                .fetch()
                .into(com.fantj.pojos.User.class);  //數(shù)據(jù)類型格式轉(zhuǎn)化

        return list;
    }

再說說我現(xiàn)在對jooq的理解。我個人覺得,如果再對jooq進行一些與ssm整合上的優(yōu)化。我們可以在serviceImpl里寫sql偽代碼,達到快速開發(fā)。仔細想想,JPA和Mybatis對復(fù)雜sql的支持還是挺不方便的。如果能再serviceImpl層里直接寫sql語句豈不更好。個人認為jooq很接近。
哈哈,不得不說這種代碼格式看上去很舒服。但是有點打破經(jīng)典三層架構(gòu)的意思,當然自然的也失去了代碼耦合性(對于大型項目而言),希望大家能提出自己好的想法。

謝謝大家賞臉!

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

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容