一、背景现象

为了支持灵活的、可自定义的脱敏规则,工程拟采用velocity实现该目的,为此,自定义了:

  • mask、substr两个指令,其中
  • mask实现
public class MaskDirective extends Directive {

    @Override
public String getName() {
return "mask";
} @Override
public int getType() {
return LINE;
} @Override
public boolean render(InternalContextAdapter ctx, Writer writer, Node node) throws IOException, ResourceNotFoundException, ParseErrorException, MethodInvocationException { //fieldName
SimpleNode snField = (SimpleNode) node.jjtGetChild(0); String maskChar = (String)snField.value(ctx); snField = (SimpleNode) node.jjtGetChild(1);
int count = (int)snField.value(ctx); StringBuilder buff = new StringBuilder(); for(int i = 0; i < count; i++) {
buff.append(maskChar);
} writer.write(buff.toString()); return true; }
}
  • substr实现
public class SubstrDirective extends Directive {

    @Override
public String getName() {
return "substr";
} @Override
public int getType() {
return LINE;
} @Override
public boolean render(InternalContextAdapter ctx, Writer writer, Node node) throws IOException, ResourceNotFoundException, ParseErrorException, MethodInvocationException { //fieldName
SimpleNode snField = (SimpleNode) node.jjtGetChild(0); String fieldValue = (String)snField.value(ctx); if(fieldValue == null || fieldValue.length() == 0){
return false;
} int items = node.jjtGetNumChildren(); int startIdx = Integer.MIN_VALUE;
int endIdx = fieldValue.length(); if(items < 1){
return false;
} try{ snField = (SimpleNode) node.jjtGetChild(1);
startIdx = (int)snField.value(ctx); if(items > 2){
snField = (SimpleNode) node.jjtGetChild(2);
endIdx = (int)snField.value(ctx); if(endIdx > fieldValue.length()){
endIdx = fieldValue.length();
} } if(startIdx >= endIdx){
return false;
} StringBuilder buff = new StringBuilder(); if(items > 2){
writer.write(fieldValue.substring(startIdx,endIdx));
}else{ if(startIdx < 0){
startIdx = startIdx + fieldValue.length();
} writer.write(fieldValue.substring(startIdx));
} return true; }catch(Exception e){
return false;
}
}
}
  • velocity初始化方法:
public final class VelocityHelper {

    private Set<String> userExtDirective ;
private final static String K_USERDIRECTIVE = "userdirective"; private final static VelocityHelper VH = new VelocityHelper();
private VelocityHelper(){ } public static final VelocityHelper getInstance(){
return VH;
} /**
* 初始化
*/
public void init() throws Exception{ Properties prop = new Properties(); StringBuilder buff = new StringBuilder(); for(String s : userExtDirective){
buff.append(s.trim()).append(",");
} LogHelper.trace("start VelocityHelper.init[" + buff.toString() + "]"); prop.setProperty(K_USERDIRECTIVE, buff.toString().substring(0, buff.length()-1));
Velocity.init(prop); LogHelper.trace("end VelocityHelper.init"); } /**
* 根据报文模板和<code>VelocityContext</code>,渲染出实际的业务报文
*
* @param context 业务上下文信息
* @param template 模板
* @return
* @throws IOException
*/
public String evaluate(String template, final VelocityContext context) throws IOException {
Writer writer = new StringWriter();
try {
Velocity.evaluate(context, writer, StringUtil.EMPTY_STRING, template);
return writer.toString();
} finally {
IOUtils.closeQuietly(writer);
}
} /**
*
* @param userExtDirective
*
*/
public void setUserExtDirective(Set<String> userExtDirective){ this.userExtDirective = userExtDirective;
} }

部署到服务器后,以下表达式始终未得到替换:#substr($maskField, 0, 3)#mask('*', 4)#substr($maskField, -4) 

二、原因分析

开始怀疑是配置等问题,经过以下方法尝试均为解决问题:

1、怀疑spring xml配置问题----未解决

2、硬编码指令,在构造函数塞入指令----未解决

无奈,登录服务器grep velocity 得到了以下的输出(其中之一),茅塞顿开。

2019-02-02 07:49:52,315 [  - /// - ] INFO  assemble.VMRender - resource=class path resource [velocity/velocity.properties]

Velocity.ini及Velocity.evalute均是全局单一模式,既然有人捷足先登,自然不在做第二次初始化。

三、解决方案

采用VelocityEngine替换Velocity,实现jar包间隔离,独立管理自己的资源。

private VelocityEngine velocityEngine = null;
velocityEngine = new VelocityEngine(prop);
velocityEngine.evaluate(context, writer, StringUtil.EMPTY_STRING, template);

velocity自定义指令不生效问题解决之旅的更多相关文章

  1. vue自定义指令导致的内存泄漏问题解决

    vue的自定义指令是一个比较容易引起内存泄漏的地方,原因就在于指令通常给元素绑定了事件,但是如果忘记了解绑,就会产生内存泄漏的问题. 看下面代码: directives: { scroll: { in ...

  2. android中ListView点击和里边按钮点击不能同时生效问题解决

    今天遇到一个问题:android中ListView点击和里边button点击不能同时生效问题解决. 原因是: listView 在开始绘制的时候,系统首先调用getCount()函数,根据他的返回值得 ...

  3. Angularjs进阶笔记(2)-自定义指令中的数据绑定

    有关自定义指令的scope参数,网上很多文章都在讲这3种绑定方式实现的效果是什么,但几乎没有人讲到底怎么使用,本篇希望聊聊到底怎么用这个话题. 一. 自定义指令 自定义指令,是Angularjs用来实 ...

  4. 浅析AngularJS自定义指令之嵌入(transclude)

    AngularJS自定义指令的嵌入功能与vue的插槽十分类似,都可以实现一些自定义内容展现.在开始之前先简单介绍下自定义指令的transclude属性和AngularJS的内置指令ng-transcl ...

  5. 从零开始学 Web 之 Vue.js(二)过滤器,按键修饰符,自定义指令

    大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...

  6. angular自定义指令命名的那个坑

    Directive 先从定义一个简单的指令开始. 定义一个指令本质上是在HTML中通过元素.属性.类或注释来添加功能.AngularJS的内置指令都是以ng开头,如果想自定义指令,建议自定义一个前缀代 ...

  7. 移动端H5长按事件 vue自定义指令

    import Vue from 'vue' Vue.directive('longpress', function (el, binding){ var timer = null; var start ...

  8. vue 自定义指令的魅力

    [第1103期]vue 自定义指令的魅力 点点 前端早读课 2017-11-08 前言 很多事情不能做过多的计划,因为计划赶不上变化.今日早读文章由富途@点点翻译分享. 正文从这开始- 在你初次接触一 ...

  9. vue之自定义指令

    1.自定义指令的作用 除了核心功能默认内置的指令 (v-model 和 v-show),Vue 也允许注册自定义指令.注意,在 Vue2.0 中,代码复用和抽象的主要形式是组件.然而,有的情况下,你仍 ...

随机推荐

  1. MySQL必知必会第十一章-

    使用数据处理函数 大多数SQL支持以下类型的函数: 1> 文本函数:用于处理文本串(删除或填充值,转换值为大写或小写) 2> 数值函数:用于在数值数据上进行算术操作(返回绝对值,进行代数运 ...

  2. NLP VS NLU

    NLP(Natural Language Processing )自然语言处理:是计算机科学,人工智能和语言学的交叉领域.目标是让计算机处理或“理解”自然语言,以执行语言翻译和问题回答等任务.NLU  ...

  3. Winfrom BackgroundWorker

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Reflection ...

  4. vue 中 相同的路由不会跳转,更改路由的办法

    vue 开发的项目,路由跳转的时候,是相同的路由是不会跳转,页面也不会有更新的 有时候 必须要跳转怎么办, 更改一个参数,num,一直在改变.就可以进入了. this.$router.push({ p ...

  5. 和2018年年初做管理系统的不同(vuex)

    从2017年底开始做公司批改后台系统(服务内部人员对熊猫小课用户的作业进行批改.对批改员工资结算等)到教务系统(服务于内部人员对熊猫小课等移动端产品的内容进行配置等).ai-boss系统(服务于内部人 ...

  6. LeetCode 简单等级

    1. 两数之和 给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标. 你可以假设每种输入只会对应一个答案.但是,你不能重复利用这 ...

  7. Java tomcat Several ports (8005, 8080, 8009) required by Tomcat v9.0 Server at localhost

    关于 下面问题是因为(8005, 8080, 8009) 被原tomcat占用了. Several ports (8005, 8080, 8009) required by Tomcat v9.0 S ...

  8. 刘志梅 2017710101152《面向对象程序设计(java)》 第十周学习总结

    实验十  泛型程序设计技术 实验时间 2018-11-1 1.实验目的与要求 (1)泛型程序设计:意味着编写的代码可以被很多不同类型的对象所重用.(ArrayList类可以聚集任何类型的对象) 如果在 ...

  9. vue $refs 无法动态拼接,获取不到对象(转)

    原文地址: http://www.php.cn/js-tutorial-410304.html 本篇文章给大家带来的内容是关于vue $refs中不使用拼接的原因以及解决方法,有一定的参考价值,有需要 ...

  10. mongodb mac

    ==> mongodb To have launchd start mongodb now and restart at login: brew services start mongodb O ...