首先先对这段代码的简单介绍,我之前在需要操作JDBC的时候总是会因为经常要重新写SQL语句感到很麻烦。所以就能拿则拿不能拿的就简单地封装了一下。

首先是Insert。Spring框架的JDBC包里面的.core.包里面的simple包有个SimpleJdbcInsert这样一个类,它给我们带来了很好的对于插入数据的实现。

  它只需要我们提供表单名称,和一个Map就可以帮我们轻松把数据放到数据库。

  1. public int insertObject(Map<String, Object> entity, String tableName) {
  2. SimpleJdbcInsert insert = new SimpleJdbcInsert(jdbcTemplate);
  3. insert.setTableName(tableName);
  4. int result = insert.execute(entity);
  5. return result;
  6. }

  这个类里面有个execute方法需要我们穿进去一个Map就能把Map里面的数据插入到数据库了,具体是怎么用的呢?下面我们就来看一下

  1. @Test
  2. public void insert() {
  3. try {
  4. Map<String, Object> entity = new HashMap<>();
  5. entity.put("STU_NAME", "AAA");
  6. entity.put("STU_AGE", 66);
  7. entity.put("STU_UPDATE_DATE", new Date());
  8. studentService.insertObject(entity, "tb_student");
  9. } catch (Exception e) {
  10. e.printStackTrace();
  11. }
  12. }

这是Service层的代码

  1. @Transactional
  2. @Override
  3. public int insertObject(Map<String, Object> entity, String tableName) {
  4. dbDao.insertObject(entity, tableName);
  5. return dbDao.insertObject(entity, tableName);
  6. }

 这样写就可以了。嗯。就这么简单,那···它是怎么实现的呢? 

  1. protected int doExecute(Map<String, ?> args) {
  2. checkCompiled();
  3. List<Object> values = matchInParameterValuesWithInsertColumns(args);
  4. return executeInsertInternal(values);
  5. }

  这里就是它的excute方法的底层的首层代码。这里可以看到需要我们传入一个HashMap对象,然后通过另外一个类对这个HashMaP进行解析得到一个List的对象数组。

那···这里它又是怎样解析的呢?于是我出于好奇继续对它进行“刨根问底”

  1. public List<Object> matchInParameterValuesWithInsertColumns(Map<String, ?> inParameters) {
  2. List<Object> values = new ArrayList<Object>();
  3. Map<String, Object> source = new LinkedHashMap<String, Object>(inParameters.size());
  4. for (String key : inParameters.keySet()) {
  5. source.put(key.toLowerCase(), inParameters.get(key));
  6. }
  7. for (String column : this.tableColumns) {
  8. values.add(source.get(column.toLowerCase()));
  9. }
  10. return values;
  11. }

  点进去一看。嗯就是通过matchInParameterValuesWithInsertColumns这个类来实现这个Map解析的。这个实现非常简单就是把我们给他的HashMap转换成LinkHashMap。这样就能把散落在内存各处的数据串成一个有序的HashMap了。

将HashMap的数据统一它们的大小写,之后有序地丢进去这个LinkHashMap。接着通过复杂的算法获得数据库表到底有多少列,接着循环把这个HashMap的数据一个个丢到List里面,接着返回一个对象List。再进行Sql拼接

  1. public String createInsertString(String... generatedKeyNames) {
  2. Set<String> keys = new LinkedHashSet<String>(generatedKeyNames.length);
  3. for (String key : generatedKeyNames) {
  4. keys.add(key.toUpperCase());
  5. }
  6. StringBuilder insertStatement = new StringBuilder();
  7. insertStatement.append("INSERT INTO ");
  8. if (getSchemaName() != null) {
  9. insertStatement.append(getSchemaName());
  10. insertStatement.append(".");
  11. }
  12. insertStatement.append(getTableName());
  13. insertStatement.append(" (");
  14. int columnCount = 0;
  15. for (String columnName : getTableColumns()) {
  16. if (!keys.contains(columnName.toUpperCase())) {
  17. columnCount++;
  18. if (columnCount > 1) {
  19. insertStatement.append(", ");
  20. }
  21. insertStatement.append(columnName);
  22. }
  23. }
  24. insertStatement.append(") VALUES(");
  25. if (columnCount < 1) {
  26. if (this.generatedKeyColumnsUsed) {
  27. logger.info("Unable to locate non-key columns for table '" +
  28. getTableName() + "' so an empty insert statement is generated");
  29. }
  30. else {
  31. throw new InvalidDataAccessApiUsageException("Unable to locate columns for table '" +
  32. getTableName() + "' so an insert statement can't be generated");
  33. }
  34. }
  35. for (int i = 0; i < columnCount; i++) {
  36. if (i > 0) {
  37. insertStatement.append(", ");
  38. }
  39. insertStatement.append("?");
  40. }
  41. insertStatement.append(")");
  42. return insertStatement.toString();
  43. }

  这里我们可以看到这就是用了一个StringBuilder把Sql的关键字和我们传进去的数据库表的名称,还有表的列名和Values之后的我们需要插入的值。最后把完整的SQL语句再传给数据库的执行对象返回受影响条数的Int类型的结果。

这里得出一个结论,我们在学习的过程中,通道说源代码就感觉是非常高大上的对自己来说是遥不可及的东西,实际上点进去分析一下就可以发现在过往在学校的学习中,我们已经接触过这个知识,只是在用的时候缺乏经验,不知道

如何去在脑海里面把知识点取出来。所以就妨碍了我们在代码上的实现。而且对于学习需要我们拥有一个“刨根问底的心”。心中有了疑问就会不知不觉地去追求这个答案。哪怕最后的结果是个很简单的东西。

我们知道了数据库有多少个字段就只需要判断SQL里面需要放多少个非空字段的内容进去就可以了。希望今天写的博文对大家有用。

第一次玩博客,今天被安利了一个很方便JDBC的基于Spring框架的一个叫SimpleInsert的类,现在就来简单介绍一下的更多相关文章

  1. 第一次玩博客 感觉自己特别low

    第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动第一天来  来好激动 ...

  2. 第一次写博客Poj1044

    Date bugs Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 3005   Accepted: 889 Descript ...

  3. Beta版本——第一次冲刺博客

    我说的都队 031402304 陈燊 031402342 许玲玲 031402337 胡心颖 03140241 王婷婷 031402203 陈齐民 031402209 黄伟炜 031402233 郑扬 ...

  4. 第一次写博客,关于前端开发deMVC在js中的应用

    对前端MVC MVC分别是model.view.controller的缩写,模型.视图.控制器.这些更加偏向于后台,在以前MVC是只属于后台的.当然随着技术的进步,前端的大牛们将后台的一些东西应用于前 ...

  5. HDU 2064 菜鸡第一次写博客

    果然集训就是学长学姐天天传授水铜的动态规划和搜索,今天讲DP由于困意加上面瘫学长"听不懂就是你不行"的呵呵传授,全程梦游.最后面对连入门都算不上的几道动态规划,我的内心一片宁静,甚 ...

  6. (转)第一次发博客-说说我的B/S开发框架(asp.net mvc + web api + easyui)

    原文地址:http://www.cnblogs.com/xqin/archive/2013/05/29/3105291.html 前言 这些年一直在.net下做企业web系统开发,前前后后经历的不同的 ...

  7. 第一次的博客-简单的Markdown语法

    第一次开通博客 从网上寻找了许多写博客的方法 我觉得对于我来说还是使用 Markdowm 比较容易一些 用我的第一篇博客来记一下一些比较简单常用的Markdown语法 本文参考来源Markdown语法 ...

  8. 第一次写博客,就写如何向外行介绍自己做的是什么,那个我是做web的

    如果想外行问你是做什么的,改如何回答.和内行说java后台就可以了,但外行听不懂,我们该如何描述呢? 我的方法是:我做的是java web开发,不是内外的外,是个英文单词web,全名叫world wi ...

  9. 第一次使用博客及Coursera课程体验

    前言: 第一天的学习目标有三个 开设博客园账户 开设Github账号 进行第一次coursera课程学习:Internet History, Technology, and Security  (网址 ...

随机推荐

  1. The Last Week

    二轮省选前的最后一周了呢. 一路走到这里,真的很希望能继续走下去. 好好调整一下状态,争取能有机会买D吧(虽然现在似乎D也没什么用了 day1 多项式 多项式ln 多项式exp day2 数据结构 L ...

  2. [GXOI/GZOI2019]旧词

    很像LNOI 2014 LCA那道题. 同样的套路,离线以后直接扫描线. k=1的话就是原题. 考虑一般情况. 原本的做法是对x到根的这条链做一下区间+1操作,目的是为了是的在深度为i的位置得到的贡献 ...

  3. elementUi中input输入字符光标在输入一个字符后,光标失去焦点

    elementUi中input输入字符光标在输入一个字符后,光标就退出,无法输入需要再次聚焦然后输入一个字符又再次退出 首先,用elementUi正常用v-model绑定输入的值是不会造成光标退出的, ...

  4. 微信小程序分包跳转主包页面

    由于公司项目比较多,我们事业部的微信小程序就在一个分包里.那分包页面要回到主包的首页,该怎么跳转呢,有以下两种方法 wx.switchTab(Object object) 跳转到 tabBar 页面, ...

  5. shell脚本 快照还原Hbase数据库

    #!/bin/bash for i in $(cat ./hbaseTable);do echo "disable '$i'" | hbase shellecho "re ...

  6. SQL-在Update中进行子查询和左联查询

    以下总结源自后边的三个参考思索和测试而来: 我们有一张行政区划表,为了查询速度的优化,我们需要在这张表中,将每个乡镇的记录中写入其所属的省.市.县, 表如下: 当然,我们可以使用游标或在存储过程中使用 ...

  7. Angular7.1.4+Typescript3.1框架学习(三)

    接着第二部分,这部分介绍常见angular指令使用 1. 在父页面文件中添加组件(通过ng命令生成的component) <app-messages></app-messages&g ...

  8. 利用itext将html页面转成pdf(不模糊)

    1.maven项目进入依赖 <dependency> <groupId>org.xhtmlrenderer</groupId> <artifactId> ...

  9. 静态方法和实例方法(mark)

    借花献佛[转自 ivony's blog] 关于静态方法和实例方法的一些误区. 一.    静态方法常驻内存,实例方法不是,所以静态方法效率高但占内存.     事实上,方法都是一样的,在加载时机和占 ...

  10. day42-python消息队列一

    消息队列”是在消息的传输过程中保存消息的容器.消息队列最经典的用法就是消费者和生成者之间通过消息管道来传递消息,消费者和生成者是不通的进程.生产者往管道中写消息,消费者从管道中读消息.操作系统提供了很 ...