Foxnic-SQL (2) —— SQL表达式(Expr)

Foxnic-SQL (2) —— SQL表达式(Expr)

常规意义的表达式

        表达式是运算符、常量和变量的组合。一个表达式可以由一个或多个操作数和零个或多个运算符组成以产生一个值。常量表达式、数学运算、关系表达式、逻辑表达式、位表达式等。

SQL表达式

        Foxnic-SQL 中所说的表达式,其实是SQL表达式。任何带有变量占位符的字符串就是一个SQL表达式,这些表达式最终可以拼装成一个完整的SQL语句。Foxnic-SQL 支持命名占位符(:NAME)和匿名占位符(?)。Foxnic-SQL 通过 Expr 类构建与解析SQL表达式,例如:

Expr expr1=new Expr("select ?+? as result from dual",8,9);
System.out.println("expr1 = "+expr1);
// 输出:expr1 = select 8 + 9 as result from dual

        这样一个查询语句就是一个表达式,其中的问号(?)就是变量占位符,他们没有名称只有顺序,所以是匿名占位符。

Map<String,Object> namedParam=new HashMap<>();
namedParam.put("P1",18);
namedParam.put("P2",19);
Expr expr2=new Expr("select :P1+:P2 as result from dual",namedParam);
System.out.println("expr2 = "+expr2);
// 输出:expr2 = select 18 + 19 as result from dual

        本示例将原来的问号(?)占位符换成了以冒号(:)开头的有名称的占位符P1、P2,这就是带有命名占位符的SQL表达式。当然表达式还可以更为简单,例如:

Expr expr1=new Expr("age>?",28);
System.out.println("expr1 = "+expr1);
// 输出:expr1 = age> 28

        也可以是:

Expr expr1=new Expr("age>? and height>?",28,182);
System.out.println("expr1 = "+expr1);
// 输出:expr1 = age> 28 and height> 182

        就技术角度而言,SQL表达式所关心的核心问题是变量占位符,至于内容是否符合SQL语法,它并不关心

SQL表达式的执行

        当一个SQL表达式完整且符合SQL语法时,它就可以被数据库执行。表达式通过DAO传递给JDBC执行时,这些占位符被解析并处理成绑定变量。当表达式被打印或输出时,占位符会被处理成符合指定数据库语法且变量已代入的SQL字符串。

/**
* 2、表达式通过DAO传递给JDBC执行时,这些占位符被解析并处理成绑定变量。当表达式被打印或输出时,占位符会被处理成符合指定数据库语法且变量已代入的SQL字符串。
* */
public static void  demo2() {

    Date date= DateUtil.parse("2022-12-05");
    Expr expr1=new Expr("select ?+? as result from my_table where create_date>?",8,9,date);
    System.out.println("传递给JDBC的语句(匿名占位符) = "+expr1.getListParameterSQL());
    // 输出:传递给JDBC的语句(匿名占位符) = select ? + ? as result from my_table where create_date> ?
    System.out.println("传递给JDBC的语句(命名占位符) = "+expr1.getNamedParameterSQL());
    // 输出:传递给JDBC的语句(命名占位符) = select :PARAM_1 + :PARAM_2 as result from my_table where create_date> :PARAM_3

    // 指定全局的SQL方言,默认MySQL,特定数据库时自动识别
    GlobalSettings.DEFAULT_SQL_DIALECT= SQLDialect.MySQL;
    System.out.println("开发人员友好的SQL(默认) = "+expr1.getSQL());
    // 输出:开发人员友好的SQL(默认) = select 8 + 9 as result from my_table where create_date> str_to_date('2022-12-05 00:00:00','%Y-%m-%d %H:%i:%s')
    System.out.println("开发人员友好的SQL(Oracle) = "+expr1.getSQL(SQLDialect.PLSQL));
    // 输出:开发人员友好的SQL(Oracle) = select 8 + 9 as result from my_table where create_date> to_date('2022-12-05 00:00:00','yyyy-mm-dd hh24:mi:ss')

    // 执行,此处请关注控制台输出日志
    Expr exprForQuery=new Expr("select * from sys_dict where create_time>?",date);
    RcdSet rs=DBInstance.DEFAULT.dao().query(exprForQuery);
    for (Rcd r : rs) {
        System.out.println(r.toJSONObject());
    }
}

        开发人员可以从表达式(Expr)获得比传统字符串拼接更加灵活,且绑定变量友好的SQL处理方式。也可以高效轻松地通过表达式避免数据库的SQL硬解析,从而获得更佳性能。

        同时,开发人员可以轻松地获得绑定变量已代入的SQL语句,这些语句只要复制就可以粘贴到数据库客户端工具执行,极大地方便了SQL语句的调试与问题排查。

表达式拼装

        Expr 可以通过 append 方法实现两个表达式的拼接,示例代码:

/**
* 3、表达式的拼装
* */
public static void  demo3() {

    // 示例-1
    Expr expr1=new Expr("age>?",28);
    Expr expr2=new Expr("and height>?",182);
    // 通过  append 方法将 expr2 拼接到 expr1
    expr1.append(expr2);
    System.out.println(expr1.getSQL());
    // 输出:age> 28 and height> 182

    // 示例-2
    Expr expr3=new Expr("age>?",28);
    expr3.append("and height>?",182);
    System.out.println(expr1.getSQL());
    // 输出:age> 28 and height> 182

}

总结

        Foxnic-SQL 的 Expr 类负责SQL语句、子语句、语句片段中的绑定变量的解析。其目的是时最终执行的SQL语句对JDBC友好、对开发人员友好。Expr 是 Foxnic-SQL 这个语句体系的基石,后续其它的语句对象都是由 Expr 继承或组合而来。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
一、SQL 结构化查询语言 包括DDL(数据定义语言)、DCL(数据控制语言)、 DQL(数据查询语言)、DML(数据操纵语言) 二、SQL的特点 SQL 语句不区分大小写 SQL 语句能输入一行或多行 关键字不能整行缩写或分离 子句通常被放置在分开的行上 缩进可提高可读性 在SQL 开发工具,SQL 语句能选择分号结束(;) .当你运行多个SQL 语句的时候,需要分号 在SQL*Plus中, 你要用一个分号结束每个SQL 语句.(;) 三、SQL*PLUS特征: 字符日期左对齐 数字右对对齐 列名默认大写 SQL PLUS 自己的命令不需以分号“;”结束 四、SQL查询时,数字和日期类型的数据可用算术运算符 + 加 - 减 * 乘 / 除 ( ) 用于改变运算符的优先级 五、空值 空值一般用NULL表示 一般表示未知的、不确定的值,也不是空格 一般运算符与其进行运算时,都会为空 空不与任何值相等 表示某个列为空用:IS NULL 不能使用COMM=NULL这种形式 某个列不为空:IS NOT NULL 不能使用COMM != NULL 这种形式 空值在作升序排列时,空值会放到最后。 相反作降序排列时,空值会放在最前。 空值作逻辑运算时: AND运算: F AND F =F F AND T =F F AND NULL =F T AND F =F T AND T =T T AND NULL IS NULL NULL AND F =F NULL AND T IS NULL NULL AND NULL IS NULL 就是说AND的优先级是:F ->NULL ->T OR运算: T OR T =T T OR F =T T OR NULL =T F OR T =T F OR F =F F OR NULL IS NULL NULL OR T =T NULL OR F IS NULL NULL OR NULL IS NULL OR运算优先级:T ->NULL ->F NOT运算: NOT T =F NOT F =T NOT NULL IS NULL 与空值相关的函数: NVL 函数 格式:NVL(表达式1,表达式2) 作用:测试表达式的值,如果表达式1为空,则返回表达式2的值;不为空,返回表达式1的值。 NVL2 函数 格式:NVL2(表达式1,表达式2,表达式3) 作用:测试表达式的值,表达式1不为空,返回表达式2的值,如果为空,则返回表达式3的值。 NULLIF 相等为空 格式:NULLIF (表达式1,表达式2) 作用:比较表达式1和表达式2的值,如果两个相等则返回为空,否则返回表达式1的值。 COALESCE 找非空 格式:COALESCE (表达式1,表达式2,表达式3,...,表达式n) 作用:返回第一个不为空的值,如果所有的都为空,则返回NULL。 六、SELECT语句的用法 SELECT *|{[DISTINCT] column|expression [alias],...} FROM table; 七、演示 */ --选择所有字段 SQL> SET LINESIZE 200 SQL> SELECT * FROM SCOTT.EMP; EMPNO ENAME JOB MGR HIREDATE SAL COMM DEPTNO ---------- ---------- --------- ---------- --------- ---------- ---------- ---------- 7369 SMITH CLERK 7902 17-DEC-80 800 20 7499 ALLEN SALESMAN 7698 20-FEB-81 1600 300 30 7521 WARD SALESMAN 7698 22-FEB-81 1250 500 30 7566 JONES MANAGER 7839 02-APR-81 2975 20 7654 MARTIN SALESMAN 7698 28-SEP-81 1250 1400 30 --选择部分字段 SQL> SELECT EMPNO,ENAME,SAL FROM SCOTT.EMP; EMPNO ENAME SAL ---------- ---------- ---------- 7369 SMITH 800 7499 ALLEN 1600 7521 WARD 1250 --算术加减运算 SQL> SELECT EMPNO,ENAME,SAL + 300 FROM SCOTT.EMP; EMPNO ENAME SAL+300 ---------- ---------- ---------- 7369 SMITH 1100 7499 ALLEN 1900 7521 WARD 1550 --优先级 SQL> SELECT EMPNO,ENAME,12 * (SAL + 300) FROM SCOTT.EMP; EMPNO ENAME 12*(SAL+300) ---------- ---------- ------------ 7369 SMITH 13200 7499 ALLEN 22800 7521 WARD 18600 SQL> SELECT EMPNO,ENAME,12 * SAL + 300 FROM SCOTT.EMP; EMPNO ENAME 12*SAL+300 ---------- ---------- ---------- 7369 SMITH 9900 7499 ALLEN 19500 7521 WARD 15300 --NULL,记录中COMM存在为NULL的情况 SQL> SELECT EMPNO,ENAME,SAL,COMM FROM SCOTT.EMP; EMPNO ENAME SAL COMM ---------- ---------- ---------- ---------- 7369 SMITH 800 7499 ALLEN 1600 300 7521 WARD 1250 500 7566 JONES 2975 --与NULL运算,结果为NULL SQL> SELECT EMPNO,ENAME,SAL,COMM + 300 FROM SCOTT.EMP; EMPNO ENAME SAL COMM+300 ---------- ---------- ---------- ---------- 7369 SMITH 800 7499 ALLEN 1600 600 7521 WARD 1250 800 7566 JONES 2975 --将COMM不为NULL的记录的COMM乘以 SQL> SELECT EMPNO,ENAME,SAL,COMM * 12 FROM SCOTT.EMP WHERE COMM IS NOT NULL EMPNO ENAME SAL COMM*12 ---------- ---------- ---------- ---------- 7499 ALLEN 1600 3600 7521 WARD 1250 6000 7654 MARTIN 1250 16800 7844 TURNER 1500 0 --字段别名,字段后用AS 别名,AS可以省略 SQL> SELECT EMPNO,ENAME AS EmpName,SAL Salary FROM SCOTT.EMP; EMPNO EMPNAME SALARY ---------- ---------- ---------- 7369 SMITH 800 7499 ALLEN 1600 7521 WARD 1250 7566 JONES 2975 --DISTINCT,过滤重复行 SQL> SELECT DISTINCT EMPNO,ENAME FROM SCOTT.EMP; --连接操作符,通过二个垂直的条描述(||),注意,日期和文字数值一定嵌入在单引号里面 SQL> SELECT EMPNO,ENAME || ' IS A ' ||JOB AS POSITION FROM SCOTT.EMP; EMPNO POSITION ---------- ------------------------- 7369 SMITH IS A CLERK 7499 ALLEN IS A SALESMAN 7521 WARD IS A SALESMAN --DESC table_name,显示表结构信息 SQL> DESC SCOTT.EMP Name Null? Type ----------------------------------------- -------- ---------------------------- EMPNO NOT NULL NUMBER(4) ENAME VARCHAR2(10) JOB VARCHAR2(9) MGR NUMBER(4) HIREDATE DATE SAL NUMBER(7,2) COMM NUMBER(7,2) DEPTNO NUMBER(2) --NVL的用法 SQL> SELECT EMPNO,ENAME,NVL(TO_CHAR(COMM),'Not Applicable') FROM SCOTT.EMP; EMPNO ENAME NVL(TO_CHAR(COMM),'NOTAPPLICABLE') ---------- ---------- ---------------------------------------- 7369 SMITH Not Applicable 7499 ALLEN 300 7521 WARD 500 7566 JONES Not Applicable --NVL2的用法 SQL> SELECT empno,ename,sal,NVL2(TO_CHAR(comm),12 * (sal + comm),sal) AS Income FROM scott.emp; EMPNO ENAME SAL INCOME ---------- ---------- ---------- ---------- 7369 SMITH 800 800 7499 ALLEN 1600 22800 7521 WARD 1250 21000 7566 JONES 2975 2975 --NULLIF的用法 --等价于CASE WHEN expr1 = expr2 THEN NULL ELSE expr1 END SQL> SELECT e.last_name, NULLIF(e.job_id, j.job_id) "Old Job ID" 2 FROM hr.employees e, hr.job_history j 3 WHERE e.employee_id = j.employee_id 4 ORDER BY last_name, "Old Job ID"; LAST_NAME Old Job ID ------------------------- ---------- De Haan AD_VP Hartstein MK_MAN Kaufling ST_MAN Kochhar AD_VP Kochhar AD_VP Raphaely PU_MAN Taylor SA_REP Taylor Whalen AD_ASST Whalen --下面是使用CASE WHEN的等价用法 SQL> SELECT e.last_name, CASE WHEN e.job_id = j.job_id THEN NULL ELSE e.job_id END AS "Old Job ID" 2 FROM hr.employees e, hr.job_history j 3 WHERE e.employee_id = j.employee_id 4 ORDER BY last_name, "Old Job ID"; LAST_NAME Old Job ID ------------------------- ---------- De Haan AD_VP Hartstein MK_MAN Kaufling ST_MAN Kochhar AD_VP Kochhar AD_VP Raphaely PU_MAN Taylor SA_REP Taylor Whalen AD_ASST Whalen --COALESCE的用法 --当COALESCE(exp1,exp2)包含两个表达式时,等价于CASE WHEN exp1 IS NOT NULL THEN exp1 ELSE exp2 END --COALESCE (expr1, expr2, ..., exprn), for n>=3 --当n >= 3时,等价于 --CASE WHEN expr1 IS NOT NULL THEN expr1 -- ELSE COALESCE (expr2, ..., exprn) END SQL> SELECT product_id, list_price, min_price, 2 COALESCE(0.9*list_price, min_price, 5) "Sale" 3 FROM oe.product_information 4 WHERE supplier_id = 102050 5 ORDER BY product_id, list_price, min_price, "Sale" PRODUCT_ID LIST_PRICE MIN_PRICE Sale ---------- ---------- ---------- ---------- 1769 48 43.2 1770 73 73 2378 305 247 274.5 2382 850 731 765 3355 5 八、更多

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值