Foxnic-SQL (7) —— DAO 特性 : 执行 SQL 语句

Foxnic-SQL (7) —— DAO 特性 : 执行 SQL 语句

概述

  Foxnic-SQL 支持多种语句执行方式,包括直接执行SQL字符串、执行SQL对象,SQL对象自执行,多语句执行与批量执行。Foxnic-SQL 显著的特征是 DAO 对象既可以执行字符串的 SQL 语句,也可以执行对象化的SQL语句。
本文中的示例代码均可在 https://gitee.com/LeeFJ/foxnic-samples 项目中找到。

执行SQL字符串

  SQL 执行最快捷、最能上手的方式,就是直接执行 SQL 字符串。DAO 对象可以执行不带参数或带参数的 SQL 字符串。如下代码所示:

/**
* 1、直接传入字符串的 SQL 语句,并执行
* */
public static void  demo1() {
    // 通过 DBInstance 拿到 DAO 对象
    DAO dao= DBInstance.DEFAULT.dao();
    // 生成 ID
    String id= IDGenerator.getNanoId(8);
    // 插入
    String insert="insert into example_address (id, name, phone_number, address, region_type, region_location, create_by, " +
        "create_time, update_by, update_time, deleted, delete_by, delete_time, version) " +
        "VALUES (?, ?, ?, ?, ?, NULL, NULL, ?, ?, ?, 0, NULL, NULL, 1)";
    int i=dao.execute(insert,id,"leefj","13852562523","宁波","国内",new Date(),"110", new Date());
    Logger.info("插入 : "+i);
    // 更新
    i=dao.execute("update example_address set address=? where id=?","上海",id);
    Logger.info("更新 : "+i);
    // 删除
    i=dao.execute("delete from example_address where id=?",id);
    Logger.info("删除 : "+i);
}

执行 SQL 对象

  DAO 对象同样支持对象话的SQL语句执行。SQL对象化的好处在之前的章节中已经介绍,感兴趣的读者可以查看 Foxnic-SQL 之前的文章《SQL表达式(Expr)》。示例代码如下所示:

import com.github.foxnic.commons.busi.id.IDGenerator;
import com.github.foxnic.dao.data.Rcd;
import com.github.foxnic.sql.expr.Delete;
import com.github.foxnic.sql.expr.Insert;
import com.github.foxnic.sql.expr.Select;
import com.github.foxnic.sql.expr.Update;
import com.leefj.foxnic.sql.demo.config.DBInstance;

import java.util.Date;

public class CRUDBySQLDemo {

    public static void main(String[] args) {
        // 插入数据
        String id=insertAddress("137771041252");
        System.out.println("addressId(Insert) = "+id);
        // 按ID查询数据
        Rcd address=queryAddress(id);
        if(address!=null) {
            System.out.println(address.toJSONObject());
        }
        // 更新
        if(id!=null) {
            id=updateAddress(id,"13852562523");
            System.out.println("addressId(Update) = "+id);
        }
        // 删除
        if(id!=null) {
            id=deleteAddress(id);
            System.out.println("addressId(Delete) = "+id);
        }
    }

    /**
    * 插入数据
    * */
    public static String insertAddress(String phone) {
        // 创建语句对象
        Insert insert=new Insert("example_address");
        String id= IDGenerator.getSnowflakeIdString();
        // 设置值
        insert.set("id",id)
            .set("name","leefj")
            // 如果是 null 则不连入SQL语句
            .setIf("phone_number",phone)
            .set("address","宁波")
            .set("region_type","国内")
            .set("create_time",new Date())
            // 设置数据库表达式
            .setExpr("update_time","now()");
        // 输出语句
        System.out.println(insert.getSQL());
        // 执行语句
        Integer suc=DBInstance.DEFAULT.dao().execute(insert);
        // 如果执行成功,返回ID,否则返回 null
        if(suc==1) {
            return id;
        } else {
            return null;
        }
    }

    /**
    * 查询
    * */
    public static Rcd queryAddress(String id) {

        // 创建语句对象
        Select select=new Select("example_address");
        // 设置值
        select.where().and("id=?",id);
        // 输出语句
        System.out.println(select.getSQL());
        // 执行语句
        Rcd address=DBInstance.DEFAULT.dao().queryRecord(select);
        // 如果执行成功,返回记录对象,否则返回 null
        return address;
    }

    /**
    * 更新
    * */
    public static String updateAddress(String id,String phone) {
        // 创建语句对象
        Update update=new Update("example_address");
        // 设置值
        update.setIf("phone_number",phone)
            // 设置数据库表达式
            .setExpr("update_time","now()")
            .set("update_by","110")
            .where().and("id=?",id);
        // 输出语句
        System.out.println(update.getSQL());
        // 执行语句
        Integer suc=DBInstance.DEFAULT.dao().execute(update);
        // 如果执行成功,返回ID,否则返回 null
        if(suc==1) {
            return id;
        } else {
            return null;
        }
    }

    /**
    * 删除
    * */
    public static String deleteAddress(String id) {
        // 创建语句对象
        Delete delete=new Delete("example_address");
    // 设置条件
    delete.where().and("id=?",id);
    // 输出语句
    System.out.println(delete.getSQL());
    // 执行语句
    Integer suc=DBInstance.DEFAULT.dao().execute(delete);
    // 如果执行成功,返回ID,否则返回 null
    if(suc==1) {
        return id;
    } else {
        return null;
    }
  }
}

ExecutableSQL 方式执行

  ExecutableSQL 是一个接口,所有实现 ExecutableSQL 接口的 SQL 类都具备语句的执行能力。各种类型的 ExecutableSQL 可以通过 DAO 对象直接创建。示例如下:

/**
* 3、利用SQL对象的  ExecutableSQL 特性执行
* */
public static void  demo3() {

    // 通过 DBInstance 拿到 DAO 对象
    DAO dao= DBInstance.DEFAULT.dao();
    // 生成 ID
    String id= IDGenerator.getNanoId(8);

    // 插入:通过 DAO 创建一个与 DAO 绑定的 Insert 语句对象
    int i=dao.insert("example_address")
        .set("id",id)
        .set("name","leefj")
        // 如果是 null 则不连入SQL语句
        .setIf("phone_number","13852562523")
        .set("address","宁波")
        .set("region_type","国内")
        .set("create_time",new Date())
        // 设置数据库表达式
        .setExpr("update_time","now()")
        // 执行语句
        .execute();
    Logger.info("插入 : "+i);

    // 更新
    i=dao.update("example_address").set("address","上海")
        .where("id=?",id)
        // 返回至顶层的 Update 语句对象
        .top()
        .execute();
    Logger.info("更新 : "+i);

    // 删除
    i=dao.delete("example_address")
        .where("id=?",id)
        // 返回至顶层的 Update 语句对象
        .top()
        .execute();
    Logger.info("删除 : "+i);
}

多语句执行

  多语句执行顾名思义就是将多个语句放在一起执行,它们将在一个事务内执行,在不调用事务接口时,可以使用该方法以支持事务。示例代码如下:

/**
* 多个语句一起执行,这些语句在一个事务内
* */
public static void demo1() {

    // 通过 DBInstance 拿到 DAO 对象
    DAO dao= DBInstance.DEFAULT.dao();
    // 生成 ID
    String id= IDGenerator.getNanoId(8);
    // 插入
    Expr insert=new Expr("insert into example_address (id, name, phone_number, address, region_type, region_location, create_by, create_time, update_by, update_time, deleted, delete_by, delete_time, version) " +
                         "VALUES (?, ?, ?, ?, ?, NULL, NULL, ?, ?, ?, 0, NULL, NULL, 1)",
                         id,"leefj","13852562523","宁波","国内",new Date(),"110", new Date());
    Expr update=new Expr("update example_address set address=? where id=?","上海",id);
    Expr delete=new Expr("delete from example_address where id=?",id);
    // 事务内同时执行多个语句,多参数并列
    Integer result=dao.multiExecute(insert,update,delete);
    // 把多个语句单独执行时影响的行数累加后返回
    Logger.info("result = "+result);

    // 产生一个 SQL 对象与字符串混合的 List
    String sqlstr="delete from example_address where id='"+id+"'";
    List sqls= Arrays.asList(insert,update,delete,sqlstr);
    // 事务内同时执行多个语句,传入列表,
    result=dao.multiExecute(sqls);
    // 把多个语句单独执行时影响的行数累加后返回
    Logger.info("result = "+result);

}

批量执行

  批量执行是在大量插入或更新数据是使用的一种高性能的执行方式,这种方式可以显著提高SQL的执行效率。下面这个示例同时用常规方法和批量执行插入数据,批量的时间只有占常规的1/3。

/**
* 批量执行
* */
public static void demo1() {

    // 通过 DBInstance 拿到 DAO 对象
    DAO dao= DBInstance.DEFAULT.dao();

    // 准备插入的 SQL 语句
    String insert="insert into example_address " +
        "(id, name, phone_number, address, region_type, region_location, create_by, create_time, " +
        "update_by, update_time, deleted, delete_by, delete_time, version) " +
        "values (?, ?, ?, ?, ?, null, null, ?, ?, ?, 0, null, null, 1)";
    // 性能日志对象
    PerformanceLogger logger=new PerformanceLogger();
    // 性能采集埋点
    logger.collect("常规插入开始");
    for (int i = 0; i < 100; i++) {
        String id="batch-"+IDGenerator.getNanoId(6);
        dao.execute(insert,id,"leefj","13852562523","宁波","国内",new Date(),"110", new Date());
    }
    // 性能采集埋点
    logger.collect("常规插入结束");
    // 性能采集埋点
    logger.collect("批量插入开始");
    BatchParamBuilder paramBuilder=new BatchParamBuilder();
    for (int i = 0; i < 100; i++) {
        String id="batch-"+IDGenerator.getNanoId(6);
        paramBuilder.add(id,"leefj","13852562523","宁波","国内",new Date(),"110", new Date());
    }
    int[] result=dao.batchExecute(insert,paramBuilder.getBatchList());
    // 性能采集埋点
    logger.collect("批量插入结束");
    // 打印性能对比
    logger.info("执行效率对比");
    // 输出结果
    for (int i : result) {
        System.out.println("结果:"+ i);
    }
}

  性能对比日志:

┏━━━ PERFORMANCE [ 执行效率对比 , total = 4240 ] ━━━
┣ point : 常规插入开始
┣━ cost : 2961
┣ point : 常规插入结束
┣━ cost : 0
┣ point : 批量插入开始
┣━ cost : 1279
┣ point : 批量插入结束
┗━━━ PERFORMANCE [ 执行效率对比 , total = 4240 ] ━━━

小结

  本节主要介绍了如何使用 Foxnic-SQL 的 DAO 对象使用不同的姿势执行 SQL 语句。本节中展示的例子主要目的是抛砖引玉,为了方便不同场景的调用,DAO 还提供了若干重载方法,以不同的参数形式去执行语句。

相关项目

  https://gitee.com/LeeFJ/foxnic
  https://gitee.com/LeeFJ/foxnic-web
  https://gitee.com/lank/eam
  https://gitee.com/LeeFJ/foxnic-samples

你可能感兴趣的:(Foxnic-SQL (7) —— DAO 特性 : 执行 SQL 语句)