mybatis源码parsing包--PropertyParser文档解析中的变量替换

mybatis的PropertyParser在处理SQL语句时,负责将占位符如#{id}替换为实际值。它基于GenericTokenParser的parser方法,通过openToken("#{")和closeToken("}")识别占位符。解析过程包括将SQL字符串转换为字符数组,逐个处理占位符,调用TokenHandler的handleToken方法获取真实值,并将值插入到构建的SQL语句中,从而完成变量替换。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

mybatis中对占位符替换成实际的变量值用到的就是PropertyParser类。例如 select * from tb where id = #{id} 需要替换成 select * from tb where id = 123

 

PropertyParser的parse方法实际上使用了GenericTokenParser 的parser方法,下面分析下GenericTokenParser的parser方法

其中GenericTokenParser类中有如下三个属性

SELECT * FROM t_action WHERE `id`=#{id} and name = #{name} ORDER BY `actionTime`  openToken就是 “#{”  closeToken 就是 “ } ”

下面这段代码干了这么个事:

把传入的sql模板转成字符数组,builder用来存储最终的sql,先截取 SELECT * FROM t_action WHERE `id`=  存到 builder然后截取出 id,然后调用TokenHandler的handleToken方法获取id的实际值再拼到builder后边,比如id实际值是123 这时builder就得到 SELECT * FROM t_action WHERE `id`=123 ,依次类推把sql模板中的所有占位符都替换掉。

  /**
   *
   * @param text 输入:
   *             SELECT * FROM t_action
   *          WHERE `id`=#{id} and name = #{name}
   *         ORDER BY `actionTime`
   *
   * @return 输出:
   *          SELECT * FROM t_action
   *          WHERE `id`=? and name = ?
   *         ORDER BY `actionTime`
   */
  public String parse(String text) {
    if (text == null || text.isEmpty()) {
      return "";
    }
    // search open token
    int start = text.indexOf(openToken);
    if (start == -1) {
      return text;
    }
    char[] src = text.toCharArray();
    int offset = 0;
    final StringBuilder builder = new StringBuilder();
    //表达式 例如 #{name} 中的name
    StringBuilder expression = null;
    while (start > -1) {
      if (start > 0 && src[start - 1] == '\\') {
        // this open token is escaped. remove the backslash and continue.
        builder.append(src, offset, start - offset - 1).append(openToken);
        offset = start + openToken.length();
      } else {
        // found open token. let's search close token.
        if (expression == null) {
          expression = new StringBuilder();
        } else {
          expression.setLength(0);
        }
        builder.append(src, offset, start - offset);
        offset = start + openToken.length();
        int end = text.indexOf(closeToken, offset);
        while (end > -1) {
          if (end > offset && src[end - 1] == '\\') {
            // this close token is escaped. remove the backslash and continue.
            expression.append(src, offset, end - offset - 1).append(closeToken);
            offset = end + closeToken.length();
            end = text.indexOf(closeToken, offset);
          } else {
            expression.append(src, offset, end - offset);
            break;
          }
        }
        if (end == -1) {
          // close token was not found.
          builder.append(src, start, src.length - start);
          offset = src.length;
        } else {
          // 这里调用了传入的handler的方法
          builder.append(handler.handleToken(expression.toString()));
          offset = end + closeToken.length();
        }
      }
      start = text.indexOf(openToken, offset);
    }
    if (offset < src.length) {
      builder.append(src, offset, src.length - offset);
    }
    return builder.toString();
  }

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值