1. 分析
JFinal中刚新增like指令
在这之前大家的通常用法一般是自己拼装,如MySQL的用法:like concat('%', #para(title), '%')
自带的like指令功能单一,个人认为需要增强。
大神的like指令
https://jfinal.com/share/398
2. 方案
为了和JFinal新增指令不冲突,本指令命名为likePara。
like指令应该具备通用性强、自动适配百分号、支持多参数拼装等:
WHERE t.title #likePara("title", [1, '2', 3L], "4"),被解析为WHERE t.title LIKE ?,参数值为:'%title1234%';可以看出,多个不同类型的参数会toString()拼装,两端没有百分号则自动补充百分号,拼装规则为全Like;
#likePara('%', "value") 和 #likePara("%value"),参数值为:'%value',左边出现了百分号,拼装规则为左Like;同样,右Like拼装方式也一样;
3. 自定义标签
源码,基于Jboot的JbootDirectiveBase。:
package com.jfinal.plugin.activerecord.sql; import java.util.ArrayList; import java.util.Collection; import java.util.List; import com.jfinal.plugin.activerecord.SqlPara; import com.jfinal.template.Env; import com.jfinal.template.TemplateException; import com.jfinal.template.expr.ast.ExprList; import com.jfinal.template.io.Writer; import com.jfinal.template.stat.ParseException; import com.jfinal.template.stat.Scope; import io.jboot.web.directive.base.JbootDirectiveBase; public class LikeParaDirective extends JbootDirectiveBase { public static final String DIRECTIVE_NAME = "likePara"; @Override public void onRender(Env env, Scope scope, Writer writer) { SqlPara sqlPara = (SqlPara)scope.get(SqlKit.SQL_PARA_KEY); if (sqlPara == null) { throw new TemplateException("#" + DIRECTIVE_NAME + " directive invoked by getSqlPara(...) method only", this.location); } Object[] paraArray = this.exprList.evalExprList(scope); List<Object> paraList = new ArrayList<>(); /* * 首先将参数全规划成List */ for (Object para : paraArray) { if (para == null) { // 优先处理参数为Null的情况 throw new ParseException("The parameter of #" + DIRECTIVE_NAME + " directive can not be null", this.location); } else if (para instanceof Collection<?>) { // 处理参数为Collection的情况 paraList.addAll((Collection<?>)para); } else if (para.getClass().isArray()) { // 处理参数为Array的情况 for (Object object : (Object[])para) { paraList.add(object); } } else { // 处理参数为一般类型的情况 paraList.add(para); } } boolean leftLike = false; boolean rightLike = false; /* * 对规划好的参数List进行处理 */ // 处理左百分号(%) if ("%".equals(paraList.get(0))) { // 单独为百分号的情况 leftLike = true; paraList.remove(0); } else if (String.valueOf(paraList.get(0)).startsWith("%")) { // 百分号开头的情况 leftLike = true; paraList.set(0, String.valueOf(paraList.get(0)).substring(1)); } // 处理右百分号(%) if (paraList.size() > 0) { int lastIndex = paraList.size() - 1; if ("%".equals(paraList.get(lastIndex))) { // 单独为百分号的情况 rightLike = true; paraList.remove(lastIndex); } else if (String.valueOf(paraList.get(lastIndex)).endsWith("%")) { // 百分号结尾的情况 rightLike = true; paraList.set(lastIndex, String.valueOf(paraList.get(0)).substring(0, String.valueOf(paraList.get(0)).length() - 1)); } } /* * 构建LIKE参数字串 */ StringBuilder sb = new StringBuilder(); // 左百分号(%) sb.append(leftLike ? '%' : ""); sb.append(!(leftLike || rightLike) ? '%' : ""); // 左右都不包括百分号,则默认加上 // 参数字串 for (Object para : paraList) { sb.append(para); } // 右百分号(%) sb.append(rightLike ? '%' : ""); sb.append(!(leftLike || rightLike) ? '%' : ""); // 左右都不包括百分号,则默认加上 /* * 将解析好的参数给设置到SQL模板中 */ this.write(writer, "LIKE ?"); sqlPara.addPara(sb.toString()); } @Override public void setExprList(ExprList exprList) { if (exprList.length() == 0) { throw new ParseException("The parameter of #" + DIRECTIVE_NAME + " directive can not be blank", this.location); } this.exprList = exprList; } }
4. 测试
结果:
--------------------------------------------------- 模板:#likePara('%', val, [1, '2', 3L], "4", '%') 入参:{"val":["v","alue",0]} 结果:LIKE '%value01234%' --------------------------------------------------- 模板:#likePara('%', val, [1, '2', 3L], "4", '%') 入参:{"val":["valu","e",0]} 结果:LIKE '%value01234%' --------------------------------------------------- 模板:#likePara(val, '%') 入参:{"val":"value"} 结果:LIKE 'value%' --------------------------------------------------- 模板:#likePara(val, '%') 入参:{"val":"%value"} 结果:LIKE '%value%' --------------------------------------------------- 模板:#likePara(val, '%') 入参:{"val":"va%lue"} 结果:LIKE 'va%lue%' --------------------------------------------------- 模板:#likePara('%', val, '%') 入参:{"val":"value"} 结果:LIKE '%value%' --------------------------------------------------- 模板:#likePara('%', val, '%') 入参:{"val":"value%"} 结果:LIKE '%value%%' --------------------------------------------------- 模板:#likePara('%', val, '%') 入参:{"val":"va%lue"} 结果:LIKE '%va%lue%' --------------------------------------------------- 模板:#likePara('%') 入参:{} 结果:LIKE '%' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":"%"} 结果:LIKE '%' --------------------------------------------------- 模板:#likePara('%', val) 入参:{"val":"value"} 结果:LIKE '%value' --------------------------------------------------- 模板:#likePara('%', val) 入参:{"val":"value%"} 结果:LIKE '%value%' --------------------------------------------------- 模板:#likePara('%', val) 入参:{"val":"va%lue"} 结果:LIKE '%va%lue' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":"value"} 结果:LIKE '%value%' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":"%value"} 结果:LIKE '%value' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":"value%"} 结果:LIKE 'value%' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":"va%lue"} 结果:LIKE '%va%lue%' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":["v","alue"]} 结果:LIKE '%value%' --------------------------------------------------- 模板:#likePara(val) 入参:{"val":["val","ue"]} 结果:LIKE '%value%'