1. 程式人生 > 程式設計 >java解析{{}}變數名以及文字內容替換操作

java解析{{}}變數名以及文字內容替換操作

發簡訊、發郵件的時候經常會遇到模板內容需要替換成實際資料的問題,本文介紹從文字模板中解析出變數列表,以及引數填入後得到實際文字內容的辦法:

/**
 * 根據正則表示式獲取文字中的變數名列表
 * @param pattern
 * @param content
 * @return
 */
public static List<String> getParams(String pattern,String content) {
 Pattern p = Pattern.compile(pattern);
 Matcher m = p.matcher(content);
 
 List<String> result = new ArrayList<String>();
 while (m.find()) {
  result.add(m.group(1));
 }
 return result;
}
 
/**
 * 根據正則表示式將文字中的變數使用實際的資料替換成無變數的文字
 * @param pattern
 * @param content
 * @param data
 * @return
 */
public static String parse(String pattern,String content,Map<String,String> data) {
 Pattern p = Pattern.compile(pattern);
 Matcher m = p.matcher(content);
 
 StringBuffer sb = new StringBuffer();
 while (m.find()) {
  String key = m.group(1);
  String value = data.get(key);
  m.appendReplacement(sb,value == null ? "" : value);
 }
 m.appendTail(sb);
 return sb.toString();
}
 
public static void main(String[] args) {
 String content = "恭喜{{姓名}}報名成功,請憑報名編號{[code]}到現場參加活動";
 String reg = "\\{\\{(.+?)\\}\\}";
 List<String> params = getParams(reg,content);
 System.out.println(params);
 
 Map<String,String> data = new HashMap<String,String>();
 data.put("姓名","張三丰");
 data.put("code","930118");
 String text = parse(reg,content,data);
 System.out.println(text);
}

上面的程式碼介紹的是針對{{}}形式的變數值的解析辦法,大家可以修改一下正則表示式,改成可以解析${}變數的辦法

補充知識:java模板字串優雅解析(佔位符解析)

專案中常常需要解析字串模板,比如user:{userId}:{userType}用於redis的key等,比較常見的做法就是使用String.format(“user:%s:%s”,1,1)方法,但個人感覺那樣的模板不夠明瞭,而使用模板解析器可更好地有助於解析此類字串。

可使用map用於解析,也可使用物件進行解析,也可使用類似String.format可變引數進行解析,多樣化解析物件值。有點類似於js的模板字串${}。

也可自定義字首字尾進行解析。如PlaceholderResolver.getResolver("{",“}”)該物件可解析{}該型別的佔位符.

效能方面:

java解析{{}}變數名以及文字內容替換操作

java解析{{}}變數名以及文字內容替換操作

PlaceholderResolve解析結果

String.format解析結果

解析100w次,佔位符解析比String.format平均都會快個0.4~0.5s,so 效能應該問題不大。最主要的是不需要呼叫太多的物件方法,自動解析,方便快捷。

原始碼如下:

/**
 * 佔位符解析器
 *
 * @author meilin.huang
 * @version 1.0
 * @date 2018-11-13 1:42 PM
 */
public class PlaceholderResolver {
 /**
  * 預設字首佔位符
  */
 public static final String DEFAULT_PLACEHOLDER_PREFIX = "${";

 /**
  * 預設字尾佔位符
  */
 public static final String DEFAULT_PLACEHOLDER_SUFFIX = "}";

 /**
  * 預設單例解析器
  */
 private static PlaceholderResolver defaultResolver = new PlaceholderResolver();

 /**
  * 佔位符字首
  */
 private String placeholderPrefix = DEFAULT_PLACEHOLDER_PREFIX;

 /**
  * 佔位符字尾
  */
 private String placeholderSuffix = DEFAULT_PLACEHOLDER_SUFFIX;


 private PlaceholderResolver(){}

 private PlaceholderResolver(String placeholderPrefix,String placeholderSuffix) {
  this.placeholderPrefix = placeholderPrefix;
  this.placeholderSuffix = placeholderSuffix;
 }

 /**
  * 獲取預設的佔位符解析器,即佔位符字首為"${",字尾為"}"
  * @return
  */
 public static PlaceholderResolver getDefaultResolver() {
  return defaultResolver;
 }

 public static PlaceholderResolver getResolver(String placeholderPrefix,String placeholderSuffix) {
  return new PlaceholderResolver(placeholderPrefix,placeholderSuffix);
 }

 /**
  * 解析帶有指定佔位符的模板字串,預設佔位符為字首:${ 字尾:}<br/><br/>
  * 如:template = category:${}:product:${}<br/>
  * values = {"1","2"}<br/>
  * 返回 category:1:product:2<br/>
  *
  * @param content 要解析的帶有佔位符的模板字串
  * @param values 按照模板佔位符索引位置設定對應的值
  * @return
  */
 public String resolve(String content,String... values) {
  int start = content.indexOf(this.placeholderPrefix);
  if (start == -1) {
   return content;
  }
  //值索引
  int valueIndex = 0;
  StringBuilder result = new StringBuilder(content);
  while (start != -1) {
   int end = result.indexOf(this.placeholderSuffix);
   String replaceContent = values[valueIndex++];
   result.replace(start,end + this.placeholderSuffix.length(),replaceContent);
   start = result.indexOf(this.placeholderPrefix,start + replaceContent.length());
  }
  return result.toString();
 }

 /**
  * 解析帶有指定佔位符的模板字串,預設佔位符為字首:${ 字尾:}<br/><br/>
  * 如:template = category:${}:product:${}<br/>
  * values = {"1",Object[] values) {
  return resolve(content,Stream.of(values).map(String::valueOf).toArray(String[]::new));
 }

 /**
  * 根據替換規則來替換指定模板中的佔位符值
  * @param content 要解析的字串
  * @param rule 解析規則回撥
  * @return
  */
 public String resolveByRule(String content,Function<String,String> rule) {
  int start = content.indexOf(this.placeholderPrefix);
  if (start == -1) {
   return content;
  }
  StringBuilder result = new StringBuilder(content);
  while (start != -1) {
   int end = result.indexOf(this.placeholderSuffix,start);
   //獲取佔位符屬性值,如${id},即獲取id
   String placeholder = result.substring(start + this.placeholderPrefix.length(),end);
   //替換整個佔位符內容,即將${id}值替換為替換規則回撥中的內容
   String replaceContent = placeholder.trim().isEmpty() ? "" : rule.apply(placeholder);
   result.replace(start,start + replaceContent.length());
  }
  return result.toString();
 }

 /**
  * 替換模板中佔位符內容,佔位符的內容即為map key對應的值,key為佔位符中的內容。<br/><br/>
  * 如:content = product:${id}:detail:${did}<br/>
  * valueMap = id -> 1; pid -> 2<br/>
  * 經過解析返回 product:1:detail:2<br/>
  *
  * @param content 模板內容。
  * @param valueMap 值對映
  * @return 替換完成後的字串。
  */
 public String resolveByMap(String content,final Map<String,Object> valueMap) {
  return resolveByRule(content,placeholderValue -> String.valueOf(valueMap.get(placeholderValue)));
 }

 /**
  * 根據properties檔案替換佔位符內容
  * @param content
  * @param properties
  * @return
  */
 public String resolveByProperties(String content,final Properties properties) {
  return resolveByRule(content,placeholderValue -> properties.getProperty(placeholderValue));
 }

 /**
  * 根據物件中欄位路徑(即類似js訪問物件屬性值)替換模板中的佔位符 <br/><br/>
  * 如 content = product:${id}:detail:${detail.id} <br/>
  * obj = Product.builder().id(1).detail(Detail.builder().id(2).build()).build(); <br/>
  * 經過解析返回 product:1:detail:2 <br/>
  *
  * @param content 要解析的內容
  * @param obj 填充解析內容的物件(如果是基本型別,則所有佔位符替換為相同的值)
  * @return
  */
 public String resolveByObject(String content,final Object obj) {
  if (obj instanceof Map) {
   return resolveByMap(content,(Map)obj);
  }
  return resolveByRule(content,placeholderValue -> String.valueOf(ReflectionUtils.getValueByFieldPath(obj,placeholderValue)));
 }
}

由於程式碼還有一些工具類的呼叫因為篇幅問題沒有貼,如有需要可以去 https://gitee.com/objs/mayfly 該專案中獲取

以上這篇java解析{{}}變數名以及文字內容替換操作就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支援我們。