写一个简单的SQL生成工具
知识点:
MyBatis 语法概览
MyBatis 是一个强大的数据持久化框架,它提供了一种半自动化的 ORM 实现方式。通过 MyBatis,开发者可以通过简单的 XML 或注解来配置和映射原生信息,将接口和 Java 的 POJOs(Plain Old Java Objects,普通的 Java 对象)映射成数据库中的记录。
基本查询语句
在 MyBatis 中,基本的查询语句使用 <select> 标签定义。例如,以下是一个简单的查询语句:
这个语句被称为 selectPerson,接受一个 int 类型的参数,并返回一个 HashMap 类型的对象。
参数和结果映射
MyBatis 允许使用 #{} 和 ${} 两种方式获取参数值。其中 #{} 是预处理语句参数(如 JDBC 的 ?),而 ${} 是直接进行字符串替换。使用 #{} 可以有效防止 SQL 注入。
动态 SQL
MyBatis 支持动态 SQL,这意味着 SQL 语句可以根据传入的参数动态变化。例如,使用 <if> 标签可以根据条件包含不同的 SQL 片段:
在这个例子中,如果 title 参数不为 null,则会包含一个按标题进行模糊查询的条件。
高级结果映射
MyBatis 的 <resultMap> 元素提供了高级结果映射功能。它允许开发者定义如何从数据库结果集中加载对象,这是 MyBatis 最强大的特性之一。
缓存
MyBatis 提供了强大的缓存功能,可以通过简单地在映射文件中添加 <cache> 标签来启用二级缓存。
总结
MyBatis 通过简化 SQL 操作和提供动态 SQL 功能,使得数据库操作变得更加灵活和强大。它的高级映射功能和缓存机制也为开发者提供了更多的便利。
代码:
import com.alibaba.fastjson2.util.DateUtils;
import org.apache.commons.lang3.ObjectUtils; import java.util.Date; public interface BaseService { default Query query() {
return new Query();
} default Update update() {
return new Update();
} default Insert insert() {
return new Insert();
} static String value(Object value) {
if (ObjectUtils.isEmpty(value)) {
return " NULL ";
}
if (value instanceof Date) {
return "'" + DateUtils.format((Date) value, "yyyy-MM-dd HH:mm:ss") + "'";
}
if (value instanceof String) {
return "'" + value + "'";
}
if (value instanceof Number) {
return value.toString();
}
throw new RuntimeException("无法解析的字段 :" + value);
} class Insert {
StringBuilder sqlBuilder = new StringBuilder();
StringBuilder inertFieldBuilder = new StringBuilder();
StringBuilder inertValueBuilder = new StringBuilder(); public Insert set(String column, Object value) {
if (inertFieldBuilder.length() > 0) inertFieldBuilder.append(" , ");
inertFieldBuilder.append(column);
if (inertValueBuilder.length() > 0) inertValueBuilder.append(" , ");
inertValueBuilder.append(" ");
inertValueBuilder.append(value(value));
inertValueBuilder.append(" ");
return this;
} public String toString() {
if (inertFieldBuilder.length() > 0 && inertValueBuilder.length() > 0) {
sqlBuilder.append(" ( ");
sqlBuilder.append(inertFieldBuilder.toString());
sqlBuilder.append(" ) ");
sqlBuilder.append(" VALUES( ");
sqlBuilder.append(inertValueBuilder.toString());
sqlBuilder.append(" ) ");
}
return sqlBuilder.toString();
}
} class Update {
StringBuilder sqlBuilder = new StringBuilder();
StringBuilder updateBuilder = new StringBuilder(); public Update set(String column, Object value) {
if (updateBuilder.length() > 0) updateBuilder.append(" , ");
updateBuilder.append(" ");
updateBuilder.append(column);
updateBuilder.append(" = ");
updateBuilder.append(value(value));
updateBuilder.append(" ");
return this;
} public String toString() {
if (updateBuilder.length() > 0) {
sqlBuilder.append(updateBuilder.toString());
}
return sqlBuilder.toString();
}
} class Query {
StringBuilder whereBuilder = new StringBuilder();
StringBuilder sqlBuilder = new StringBuilder(); public Query eq(String column, Object value) {
if (ObjectUtils.isEmpty(value)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append(" ");
whereBuilder.append(column);
whereBuilder.append(" = ");
whereBuilder.append(value(value));
whereBuilder.append(" ");
return this;
} public Query or(String column, Object value) {
if (ObjectUtils.isEmpty(value)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" OR ");
whereBuilder.append(" ");
whereBuilder.append(column);
whereBuilder.append(" = ");
whereBuilder.append(value(value));
whereBuilder.append(" ");
return this;
} public Query like(String column, String value) {
if (ObjectUtils.isEmpty(value)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append(" ");
whereBuilder.append(column);
whereBuilder.append(" LIKE '%");
whereBuilder.append(value);
whereBuilder.append("%' ");
return this;
} public Query in(String column, Object... values) {
if (ObjectUtils.isEmpty(values)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append(" ");
whereBuilder.append(column);
whereBuilder.append(" IN( ");
int i;
i = 0;
for (Object value : values) {
if (i > 0) whereBuilder.append(" , ");
i++;
whereBuilder.append(value(value));
}
whereBuilder.append(" )");
return this;
} public Query noIn(String column, Object... values) {
if (ObjectUtils.isEmpty(values)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append(" ");
whereBuilder.append(column);
whereBuilder.append(" NOT IN( ");
int i;
i = 0;
for (Object value : values) {
if (i > 0) whereBuilder.append(" , ");
i++;
whereBuilder.append(value(value));
}
whereBuilder.append(" )");
return this;
} public Query isNull(String column) {
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append(" ( LENGTH( ");
whereBuilder.append(column);
whereBuilder.append(" ) <=0 OR IS NULL ) ");
return this;
} public Query notNull(String column) {
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append(" ( LENGTH( ");
whereBuilder.append(column);
whereBuilder.append(" ) >0 OR IS NOT NULL ) ");
return this;
} public Query sql(Object sql) {
if (ObjectUtils.isEmpty(sql)) return this;
whereBuilder.append(sql.toString());
return this;
} public Query and(String whereSql) {
if (ObjectUtils.isEmpty(whereSql)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" AND ");
whereBuilder.append("( ");
whereBuilder.append(whereSql);
whereBuilder.append(" )");
return this;
} public Query or(String whereSql) {
if (ObjectUtils.isEmpty(whereSql)) return this;
if (whereBuilder.length() > 0) whereBuilder.append(" OR ");
whereBuilder.append("( ");
whereBuilder.append(whereSql);
whereBuilder.append(" )");
return this;
} public String toString() {
if (whereBuilder.length() > 0) {
sqlBuilder.append(whereBuilder.toString());
}
return sqlBuilder.toString();
} }
}
作为参数通过mapper传入XML作为【条件|行为】使用:
// name = "张三"
query().eq("name","张三").toString();
// name = "张三"
update().set("name","张三").toString();
// (name) values("张三")
insert().set("name","张三").toString();
写一个简单的SQL生成工具的更多相关文章
- DAY5 php + mysql 写一个简单的sql注入平台
php mysql 在浏览器输入用户名,去数据库查询.查到则显示在浏览器,查不到则显示空. sql 里面三个字段 id username password create table t1 (id in ...
- 原生js 基于canvas写一个简单的前端 截图工具
先看效果 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <titl ...
- 分享:计算机图形学期末作业!!利用WebGL的第三方库three.js写一个简单的网页版“我的世界小游戏”
这几天一直在忙着期末考试,所以一直没有更新我的博客,今天刚把我的期末作业完成了,心情澎湃,所以晚上不管怎么样,我也要写一篇博客纪念一下我上课都没有听,还是通过强大的度娘完成了我的作业的经历.(当然作业 ...
- SQL点滴10—使用with语句来写一个稍微复杂sql语句,附加和子查询的性能对比
原文:SQL点滴10-使用with语句来写一个稍微复杂sql语句,附加和子查询的性能对比 今天偶尔看到sql中也有with关键字,好歹也写了几年的sql语句,居然第一次接触,无知啊.看了一位博主的文章 ...
- 动手写一个简单的Web框架(Werkzeug路由问题)
动手写一个简单的Web框架(Werkzeug路由问题) 继承上一篇博客,实现了HelloWorld,但是这并不是一个Web框架,只是自己手写的一个程序,别人是无法通过自己定义路由和返回文本,来使用的, ...
- 手把手教你从零写一个简单的 VUE
本系列是一个教程,下面贴下目录~1.手把手教你从零写一个简单的 VUE2.手把手教你从零写一个简单的 VUE--模板篇 今天给大家带来的是实现一个简单的类似 VUE 一样的前端框架,VUE 框架现在应 ...
- 手把手教你从零写一个简单的 VUE--模板篇
教程目录1.手把手教你从零写一个简单的 VUE2.手把手教你从零写一个简单的 VUE--模板篇 Hello,我又回来了,上一次的文章教会了大家如何书写一个简单 VUE,里面实现了VUE 的数据驱动视图 ...
- [Winform]一个简单的账户管理工具
最近一直觉得注册的账户越来越多,帐号密码神马的容易弄混.自己就折腾了一个简单的账户管理工具,其实实现也挺简单,将每个账户的密码及相关密码提示信息,经aes算法加密之后保存到数据库,当前登录用户可以查询 ...
- linux设备驱动第三篇:如何写一个简单的字符设备驱动?
在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存 ...
- 用node.js从零开始去写一个简单的爬虫
如果你不会Python语言,正好又是一个node.js小白,看完这篇文章之后,一定会觉得受益匪浅,感受到自己又新get到了一门技能,如何用node.js从零开始去写一个简单的爬虫,十分钟时间就能搞定, ...
随机推荐
- bug记录:Vue.use 加载 TabPane ,浏览器卡死
问题描述 Vue.use 加载 TabPane ,浏览器卡死 原因分析 参考资料:https://blog.csdn.net/ye987987/article/details/103780297 经过 ...
- Linux 添加开机自启动
rc.local 方式 一.& 在 Linux 命令后加上 & 可以在后台运行 二.nohup 对 SIGHUP 信号免疫,对 SIGINT 信号不免疫,可用 shopt | gre ...
- springboot 2.x 集成quartz持久化到一个单独的dataSource时遇到的坑
由于希望可以在集群环境中运行定时job,但考虑到多个job实例有可能带来job重复执行的问题,新项目的job打算从原生的spring task实现改成quartz job实现,并采用jdbc的存储方式 ...
- 展锐Android平台增加gadget 虚拟usb串口
方案一:需要修改展锐现有Windows端驱动,增加一组MI接口.由于无法推动展锐修改Windows驱动,该方案不推荐. SL8541E/device/sprd/sharkle/common/rootd ...
- 《计算机体系结构与SoC设计》(二)
1. 多指令流单数据流 多指令流单数据流(Multiple Instruction Stream, Single Data Stream,简称 MISD)是一种处理器设计概念,它允许处理器在单个时钟周 ...
- 云辅助隐私集合求交(Server-Aided PSI)协议介绍:学习
原文来自:云辅助隐私集合求交(Server-Aided PSI)协议介绍,下面学习一波,并记录一些笔记. 背景 总结: 1.PSI-CA和PSI相比,前者在乎的是交集的大小,后者在乎的是交集本身.另外 ...
- LinkedList可以同时作为堆栈和队列使用
Java里的LinkedList可以同时作为堆栈和队列使用,因此在使用的时候总是会弄混他们的方法,此文就简单总结一下作为不同数据结构使用时的用法. 作为队列 方法 声明 任意两种方法: 一是直接声 ...
- 从理念到实践,解构HBlock降本增效黑科技!
"新存储 更轻量" 天翼云存储资源盘活系统HBlock 2024年线上技术分享会来啦! 在2023年HBlock分享会之后 不少小伙伴积极参与了1PB免费尝鲜活动 好评如潮的同时 ...
- Project Euler 307 题解
主要是规避误差.即求 \[\frac{k^n}{n^k} \] 微分一下得到递推式.然后根据斯特林近似(byd 这里还需要 \(1\) 后的第一项..) ...
- ClickHouse常用操作
一.客户端连接1.1 客户端连接ck./clickhouse-client -h 127.0.0.1 --port 9900 -u default --password 123456 -m 1.2 h ...