java解析{{}}變量名以及文本內容替換操作
發(fā)短信、發(fā)郵件的時候經(jīng)常會遇到模板內容需要替換成實際數(shù)據(jù)的問題,本文介紹從文本模板中解析出變量列表,以及參數(shù)填入后得到實際文本內容的辦法:
/** * 根據(jù)正則表達式獲取文本中的變量名列表 * @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ù)正則表達式將文本中的變量使用實際的數(shù)據(jù)替換成無變量的文本 * @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]}到現(xiàn)場參加活動"; 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模板字符串優(yōu)雅解析(占位符解析)
項目中常常需要解析字符串模板,比如user:{userId}:{userType}用于redis的key等,比較常見的做法就是使用String.format(“user:%s:%s”, 1, 1)方法,但個人感覺那樣的模板不夠明了,而使用模板解析器可更好地有助于解析此類字符串。
可使用map用于解析,也可使用對象進行解析,也可使用類似String.format可變參數(shù)進行解析,多樣化解析對象值。有點類似于js的模板字符串${}。
也可自定義前綴后綴進行解析。如PlaceholderResolver.getResolver("{", “}”)該對象可解析{}該類型的占位符.
性能方面:
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", "2"}<br/> * 返回 category:1:product:2<br/> * * @param content 要解析的帶有占位符的模板字符串 * @param values 按照模板占位符索引位置設置對應的值 * @return */ public String resolve(String content, Object[] values) { return resolve(content, Stream.of(values).map(String::valueOf).toArray(String[]::new)); } /** * 根據(jù)替換規(guī)則來替換指定模板中的占位符值 * @param content 要解析的字符串 * @param rule 解析規(guī)則回調 * @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}值替換為替換規(guī)則回調中的內容 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(); } /** * 替換模板中占位符內容,占位符的內容即為map key對應的值,key為占位符中的內容。<br/><br/> * 如:content = product:${id}:detail:${did}<br/> * valueMap = id -> 1; pid -> 2<br/> * 經(jīng)過解析返回 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))); } /** * 根據(jù)properties文件替換占位符內容 * @param content * @param properties * @return */ public String resolveByProperties(String content, final Properties properties) { return resolveByRule(content, placeholderValue -> properties.getProperty(placeholderValue)); } /** * 根據(jù)對象中字段路徑(即類似js訪問對象屬性值)替換模板中的占位符 <br/><br/> * 如 content = product:${id}:detail:${detail.id} <br/> * obj = Product.builder().id(1).detail(Detail.builder().id(2).build()).build(); <br/> * 經(jīng)過解析返回 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解析{{}}變量名以及文本內容替換操作就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Java?多個文件生成zip包、下載zip包的實現(xiàn)代碼
這篇文章主要介紹了Java?多個文件生成zip包、下載zip包,包括文件上傳,文件下載,多個文件打成zip包的操作代碼,本文給大家介紹的非常詳細,需要的朋友可以參考下2024-01-01java工具類static靜態(tài)方法讀取yml配置過程
文章介紹了在工具類中獲取YAML配置時遇到的問題,由于變量是靜態(tài)的,而Spring加載靜態(tài)方法比IOC容器早,導致無法直接使用@Value注解讀取YAML配置,從而讀取結果為null2024-11-11Java使用modbus-master-tcp實現(xiàn)modbus tcp通訊
這篇文章主要為大家詳細介紹了另外一種Java語言的modbux tcp通訊方案,那就是modbus-master-tcp,文中的示例代碼講解詳細,需要的可以了解下2023-12-12RocketMQ producer同步發(fā)送單向發(fā)送源碼解析
這篇文章主要為大家介紹了RocketMQ producer同步發(fā)送單向發(fā)送源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03