package com.dx.efuwu.core

 import org.apache.commons.lang.StringUtils
import java.sql.PreparedStatement /**
* sql 模板处理
* @author sunzq
* 2017/06/02
*/ /**
* 查询的一个条件句
*/
class QueryBranch(val content: String, val key: String, val type: String) { override fun toString(): String {
return "{content:$content, key:$key, type: $type}"
} fun build(input: String?): String {
return if (StringUtils.isNotBlank(input)) content.replace("""(?:#|##|@|@@)\{.*}""".toRegex(), replacement = "?")
else if (type.length > ) " 1=0 "
else " 1=1 "
} fun doSetParameter(input: String?, index: Int, pstmt: PreparedStatement): Boolean {
input?.takeIf {
StringUtils.isNotBlank(it)
}?.run {
when (type) {
"#", "##" -> pstmt.setString(index, input)
"@", "@@" -> pstmt.setLong(index, input.toLong())
}
return true
}
return false
} } /**
* sql 查询模板处理器,并不提供 sql 是否正确的解析.
* @ 原样填写,不自动填充单引号, 为空则 true @@ 为空则 false
* # 自动包一下引号, 为空则 true ## 为空则 false
*/
class SQLTemplateExecutor(sql: String) { /**
* 查询条件句
*/
val queryBranches = ArrayList<QueryBranch>() /**
* 切割后的查询 sql
*/
val splitQueryStrings = ArrayList<String>() init {
val _sql = sql.replace("""\n""".toRegex(), " ")
val regex = """\b\S+(?:\s|=|>|<|in|not|like|IN|NOT|LIKE)+(#|##|@|@@)\{([^}]+)}""".toRegex()
splitQueryStrings.addAll(_sql.split(regex))
regex.findAll(_sql).iterator().forEach { matchResult ->
matchResult.groupValues.apply {
queryBranches.add(QueryBranch(content = get(), key = get().trim(), type = get()))
}
}
} fun doQuery(dict: Map<String, String?>): ArrayList<Map<String, Any>> {
val sqlBuilder = StringBuilder()
val queryBranchIterator = queryBranches.iterator()
splitQueryStrings.forEach { queryString ->
sqlBuilder.append(queryString)
if (queryBranchIterator.hasNext()) {
queryBranchIterator.next().apply {
sqlBuilder.append(build(dict[key]))
}
}
} val queryResults = ArrayList<Map<String, Any>>()
SQL_SERVER_DATA_SOURCE.connection.apply {
prepareStatement(sqlBuilder.toString()).apply {
// 设置查询参数
var paramIndex =
queryBranches.forEach { queryBranch ->
val param = dict[queryBranch.key]
if (queryBranch.doSetParameter(param, paramIndex, this)) {
paramIndex++
}
}
// 执行查询
executeQuery().apply {
while (next()) {
val result = HashMap<String, Any>()
metaData.apply {
(..columnCount).forEach { i ->
result[getColumnLabel(i)] =
when (getColumnTypeName(i)) {
"int" -> getInt(i)
else -> getString(i)
}
}
}
queryResults.add(result)
}
close()
}
close()
}
close()
}
return queryResults
}
} fun main(args: Array<String>) {
val sql = """select * from test"""
println(SQLTemplateExecutor(sql).doQuery(hashMapOf(
"userName1" to "大%",
"age" to "",
"s2" to "hello,world",
"userId" to ""
)))
}

kotlin 写的一个简单 sql 查询解析器的更多相关文章

  1. 简单sql字段解析器实现参考

    用例:有一段sql语句,我们需要从中截取出所有字段部分,以便进行后续的类型推断,请给出此解析方法. 想来很简单吧,因为 sql 中的字段列表,使用方式有限,比如 a as b, a, a b... 1 ...

  2. 自己动手实现一个简单的JSON解析器

    1. 背景 JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式.相对于另一种数据交换格式 XML,JSON 有着诸多优点.比如易读性更好,占用空间更少等.在 ...

  3. 一个简单的json解析器

    实现一个简单地json解析器. 两部分组成,词法分析.语法分析 词法分析 package com.mahuan.json; import java.util.LinkedList; import ja ...

  4. 用c#自己实现一个简单的JSON解析器

    一.JSON格式介绍 JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式.相对于另一种数据交换格式 XML,JSON 有着很多优点.例如易读性更好,占用空间更 ...

  5. 使用ssm(spring+springMVC+mybatis)创建一个简单的查询实例(一)

    梳理下使用spring+springMVC+mybatis 整合后的一个简单实例:输入用户的 ID,之后显示用户的信息(此次由于篇幅问题,会分几次进行说明,此次是工程的创建,逆向生成文件以及这个简单查 ...

  6. 使用ssm(spring+springMVC+mybatis)创建一个简单的查询实例(三)(错误整理篇)

    使用ssm(spring+springMVC+mybatis)创建一个简单的查询实例(一) 使用ssm(spring+springMVC+mybatis)创建一个简单的查询实例(二) 以上两篇已经把流 ...

  7. 使用ssm(spring+springMVC+mybatis)创建一个简单的查询实例(二)(代码篇)

    这篇是上一篇的延续: 用ssm(spring+springMVC+mybatis)创建一个简单的查询实例(一) 源代码在github上可以下载,地址:https://github.com/guoxia ...

  8. c# 怎样能写个sql的解析器

    c# 怎样能写个sql的解析器 本示例主要是讲明sql解析的原理,真实的源代码下查看 sql解析器源代码 详细示例DEMO 请查看demo代码 前言 阅读本文需要有一定正则表达式基础 正则表达式基础教 ...

  9. 只是一个用EF写的一个简单的分页方法而已

    只是一个用EF写的一个简单的分页方法而已 慢慢的写吧.比如,第一步,先把所有数据查询出来吧. //第一步. public IQueryable<UserInfo> LoadPagesFor ...

随机推荐

  1. 【bzoj2705】[SDOI2012]Longge的问题 欧拉函数

    题目描述 Longge的数学成绩非常好,并且他非常乐于挑战高难度的数学问题.现在问题来了:给定一个整数N,你需要求出∑gcd(i, N)(1<=i <=N). 输入 一个整数,为N. 输出 ...

  2. 浅谈javascript的原型及原型链

    浅谈javascript的原型及原型链 这里,我们列出原型的几个概念,如下: prototype属性 [[prototype]] __proto__ prototype属性 只要创建了一个函数,就会为 ...

  3. [CF1031E]Triple Flips

    题目大意:给你一个长度为$n$的$01$串,一次操作定义为:选取$3$个等距的元素,使其$0$变$1$,$1$变$0$,要求在$\Big\lfloor \dfrac n 3\Big\rfloor+12 ...

  4. 洛谷P4591 [TJOI2018]碱基序列 【KMP + dp】

    题目链接 洛谷P4591 题解 设\(f[i][j]\)表示前\(i\)个串匹配到位置\(j\)的方案数,匹配一下第\(i\)个串进行转移即可 本来写了\(hash\),发现没过,又写了一个\(KMP ...

  5. BZOJ3132 上帝造题的七分钟 【二维树状数组】

    题目 "第一分钟,X说,要有矩阵,于是便有了一个里面写满了0的n×m矩阵. 第二分钟,L说,要能修改,于是便有了将左上角为(a,b),右下角为(c,d)的一个矩形区域内的全部数字加上一个值的 ...

  6. Link Cat Tree (连喵树) 学习笔记

    Link Cat Tree 一.感性定义 所谓连喵树,即一种对森林支持修改,查询,连边,删边等操作的数据结构(姑且算她是吧).她用一颗颗互相连接的辅助树维护原森林的信息,辅助树相互连接的边叫虚边,辅助 ...

  7. 关于final局部变量引用的研究

    嵌套类(内部类)方法安全引用外部方法局部变量的原理 嵌套类方法引用外部局部变量,必需将声明为final,否则将出现 Cannot refer to a non-final variable * ins ...

  8. json 串转成 java 对象再拼接成前台 html 元素

    获取商品参数 json 串,转成 java 对象,再拼接成前台 html 的Service方法 @Override public String getItemParam(Long itemId) { ...

  9. MySQL 数据库性能优化之缓存参数优化

    在平时被问及最多的问题就是关于 MySQL 数据库性能优化方面的问题,所以最近打算写一个MySQL数据库性能优化方面的系列文章,希望对初中级 MySQL DBA 以及其他对 MySQL 性能优化感兴趣 ...

  10. 把java的class文件打成jar包的步骤

    现在我的文件夹的目录在: C:\Users\linsenq\Desktop\cglibjar 我要把位于这个目录下的所有文件夹以及这个文件夹下的.class文件打成jar包 第一步:用win+R 打开 ...