欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java解析{{}}變量名以及文本內(nèi)容替換操作

 更新時(shí)間:2020年09月23日 14:39:59   作者:盛桃云  
這篇文章主要介紹了java解析{{}}變量名以及文本內(nèi)容替換操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧

發(fā)短信、發(fā)郵件的時(shí)候經(jīng)常會(huì)遇到模板內(nèi)容需要替換成實(shí)際數(shù)據(jù)的問(wèn)題,本文介紹從文本模板中解析出變量列表,以及參數(shù)填入后得到實(shí)際文本內(nèi)容的辦法:

/**
 * 根據(jù)正則表達(dá)式獲取文本中的變量名列表
 * @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;
}
 
/**
 * 根據(jù)正則表達(dá)式將文本中的變量使用實(shí)際的數(shù)據(jù)替換成無(wú)變量的文本
 * @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 = "恭喜{{姓名}}報(bào)名成功,請(qǐng)憑報(bào)名編號(hào){[code]}到現(xiàn)場(chǎng)參加活動(dòng)";
 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);
}

上面的代碼介紹的是針對(duì){{}}形式的變量值的解析辦法,大家可以修改一下正則表達(dá)式,改成可以解析${}變量的辦法

補(bǔ)充知識(shí):java模板字符串優(yōu)雅解析(占位符解析)

項(xiàng)目中常常需要解析字符串模板,比如user:{userId}:{userType}用于redis的key等,比較常見(jiàn)的做法就是使用String.format(“user:%s:%s”, 1, 1)方法,但個(gè)人感覺(jué)那樣的模板不夠明了,而使用模板解析器可更好地有助于解析此類(lèi)字符串。

可使用map用于解析,也可使用對(duì)象進(jìn)行解析,也可使用類(lèi)似String.format可變參數(shù)進(jìn)行解析,多樣化解析對(duì)象值。有點(diǎn)類(lèi)似于js的模板字符串${}。

也可自定義前綴后綴進(jìn)行解析。如PlaceholderResolver.getResolver("{", “}”)該對(duì)象可解析{}該類(lèi)型的占位符.

性能方面:

PlaceholderResolve解析結(jié)果

String.format解析結(jié)果

解析100w次,占位符解析比String.format平均都會(huì)快個(gè)0.4~0.5s,so 性能應(yīng)該問(wèn)題不大。最主要的是不需要調(diào)用太多的對(duì)象方法,自動(dòng)解析,方便快捷。

源代碼如下:

/**
 * 占位符解析器
 *
 * @author meilin.huang
 * @version 1.0
 * @date 2018-11-13 1:42 PM
 */
public class PlaceholderResolver {
 /**
  * 默認(rèn)前綴占位符
  */
 public static final String DEFAULT_PLACEHOLDER_PREFIX = "${";

 /**
  * 默認(rèn)后綴占位符
  */
 public static final String DEFAULT_PLACEHOLDER_SUFFIX = "}";

 /**
  * 默認(rèn)單例解析器
  */
 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;
 }

 /**
  * 獲取默認(rèn)的占位符解析器,即占位符前綴為"${", 后綴為"}"
  * @return
  */
 public static PlaceholderResolver getDefaultResolver() {
  return defaultResolver;
 }

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

 /**
  * 解析帶有指定占位符的模板字符串,默認(rèn)占位符為前綴:${ 后綴:}<br/><br/>
  * 如:template = category:${}:product:${}<br/>
  * values = {"1", "2"}<br/>
  * 返回 category:1:product:2<br/>
  *
  * @param content 要解析的帶有占位符的模板字符串
  * @param values 按照模板占位符索引位置設(shè)置對(duì)應(yīng)的值
  * @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();
 }

 /**
  * 解析帶有指定占位符的模板字符串,默認(rèn)占位符為前綴:${ 后綴:}<br/><br/>
  * 如:template = category:${}:product:${}<br/>
  * values = {"1", "2"}<br/>
  * 返回 category:1:product:2<br/>
  *
  * @param content 要解析的帶有占位符的模板字符串
  * @param values 按照模板占位符索引位置設(shè)置對(duì)應(yīng)的值
  * @return
  */
 public String resolve(String content, Object[] values) {
  return resolve(content, Stream.of(values).map(String::valueOf).toArray(String[]::new));
 }

 /**
  * 根據(jù)替換規(guī)則來(lái)替換指定模板中的占位符值
  * @param content 要解析的字符串
  * @param rule 解析規(guī)則回調(diào)
  * @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);
   //替換整個(gè)占位符內(nèi)容,即將${id}值替換為替換規(guī)則回調(diào)中的內(nèi)容
   String replaceContent = placeholder.trim().isEmpty() ? "" : rule.apply(placeholder);
   result.replace(start, end + this.placeholderSuffix.length(), replaceContent);
   start = result.indexOf(this.placeholderPrefix, start + replaceContent.length());
  }
  return result.toString();
 }

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

 /**
  * 根據(jù)properties文件替換占位符內(nèi)容
  * @param content
  * @param properties
  * @return
  */
 public String resolveByProperties(String content, final Properties properties) {
  return resolveByRule(content, placeholderValue -> properties.getProperty(placeholderValue));
 }

 /**
  * 根據(jù)對(duì)象中字段路徑(即類(lèi)似js訪問(wèn)對(duì)象屬性值)替換模板中的占位符 <br/><br/>
  * 如 content = product:${id}:detail:${detail.id} <br/>
  * obj = Product.builder().id(1).detail(Detail.builder().id(2).build()).build(); <br/>
  * 經(jīng)過(guò)解析返回 product:1:detail:2 <br/>
  *
  * @param content 要解析的內(nèi)容
  * @param obj 填充解析內(nèi)容的對(duì)象(如果是基本類(lèi)型,則所有占位符替換為相同的值)
  * @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)));
 }
}

由于代碼還有一些工具類(lèi)的調(diào)用因?yàn)槠鶈?wèn)題沒(méi)有貼,如有需要可以去 https://gitee.com/objs/mayfly 該項(xiàng)目中獲取

以上這篇java解析{{}}變量名以及文本內(nèi)容替換操作就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java中的TreeSet源碼解讀

    Java中的TreeSet源碼解讀

    這篇文章主要介紹了Java中的TreeSet源碼解讀,TreeSet 是一個(gè) 有序集合,它擴(kuò)展了 AbstractSet 類(lèi)并實(shí)現(xiàn)了 NavigableSet 接口,對(duì)象根據(jù)其自然順序以升序排序和存儲(chǔ),該 TreeSet 中使用 平衡樹(shù),更具體的一個(gè) 紅黑樹(shù),需要的朋友可以參考下
    2023-09-09
  • 推薦史上最全的IDEA好用插件

    推薦史上最全的IDEA好用插件

    學(xué)會(huì)這些Idea插件敲代碼也是一種享受,分享史上最全的IDEA好用插件,可以幫助我們提高工作開(kāi)發(fā)效率,對(duì)于開(kāi)發(fā)人員很有幫助,感興趣的朋友開(kāi)來(lái)看看吧
    2021-03-03
  • Java編程實(shí)現(xiàn)軌跡壓縮算法開(kāi)放窗口實(shí)例代碼

    Java編程實(shí)現(xiàn)軌跡壓縮算法開(kāi)放窗口實(shí)例代碼

    這篇文章主要介紹了Java編程實(shí)現(xiàn)軌跡壓縮算法開(kāi)放窗口實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下。
    2017-11-11
  • Java?多個(gè)文件生成zip包、下載zip包的實(shí)現(xiàn)代碼

    Java?多個(gè)文件生成zip包、下載zip包的實(shí)現(xiàn)代碼

    這篇文章主要介紹了Java?多個(gè)文件生成zip包、下載zip包,包括文件上傳,文件下載,多個(gè)文件打成zip包的操作代碼,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2024-01-01
  • java工具類(lèi)static靜態(tài)方法讀取yml配置過(guò)程

    java工具類(lèi)static靜態(tài)方法讀取yml配置過(guò)程

    文章介紹了在工具類(lèi)中獲取YAML配置時(shí)遇到的問(wèn)題,由于變量是靜態(tài)的,而Spring加載靜態(tài)方法比IOC容器早,導(dǎo)致無(wú)法直接使用@Value注解讀取YAML配置,從而讀取結(jié)果為null
    2024-11-11
  • Java使用modbus-master-tcp實(shí)現(xiàn)modbus tcp通訊

    Java使用modbus-master-tcp實(shí)現(xiàn)modbus tcp通訊

    這篇文章主要為大家詳細(xì)介紹了另外一種Java語(yǔ)言的modbux tcp通訊方案,那就是modbus-master-tcp,文中的示例代碼講解詳細(xì),需要的可以了解下
    2023-12-12
  • servlet實(shí)現(xiàn)用戶登錄小程序

    servlet實(shí)現(xiàn)用戶登錄小程序

    這篇文章主要為大家詳細(xì)介紹了servlet實(shí)現(xiàn)用戶登錄的小程序,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • Java 敏感詞檢測(cè)工具的實(shí)現(xiàn)

    Java 敏感詞檢測(cè)工具的實(shí)現(xiàn)

    本文介紹了Java敏感詞檢測(cè)工具的使用方法,包括依賴引入、核心方法使用實(shí)例、常規(guī)用法、自定義替換檢測(cè)策略等內(nèi)容,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-11-11
  • Idea安裝及涉及springboot詳細(xì)配置的圖文教程

    Idea安裝及涉及springboot詳細(xì)配置的圖文教程

    這篇文章主要介紹了Idea安裝及涉及springboot詳細(xì)配置,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-10-10
  • RocketMQ producer同步發(fā)送單向發(fā)送源碼解析

    RocketMQ producer同步發(fā)送單向發(fā)送源碼解析

    這篇文章主要為大家介紹了RocketMQ producer同步發(fā)送單向發(fā)送源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03

最新評(píng)論