java解析{{}}變量名以及文本內(nè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編程實(shí)現(xiàn)軌跡壓縮算法開(kāi)放窗口實(shí)例代碼
這篇文章主要介紹了Java編程實(shí)現(xiàn)軌跡壓縮算法開(kāi)放窗口實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-11-11Java?多個(gè)文件生成zip包、下載zip包的實(shí)現(xiàn)代碼
這篇文章主要介紹了Java?多個(gè)文件生成zip包、下載zip包,包括文件上傳,文件下載,多個(gè)文件打成zip包的操作代碼,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2024-01-01java工具類(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é)果為null2024-11-11Java使用modbus-master-tcp實(shí)現(xiàn)modbus tcp通訊
這篇文章主要為大家詳細(xì)介紹了另外一種Java語(yǔ)言的modbux tcp通訊方案,那就是modbus-master-tcp,文中的示例代碼講解詳細(xì),需要的可以了解下2023-12-12Java 敏感詞檢測(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-11Idea安裝及涉及springboot詳細(xì)配置的圖文教程
這篇文章主要介紹了Idea安裝及涉及springboot詳細(xì)配置,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10RocketMQ producer同步發(fā)送單向發(fā)送源碼解析
這篇文章主要為大家介紹了RocketMQ producer同步發(fā)送單向發(fā)送源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03