Mybatis获取多参数完整执行SQL

上一篇介绍了Mybatis如何获取查询接口的有序参数列表,这一篇会结合上一篇的内容解决实际问题,最近遇到了查询报表并将查询内容导出Excel文件的实际场景,需要获取到查询的完整SQL再通过导出工具导出Excel。

主要代码如下

//methodPath为执行方法的全限定名
//map为methodPath的参数列表,在查询接口多参数和单个参数的情况下会有一点差别
public String InSql(String methodPath, Map<String, Object> map) {
        BoundSql boundSql = sqlSessionFactory.getConfiguration().getMappedStatement(methodPath).getBoundSql(map);
        String sql = boundSql.getSql();
        //根据SQL获取参数列表
        List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();

        //定义集合存储映射参数
        List<Object> paramList = new ArrayList<>();

        //foreach标签标记 只进入if判断一次即把对应参数全部存入Set
        //由于foreach标签生成的元素的以"__frch_"开头的格式
        Set<String> flagSet = new HashSet<>();
        for (ParameterMapping pm : parameterMappings) {
            //判断是否为输入参数
            if (pm.getMode() != ParameterMode.OUT) {
                //获取Dao接口参数名
                String paramName = pm.getProperty();

                if (paramName.contains("__frch_")) {
                    int firstIndex = paramName.lastIndexOf("_", paramName.lastIndexOf("_") - 1);
                    int lastIndex = paramName.lastIndexOf("_");
                    //获取参数名称,这里获取到的参数名为XML文件中foreach标签中的item元素的值,这个值需要与第二层参数列表map设置的key保持一致才能获取到value。
                    paramName = paramName.substring(firstIndex + 1, lastIndex);
                    if (!flagSet.contains(paramName)) {
                        Map param = (Map) map.get("param");
                        String paramValues = (String) param.get(paramName);
                        String[] strings = paramValues.split(",");
                        for (String value : strings) {
                            paramList.add(value);
                        }
                        flagSet.add(paramName);
                    }
                } else if (paramName.contains("param.")) {
                    paramName = paramName.replace("param.", "");
                    Map param = (Map) map.get("param");
                    Object paramValue = param.get(paramName);
                    paramList.add(paramValue);
                } else {
                    Object o = map.get(paramName);
                    paramList.add(o);
                }
            }
        }
        while (sql.indexOf("?") != -1 && paramList != null && paramList.size() > 0 && paramList.get(0) != null) {
            Object paramValue = paramList.get(0);
            if (paramValue != null) {
                String value = "\'" + paramValue.toString() + "\'";
                /*if (paramValue instanceof String){
                    value = "\'"+paramValue.toString()+"\'";
                }else if(paramValue instanceof Date || paramValue instanceof Timestamp){
                    value = DateUtils.parseDateToStr("yyyy-MM-dd HH:mm:ss", (Date) paramValue);
                    value = "str_to_date('" + value + "','%Y-%m-%d %T')";
                }*/
                sql = sql.replaceFirst("\\?", value);
                paramList.remove(0);
            }
        }
        return sql;
    }

多个参数封装

//查询接口
List<Map<String,Object>> getCarBackupData(@Param("currentUserId") int currentUserId, @Param("param") YsSearchCarParam param);

假设需要获取上面多参数接口的完整SQL,封装map参数:
Map<String,Object> param = new hashmap();
//在查询接口获取到currentUserId的值
param.put("currentUserId",param.getCurrentUserId);
//封装第二个map,key为XML文件中判断条件的字符串,value为判断条件的值
Map<String,Obejct> map = new hashMap();
map.put("yy1","yy1");
//在XML文件解析时,xx2被foreach标签解析
//生成类似于__frch_xx0:xx1,__frch_xx1:xx2,__frch__xx2:xx3的三个额外参数
map.put("xx2","xx1,xx2,xx3");
//再由最外层的map将第二参数的值封装起来,这里的键是@param设置的值
param.put("param",map);

单个参数封装

//查询接口
List<YsPeopleResult> selectPeopleLibResultList(YsSearchPeopleParam param);

//直接将SQL中需要的过滤参数放入map即可,map的键为XML文件中对应判断条件字符串
Map<String, Object> map = new HashMap<>();
map.put("searchNum", pici);

如果有bug或问题,欢迎各位在评论区交流。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
【源码】mysql版本_spring3.0 系统模块 1. 组织管理:角色管理,分角色组和成员,有组权限和成员权限。 2. 系统用户:对各个基本的组会员增删改查,单发、群发邮件短信,导入导出excel表格,批量删除 3. 会员管理:对前台用户管理,分配会员级别,到期时间,状态,联系信息等资料 4. 菜单管理:增删改查菜单 ztree(自定义菜单)业务菜单和系统菜单分离 5. 数据字典:无限级别,支持多级别无限分类。内设编号,排序等 6. 系统设置:修改系统名称,邮件服务器配置,短信账号设置,图片水印配置,微信配置 7. 代码生成:打开代码生成器模块 8. 图库管理:对批量上传的图片统一管理 9. 性能监控:监控整个系统的性能,SQL监控,SQL防火墙,URL监控,SPRING监控,SESSION监控等 10. 接口测试:POST or GET 方式检测系统接口,参数加密,json返回结果,计算服务器响应时间 11. 发送邮件:单发,群发邮件 12. 置二维码:生成 or 解析二维码 13.地图工具:经纬度操作 14.即时通讯:打开即时聊天窗口 技术点 1. 导出 导入 excel 文件 2 导出word文件 3. IO 流上传下载文件 4. 群发邮件,可以发html、纯文本格式,可以发给任意邮箱(实现批量发送广告邮件) 5. 群发or单独 发送短信,支持两种第三方短信商接口 6. spring aop 事物处理 7. 代码生成器 (freemarker), 代码 zip 压缩打包 8. MD5加密 SHA加密(登录密码用此加密) 9. 数据库连接池 阿里的 druid。Druid在监控、可扩展性、稳定性和性能方面都有明显的优势,支持并发 10.加入安全框架 shiro (登录授权)(session管理) 11.根据汉字 解析汉字的全拼(拼音)和首字母(导入excel到用户表,根据用户的汉字姓名生成拼音的用户名) 12.app接口(支持与其它语言数据交互) 12.极光推送 (推送给APP及时消息,APP不启动也能收到) 14.微信接口(身份验证,文本、图文回复等) 微信远程控制服务器重启、锁定、其它应用程序 15.java Quartz 定时器 (定时执行某程序,精确到秒,可设置周期) 16.java websocket 即时通讯技术,点对点,群聊,单聊,EXT4对话框 17.新增Lucene全文检索 18.Base64传输图片 19.图片加水印(图片水印,文字水印) 20.生成 or 解析 二维码 21.HTML5 + JAVAEE WebSocket 通信技术 22.批量异步上传图片,可预览,有进度条,支持拖拽上传(百度webuploader )。列表动态滑动放大展示。 23.ehcache 自定义二级缓存 ,选择缓存存放目录,处理并发,增加系统性能 24.服务器内部GET POST 请求 25.uploadify 上传插件,单条、批量上传,带进度条,异步,图片、视频, 其它文件格式均可上传 26.地图选点获取经纬度坐标,根据俩经纬度计算距离
下面是一个示例,在这个示例中,我们将使用MyBatis执行两个SQL语句:一个是插入一条用户记录,另一个是更新一条用户记录。 1. 创建一个Mapper接口,定义插入和更新用户记录的方法。 ``` public interface UserMapper { void insertUser(User user); void updateUser(User user); } ``` 2. 在Mapper.xml文件中,编写插入和更新用户记录的SQL语句,并使用<batch>标签将它们组合在一起。 ``` <mapper namespace="com.example.mapper.UserMapper"> <insert id="insertUser"> INSERT INTO user(name, age) VALUES(#{name}, #{age}) </insert> <update id="updateUser"> UPDATE user SET name=#{name}, age=#{age} WHERE id=#{id} </update> <batch> ${sql} </batch> </mapper> ``` 在上面的Mapper.xml文件中,我们使用了${sql}占位符来代表要执行的多条SQL语句。在执行batch()方法时,我们将通过如下代码来设置这个占位符的值: ``` Map<String, Object> paramMap = new HashMap<>(); List<String> sqlList = new ArrayList<>(); sqlList.add("INSERT INTO user(name, age) VALUES('Tom', 20)"); sqlList.add("UPDATE user SET name='Jerry', age=21 WHERE id=1"); paramMap.put("sql", StringUtils.join(sqlList, ";")); session.update("com.example.mapper.UserMapper.batch", paramMap); ``` 在上面的代码中,我们创建了一个包含两条SQL语句的List,并将它们使用分号(;)拼接为一个字符串,在调用batch()方法时,将这个字符串设置为${sql}占位符的值。 3. 在Java代码中,使用SqlSession的batch()方法来执行多条SQL语句。 ``` SqlSession session = sqlSessionFactory.openSession(ExecutorType.BATCH, false); UserMapper mapper = session.getMapper(UserMapper.class); mapper.insertUser(new User("Tom", 20)); mapper.updateUser(new User(1, "Jerry", 21)); session.flushStatements(); ``` 在上面的代码中,我们首先通过SqlSessionFactory创建了一个SqlSession对象,并使用ExecutorType.BATCH参数指定了执行模式为BatchExecutor。然后,我们通过SqlSession的getMapper()方法获取了UserMapper接口的代理对象,并使用它来分别执行插入和更新用户记录的方法。最后,我们调用SqlSession的flushStatements()方法来提交执行结果。 注意,在调用batch()方法之前,我们需要将SqlSession的自动提交模式设置为false,以便在执行完所有SQL语句之后再手动提交执行结果。此外,在使用BatchExecutor执行多条SQL语句时,需要注意SQL语句之间的依赖关系,确保它们的执行顺序正确。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值