生成sql:select

上一篇讲了怎样生成一个sqlwhere的一部分,之后我们要做事情就简单很多了,就只要像最开始一样的生成各种sql语句就好了,之后只要再加上我们需要的条件,一个完整的sql就顺利的做好了。

现在我们开始写生成查询语句的sql。一个查询语句大致上是这样的:

  1. SELECT name, id, create_date, age, mark, status FROM user

这里可以看出来,一个基础的查询语句基本上就是一个 SELECT 后面加上需要查询的字段,跟上 FROM 和要查询的表名称就好了。 最多后面可能需要加上 ORDER BY/GROUP BY/LIMIT ....之类的就好了,因为比较简单,这里就不写了。(太复杂的就直接写sql就好了,我自己不需要这种操作)

思路

  1. 从之前拿到的映射关系中拿到属性和字段名的映射,然后拼接sql。
  2. 执行sql,并取出结果。
  3. 实例化class,使用反射给class的属性赋值。

这几步都还是比较好做的,第一步很简单,仿照着之前写的就可以了。因为这里在执行sql的时候,我使用的是JdbcTemplate,这里有一个不大不小的坑,下面我说一下。

一个不大不小的坑

这个坑是我在使用我写好的这个项目给公司做报表的时候碰到的。原因是这样,因为数据库中有些字段是datetime类型的,这个字段有时候在表中的值是:0000-00-00 00:00:00,(我也不知道这个值是怎么进去的,但是就是存在/(ㄒoㄒ)/~~)但是这个值是无法转换成为java中的Date类型。所以这里会报错。

我在这里写了一个继承SpringJdbc中的ColumnMapRowMapper的类,是这样的:

  1. import org.springframework.jdbc.core.ColumnMapRowMapper;
  2. import java.sql.ResultSet;
  3. import java.sql.SQLException;
  4. /**
  5. * 捕获取值的错误
  6. *
  7. * @author hjx
  8. */
  9. public class PlusColumnMapRowMapper extends ColumnMapRowMapper {
  10. /**
  11. * 数据库类型为时间时, 如果值为 0000-00-00 00:00:00
  12. * 会报错,所以重写此方法,返回null
  13. *
  14. * @param rs
  15. * @param index
  16. * @return
  17. * @throws SQLException
  18. */
  19. @Override
  20. protected Object getColumnValue(ResultSet rs, int index) throws SQLException {
  21. Object columnValue = null;
  22. try {
  23. columnValue = super.getColumnValue(rs, index);
  24. } catch (SQLException e) {
  25. e.printStackTrace();
  26. }
  27. return columnValue;
  28. }
  29. }

这个类具体在哪里使用,会在下面说明。

实现

现在说一下怎么实现上面的思路,首先因为第一步比较简单,就不写了。我直接从第二步开始。

  1. 执行sql,并取出结果。

    这里我用的是JdbcTemplate的方法,这给我们提供了一个方法:

    1. <T> List<T> query(String sql, Object[] args, RowMapper<T> rowMapper)

    这里前两个参数比较好理解,一个是sql,一个是sql中的参数。第三个是需要传一个接口RowMapper,这个接口具体是干啥的上网一查就知道了~~~

    这里面有一个方法:

    1. T mapRow(ResultSet rs, int rowNum) throws SQLException

    第一个参数是查询的结果,第二个是指现在在第几行结果,返回值是你要返回什么对象。这里我们需要重写这个方法,把查询出的结果转换成为我们需要的对象。我们可以这么写:

    1. /**
    2. * 把数据库查询的结果与对象进行转换
    3. *
    4. * @param resultSet
    5. * @param rowNum
    6. * @return
    7. * @throws SQLException
    8. */
    9. @Override
    10. public T mapRow(ResultSet resultSet, int rowNum) throws SQLException {
    11. Map<String, Object> resultMap = columnMapRowMapper.mapRow(resultSet, rowNum);
    12. 。。。。

    这个方法中的columnMapRowMapper 就是上面我们写的PlusColumnMapRowMapper,它的作用就是将查询结果第 rowNum 拿出来,并且将结果转换过成为一个 Map<String, Object>。其中:

    key :是表字段名称。

    Object :该字段的值。

    上面写的PlusColumnMapRowMapper主要作用就是在获取值的时候如果发生异常,返回一个null

    在这一步里我们已经拿到了执行sql的结果,现在我们要将结果转换过为我们需要的class。

  2. 将结果转换为class

    在上一步我们拿到了存放结果Map,现在只需要将map遍历一下,然后实例化java对象,根据字段和属性的映射关系使用反射将属性一个个的set进去就好了。现在贴上上一步的完整代码:

    1. public T mapRow(ResultSet resultSet, int rowNum) throws SQLException {
    2. Map<String, Object> resultMap = columnMapRowMapper.mapRow(resultSet, rowNum);
    3. T instance = getInstance(tableClass);
    4. for (Map.Entry<String, Object> entry : resultMap.entrySet()) {
    5. //数据库字段名
    6. String key = entry.getKey();
    7. if (!columnFieldMapper.containsKey(key)) {
    8. continue;
    9. }
    10. Field declaredField = columnFieldMapper.get(key);
    11. if (declaredField == null) {
    12. continue;
    13. }
    14. //数据库字段值
    15. Object value = entry.getValue();
    16. setFieldValue(instance, declaredField, value);
    17. }
    18. return instance;
    19. }

    其中 columnFieldMapper 是一个Map<String, Field>key是表的字段个名称。value是对应的class的属性。

    下面是 setFieldValue的具体代码:

    1. boolean setFieldValue(T t, Field field, Object value) {
    2. field.setAccessible(true);
    3. try {
    4. if (value != null) {
    5. field.set(t, value);
    6. return true;
    7. }
    8. } catch (IllegalAccessException e) {
    9. e.printStackTrace();
    10. }
    11. return false;
    12. }

    这样,就可以将查询出的结果根据映射关系转换成为我们需要的class了。

其他的

如果查询需要添加条件的话,可以使用之前讲的 生成条件的工具将条件的sql拼接在这里的sql后面,相应的,where里的参数也要按照顺序添加进数组就好了。

相同的,如果要添加 ORDER BY/GROUP BY/LIMIT这些东西的话也是一样的操作。主要还是要看自己的代码是怎么设计的了。我自己用的只写了ORDER BYLIMIT 。可以在我的github上找到。地址在这里:https://github.com/hjx601496320/JdbcPlus

生成sql:delete

思路

诶呀, 这个太简单了。不写了哦~~~

参照我之前写的,分析一下,想一想思路,然后每一步要怎么做,一点一点的就写好了。

~~~

实现

你自己写咯~~~。

其他的

这一篇写的真快~~

生成sql:update

最后一部分了,马上就写完了。写东西真的好累啊~~~

思路

更新的语句也比较好做,sql后面的条件因为在之前已经写了where这一篇,所以这里就只写sqlwhere左边的一部分。现在还是先分析一下 **update **语句:

  1. UPDATE user SET name = ? , id = ? , create_date = ? , age = ? , status = ? WHERE id = ?

可以看到的,大体上就是 UPDATE 表名称 SET 字段名称 = ? 这个样子的。(因为现在不写WHERE右边的

所以具体的思路就是:

  1. 根据映射关系拼装sql。

    这里可能有一个可以选择的地方,就是如果某一个属性的值是null,这时要不要把这个属性更新为null

  2. 拿到要更新的值。

  3. 执行sql。

实现

  1. 从映射中拿到所有的属性。

    这一步的代码就不放了~~~,和前面写的没有什么区别。

  2. 拿到要更新的属性名称,和值。

    这里我们需要三个参数:

    1:用来标示更新的时候是否需要忽略值是null的属性。 boolean ignoreNull

    2:用来保存需要更新的字段的有序集合。 List updataColumn

    3:保存需要更新的字段的值的有序集合。 List values

    代码是这样的:

    1. List<String> columnNames = new ArrayList<>(entityTableRowMapper.getColumnNames());
    2. Map<String, Field> columnFieldMapper = entityTableRowMapper.getColumnFieldMapper();
    3. List<Object> values = new ArrayList<>();
    4. for (int i = 0; i < columnNames.size(); i++) {
    5. String columnName = columnNames.get(i);
    6. if (!sqlColumns.contains(columnName)) {
    7. continue;
    8. }
    9. Field field = columnFieldMapper.get(columnName);
    10. Object value = EntityUtils.getValue(entity, field);
    11. //如果class中的值是null,并且设置忽略null,跳过
    12. if (ignoreNull && value == null) {
    13. continue;
    14. }
    15. updataColumn.add(columnName);
    16. values.add(value);
    17. }
  3. 根据拿到的数据拼装sql

    拿到上面需要的数据后,我们还需要拿到表的名称,这一步直接从映射关系中取就好了。下面的是拼装sql的代码:

    1. StringBuilder sql = new StringBuilder();
    2. sql.append("UPDATE ").append(getTableName()).append(StringUtils.SPACE);
    3. sql.append("SET ");
    4. for (int i = 0; i < updataColumn.size(); i++) {
    5. String column = updataColumn.get(i);
    6. if (i == 0) {
    7. sql.append(StringUtils.append(column, " = ? "));
    8. } else {
    9. sql.append(StringUtils.append(", ", column, " = ? "));
    10. }
    11. }

    这样就好了,大致上是这样的:

    1. UPDATE user SET name = ? , id = ? , create_date = ? , age = ? , status = ?

    条件的话,用之前写的where生成就好了,where中的值加在集合values的后面就好了。

  4. 执行sql。

    太简单了,就不写了~

最后

终于写完了。

还是说一下,因为代码已经在github上了,所以没有把全部的代码写在上面,主要还是以说明思路为主。另外刚开始写博客,有些可能表达的不是很明白。吃了没文化的亏啊~~~

这个项目还有很多可以但是还没有实现的功能,比如一些比较复杂的查询,执行函数之类的。我并没去写它。一是不需要,因为这个东西平时主要是做导出报表的时候用的,二是我自己写项目的话压根就不会用到这些东西,能用java写的我都用java写了。数据库嘛,对我来说就存个数据就好了,数据处理上的事情还是交给java来做好一点。

完了

手把手教你写一个java的orm(完)的更多相关文章

  1. 手把手教你写一个java的orm(一)

    写之前的说明 其实吧. 这个东西已经写好了,地址在:https://github.com/hjx601496320/JdbcPlus 这系列文章算是我写的过程的总结吧.(恩系列,说明我可能会写好久,╮ ...

  2. 手把手教你写一个java的orm(五)

    生成sql:where 上一篇里我们实现了生成insert的sql,下面要开始实现update,delete,select的sql语句了.但是这些语句有一个比较麻烦的地方是:它们一般后面都会有wher ...

  3. 手把手教你写一个java的orm(二)

    创建映射关系 ​ 想要实现一个orm的功能,我觉得就是要将class和数据库中的表创建映射关系.把class的名称和表的名称,class属性名称和表的字段名称,属性类型与表的字段类型一一对应起来.可以 ...

  4. 手把手教你写一个java的orm(三)

    使用反射解析class 上一篇我们完成了class到表映射关系的建立,但是这个并不能被代码正确处理,我们还需要让程序能够正确的识别这些映射关系. 这一篇主要讲的是建立一个从class到表的模型,使我们 ...

  5. 手把手教你写一个java的orm(四)

    开始准备生成sql 在上一篇里,我们已经取到了我们在生成sql语句中所需要的信息,这一篇里我们开始根据class来生成我们需要的sql.在这之前我们先确认几件事情 sql里的参数我们使用占位符的形式. ...

  6. 只有20行Javascript代码!手把手教你写一个页面模板引擎

    http://www.toobug.net/article/how_to_design_front_end_template_engine.html http://barretlee.com/webs ...

  7. 手把手教你写一个RPC

    1.1 RPC 是什么 定义:RPC(Remote Procedure Call Protocol)--远程过程调用协议 ,RPC协议假定某些传输协议的存在,如TCP或UDP,为通信程序之间携带信息数 ...

  8. 让我手把手教你写一个强大、方便使用的 IOC 容器

    一.介绍 1.介绍 最近无聊,也没什么事做,没事做总是要给自己找点事情做吧,毕竟人的生活在与折腾.于是,决定自己手动写一个 IOC 的框架.我们知道在 NetCore 的版本里面已经内置了 IOC 容 ...

  9. 手把手教你写一个SpringMVC框架

    一.介绍 在日常的 web 开发中,熟悉 java 的同学一定知道,Spring MVC 可以说是目前最流行的框架,之所以如此的流行,原因很简单:编程简洁.上手简单! 我记得刚开始入行的时候,最先接触 ...

随机推荐

  1. oh-my-zsh 安装及使用

    什么是 oh-my-zsh oh-my-zsh 是基于 zsh 的功能做了一个扩展,方便的插件管理.主题自定义,以及漂亮的自动完成效果. oh-my-zsh 安装及插件安装 查看系统中 zsh 及版本 ...

  2. Python web后端接收到的json数据有前端格式的布尔值 true false

    最近在后端处理前端传过来的json数据,发现,因为数据是各种数据格式的嵌套,使用json.loads(),无法将内层的数据转换为原来格式的数据,所以需要使用eval( )函数进行转换,但是如果数据含有 ...

  3. 日志分析与splunk浅谈

    难易程度:★★★ 阅读点:linux;python;web安全;日志分析; 文章作者:xiaoye 文章来源:i春秋 关键字:网络渗透技术 前言 linux下的日志分析对企业来说非常重要,对我们分析p ...

  4. Kafka Java 客户端开发

    依赖包导入 <dependency> <groupId>org.apache.kafka</groupId> <artifactId>kafka_2.1 ...

  5. 二:maven构建module

    通常情况下,我们一个项目是需要分多个模块的,这是我们用maven管理项目就需要构建一个多模块的项目: 通常的结构是一个模块中有一个主项目,下面包含多个子项目,如果是web项目则子项目中有一个是java ...

  6. 【wireshark】总体结构

    1. 总体结构 wireshark的总体结构如下图所示. 2. 功能模块 模块名 功能 源码子目录 GTK/Qt 处理所有的用户输入/输出(所有的窗口,对话框等等) /ui GTK: /ui/gtk ...

  7. 五:MyBatis学习总结(五)——实现关联表查询

    一.一对一关联 1.1.提出需求 根据班级id查询班级信息(带老师的信息) 1.2.创建表和数据 创建一张教师表和班级表,这里我们假设一个老师只负责教一个班,那么老师和班级之间的关系就是一种一对一的关 ...

  8. 6. Ensemble learning & AdaBoost

    1. ensemble learning 集成学习 集成学习是通过构建并结合多个学习器来完成学习任务,如下图: 集成学习通过将多个学习学习器进行结合,常可以获得比单一学习器更优秀的泛化性能 从理论上来 ...

  9. 【bzoj4240】 有趣的家庭菜园 树状数组

    这一题最终要构造的序列显然是一个单峰序列 首先有一个结论:一个序列通过交换相邻的元素,进行排序,最少的交换次数为该序列的逆序对个数 (该结论很久之前打表意外发现的,没想到用上了.....) 考虑如何构 ...

  10. commons-logging.jar 和 log4j.jar 的关系

    在用springmvc开发项目的时候,在日志管理这一块,我们一般用的都是log4j进行日志管理,但是我们在导入spring相关的jar的时候,都会看到commons-logging.jar包,为什么我 ...