All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.github.jinghui70.rainbow.dbaccess.Dba Maven / Gradle / Ivy

There is a newer version: 5.2.1
Show newest version
package io.github.jinghui70.rainbow.dbaccess;

import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.lang.Assert;
import cn.hutool.db.dialect.DriverUtil;
import io.github.jinghui70.rainbow.dbaccess.dialect.Dialect;
import io.github.jinghui70.rainbow.dbaccess.dialect.DialectDefault;
import io.github.jinghui70.rainbow.dbaccess.dialect.DialectOracle;
import io.github.jinghui70.rainbow.dbaccess.object.ObjectDao;
import io.github.jinghui70.rainbow.utils.StringBuilderX;
import org.springframework.dao.DataAccessException;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.namedparam.MapSqlParameterSource;
import org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate;
import org.springframework.jdbc.core.namedparam.SqlParameterSource;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.support.TransactionCallback;
import org.springframework.transaction.support.TransactionTemplate;

import javax.sql.DataSource;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Objects;

import static io.github.jinghui70.rainbow.dbaccess.DbaUtil.INSERT;
import static io.github.jinghui70.rainbow.dbaccess.DbaUtil.MERGE;

public class Dba {

    protected JdbcTemplate jdbcTemplate;

    protected NamedParameterJdbcTemplate namedParameterJdbcTemplate;

    protected TransactionTemplate transactionTemplate;

    protected Dialect dialect = DialectDefault.INSTANCE;

    protected Dba() {
    }

    protected void initDataSource(DataSource dataSource, Dialect dialect) {
        this.jdbcTemplate = new JdbcTemplate(dataSource);
        this.namedParameterJdbcTemplate = new NamedParameterJdbcTemplate(jdbcTemplate);
        DataSourceTransactionManager transactionManager = new DataSourceTransactionManager(dataSource);
        this.transactionTemplate = new TransactionTemplate(transactionManager);
        if (dialect != null)
            this.dialect = dialect;
        else
            initDialect();
    }

    public Dba(DataSource dataSource) {
        initDataSource(dataSource, null);
    }

    public Dba(DataSource dataSource, Dialect dialect) {
        initDataSource(dataSource, dialect);
    }

    public Dba(JdbcTemplate jdbcTemplate, NamedParameterJdbcTemplate namedParameterJdbcTemplate,
               TransactionTemplate transactionTemplate) {
        this.jdbcTemplate = jdbcTemplate;
        this.namedParameterJdbcTemplate = namedParameterJdbcTemplate;
        this.transactionTemplate = transactionTemplate;
        initDialect();
    }

    public Dba(JdbcTemplate jdbcTemplate, NamedParameterJdbcTemplate namedParameterJdbcTemplate,
               TransactionTemplate transactionTemplate, Dialect dialect) {
        this.jdbcTemplate = jdbcTemplate;
        this.namedParameterJdbcTemplate = namedParameterJdbcTemplate;
        this.transactionTemplate = transactionTemplate;
        this.dialect = dialect;
    }

    public String getDriver() {
        DataSource dataSource = Objects.requireNonNull(jdbcTemplate.getDataSource());
        return DriverUtil.identifyDriver(dataSource);
    }

    protected void initDialect() {
        String driver = getDriver();
        if (driver == null) return;
        if (driver.toLowerCase().contains("oracle"))
            this.dialect = new DialectOracle();
    }

    public JdbcTemplate getJdbcTemplate() {
        return jdbcTemplate;
    }

    public NamedParameterJdbcTemplate getNamedParameterJdbcTemplate() {
        return namedParameterJdbcTemplate;
    }

    public TransactionTemplate getTransactionTemplate() {
        return transactionTemplate;
    }

    public Dialect getDialect() {
        return dialect;
    }

    public Sql sql() {
        return new Sql(this);
    }

    public Sql sql(String sql) {
        return sql().append(sql);
    }

    public Sql select() {
        return sql("SELECT ");
    }

    public Sql selectAll() {
        return select("*");
    }

    public Sql select(String select) {
        return sql("SELECT ").append(select);
    }

    public  ObjectSql select(Class selectClass) {
        return new ObjectSql(this, selectClass).append("SELECT * FROM ").append(DbaUtil.tableName(selectClass));
    }

    public Sql update(String table) {
        return sql("UPDATE ").append(table);
    }

    public  ObjectSql update(Class updateClass) {
        return new ObjectSql(this, updateClass).append("UPDATE ").append(DbaUtil.tableName(updateClass));
    }

    public Sql deleteFrom(String table) {
        return sql("DELETE FROM ").append(table);
    }

    @SuppressWarnings("unchecked")
    public  int delete(T object) {
        return new ObjectDao(this, (Class) object.getClass()).delete(object);
    }

    public  ObjectSql deleteFrom(Class deleteClass) {
        return new ObjectSql(this, deleteClass).append("DELETE FROM ").append(DbaUtil.tableName(deleteClass));
    }

    public NamedSql namedSql() {
        return new NamedSql(this);
    }

    public NamedSql namedSql(String sql) {
        return new NamedSql(this).append(sql);
    }

    /**
     * 插入一个对象
     *
     * @param bean 需要插入的对象
     * @param   对象泛型
     * @return 插入改变的行数,正常应该是1
     */
    @SuppressWarnings("unchecked")
    public  int insert(T bean) {
        Assert.notNull(bean, "can't insert null object");
        return new ObjectDao<>(this, (Class) bean.getClass()).insert(bean);
    }

    /**
     * 插入一个对象,如果已经存在就更新。这个函数H2支持,别的数据库未必支持
     *
     * @param bean 需要插入的对象
     * @param   对象泛型
     * @return 插入改变的行数,正常应该是1
     */
    @SuppressWarnings("unchecked")
    public  int merge(T bean) {
        Assert.notNull(bean, "can't insert null object");
        return new ObjectDao<>(this, (Class) bean.getClass()).merge(bean);
    }

    /**
     * 插入一组数据
     *
     * @param beans 数据的集合
     * @param    对象泛型
     */
    public  void insert(List beans) {
        insert(beans, 0);
    }

    @SuppressWarnings("unchecked")
    public  void insert(List beans, int batchSize) {
        if (CollUtil.isEmpty(beans)) return;
        new ObjectDao<>(this, (Class) beans.get(0).getClass()).insert(beans, batchSize);
    }

    /**
     * 插入一组数据,如果已经存在就更新。这个函数H2支持,别的数据库未必支持
     *
     * @param beans 数据的集合
     * @param    对象泛型
     */
    @SuppressWarnings("unchecked")
    public  void merge(List beans) {
        if (CollUtil.isEmpty(beans)) return;
        new ObjectDao<>(this, (Class) beans.get(0).getClass()).merge(beans);
    }

    /**
     * 插入一个map到一个数据表中
     *
     * @param tableName 表名
     * @param map       map对象
     * @return 插入改变的行数,正常应该是1
     */
    public int insert(String tableName, Map map) {
        return doInsert(tableName, map, INSERT);
    }

    /**
     * 插入一条记录到指定的表里,如果已经存在就更新。这个函数H2支持,别的数据库未必支持
     *
     * @param tableName 数据表名
     * @param map       map对象
     * @return 插入改变的行数,正常应该是1
     */
    public int merge(String tableName, Map map) {
        return doInsert(tableName, map, MERGE);
    }

    public int doInsert(String tableName, Map map, String action) {
        Sql sql = sql(action).append(" into ").append(tableName).append("(");
        for (Map.Entry entry : map.entrySet()) {
            sql.append(entry.getKey()).appendTempComma().addParam(entry.getValue());
        }
        sql.clearTemp().append(") values (").repeat("?", map.size()).append(")");
        return sql.execute();
    }

    /**
     * 插入一组Map到指定的表里
     *
     * @param tableName 数据表名
     * @param data      数据列表
     */
    public void insert(String tableName, List> data) {
        doInsert(tableName, data, INSERT);
    }

    /**
     * 插入一组数据到指定的表里,如果已经存在就更新。这个函数H2支持,别的数据库未必支持
     *
     * @param tableName 数据表名
     * @param data      数据
     */
    public void merge(String tableName, List> data) {
        doInsert(tableName, data, MERGE);
    }

    private void doInsert(String tableName, List> data, String action) {
        if (CollUtil.isEmpty(data))
            return;
        List keys = new ArrayList<>(data.get(0).keySet());
        StringBuilderX sql = new StringBuilderX(action).append(" into ").append(tableName) //
                .append("(").join(keys).append(") values("); //
        for (String key : keys) {
            sql.append(":").append(key).appendTempComma();
        }
        sql.clearTemp().append(")");
        SqlParameterSource[] spss = data.stream().map(MapSqlParameterSource::new).toArray(SqlParameterSource[]::new);
        namedParameterJdbcTemplate.batchUpdate(sql.toString(), spss);
    }

    /**
     * 更新一个对象
     *
     * @param bean 待更新对象对象
     * @param   对象泛型
     * @return 更新数据库行数,正常情况下应该为1
     */
    @SuppressWarnings("unchecked")
    public  int update(T bean) {
        Assert.notNull(bean);
        return new ObjectDao<>(this, (Class) bean.getClass()).update(bean);
    }

    /**
     * 更新一个对象
     *
     * @param tableName 表名
     * @param bean      待更新对象对象
     * @param        对象泛型
     * @return 更新数据库行数,正常情况下应该为1
     */
    @SuppressWarnings("unchecked")
    public  int update(String tableName, T bean) {
        Assert.notNull(bean);
        return new ObjectDao<>(this, (Class) bean.getClass()).update(tableName, bean);
    }

    public  int deleteById(Class deleteClass, Object id) {
        return deleteFrom(DbaUtil.tableName(deleteClass)).where("id", id).execute();
    }

    public  int deleteByKey(Class deleteClass, Object... keys) {
        return new ObjectDao(this, deleteClass).deleteByKey(keys);
    }

    public  T selectById(Class selectClass, Object id) {
        return select(selectClass).where("id", id).queryForObject();
    }

    public  T selectByKey(Class selectClass, Object... keys) {
        return new ObjectDao(this, selectClass).selectByKey(keys);
    }
    
    /**
     * 检查一个数据表是否存在
     *
     * @param tableName 数据表名
     * @return 是否存在
     */
    public boolean exist(String tableName) {
        String sql = String.format("SELECT COUNT(*) FROM %s where 1!=1", tableName);
        try {
            jdbcTemplate.queryForObject(sql, Integer.class);
        } catch (DataAccessException e) {
            return false;
        }
        return true;
    }

    public void transaction(Runnable runnable) {
        transactionTemplate.execute(status -> {
            runnable.run();
            return null;
        });
    }

    public  T transaction(TransactionCallback action) {
        return transactionTemplate.execute(action);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy