前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java 一步一步实现高逼格的字符串替换工具(一)

Java 一步一步实现高逼格的字符串替换工具(一)

作者头像
一灰灰blog
发布2018-02-06 14:43:33
5.3K1
发布2018-02-06 14:43:33
举报
文章被收录于专栏:小灰灰小灰灰

Java 一步一步实现高逼格的字符串替换工具(一)

如果你有一段模板, 需要用某些数据替换其中的关键信息,怎么做 "hello, {user}, welcome to {place}!" 通过传不同的user, 和 place 来输出不同的文案

##1.一般做法

用String.replaceAll 来进行替换就好了, 无非是多调用几遍,代码写起来也简单,如下

代码语言:javascript
复制
@Test
public void testReplace() {
   String text = "hello {user}, welcome to {place}!";
   String user = "Lucy";
   String place = "China";

   String res = text.replace("{user}", user).replace("{place}", place);
   System.out.println(res);  // 输出   hello Lucy, welcome to China!
}

上面看着也没什么问题,实现起来也不难,实际呢 ? 如果我想要一个通用的替换方法, 如下面的接口定义, 约定text中用大括号包起来的由后面的参数进行替换

2. 通用的工具怎么玩

要求实现下面这个接口,text为需要被替换的字符串, 用后面的参数来替换text中用 {} 包含起来的内容

代码语言:javascript
复制
public String replace(String text, String ... args);

这时,该怎么用上面的方法来实现替换呢 ?

如果有了解过 MessageFormat 的同学,估计很快就能想到,这个工具就是jdk提供给我们来实现文本格式化的利器,那么简单的实现如下

代码语言:javascript
复制
public String replace(String text, Object ... args) {
   return MessageFormat.format(text, args);
}


@Test
public void testReplace2() {
   String text = "hello {0}, welcome to {1}!";
   String user = "Lucy";
   String place = "China";

   String ans = replace(text, user, place);
   System.out.println(ans); // 输出   hello Lucy, welcome to China!
}

仔细瞅瞅,实现了我们的部分需求,但是还不完美,上面的实现要求{}中的是后面参数再参数列表中的下标,而我们希望直接在 {} 中填写参数名, 直接用后面的参数名来替换, 这个时候可以怎么处理 ?

3. 进阶

要实现也简单,我自己先用正则把你的参数捞出来,然后替换成下标数字就可以了,麻烦的无非是如何写正则, 如何获取参数名罢了,正则还好讲,参数名的话如果不想用反射,那么直接改造下 传参的方式即可,丢一个map进去就完美了

代码语言:javascript
复制
// 获取patter的过程较为负责,这里初始化时,做一次即可
    private static Pattern pattern;

    static {
        pattern = Pattern.compile("((?<=\\{)([a-zA-Z_]{1,})(?=\\}))");
    }

    public String replaceV2(String text, Map<String, Object> map) {
        List<String> keys = new ArrayList<>();

        // 把文本中的所有需要替换的变量捞出来, 丢进keys
        Matcher matcher = pattern.matcher(text);
        while (matcher.find()) {
            String key = matcher.group();
            if (!keys.contains(key)) {
                keys.add(key);
            }
        }

        // 开始替换, 将变量替换成数字,  并从map中将对应的值丢入 params 数组
        Object[] params = new Object[keys.size()];
        for (int i = 0; i < keys.size(); i++) {
            text = text.replaceAll(keys.get(i), i + "");
            params[i] = map.get(keys.get(i));
        }


        return replace(text, params);
    }


    @Test
    public void testReplaceV2() {
        String text = "hello {user}, welcome to {place}! {place} is very beautiful ";

        Map<String, Object> map = new HashMap<>(2);
        map.put("user", "Lucy");
        map.put("place", "China");

        String res = replaceV2(text, map);
        System.out.println(res);  // hello Lucy, welcome to China! China is very beautiful
    }

这下是不是就完美的实现你的需求?

上面的实现,功能是满足了,但是又是正则,又是替换,又是 调用MessageFormat.format, 这么多步骤,这不是我想要的结果,干嘛不直接再 MessageFormat.format 中就把功能实现了,作为一个有追求的人,怎么能容忍这种曲线救国!!!(讲道理,我是个完全没追求的人

先捋一把MessageFormat的实现源码,然后发现上面有个坑,当被替换的是Long型数据时,输出有点鬼畜

代码语言:javascript
复制
    @Test
    public void testReplaceV2() {
        String text = "hello {user}, welcome to {place}! now timestamp is: {time} !";

        Map<String, Object> map = new HashMap<>(2);
        map.put("user", "Lucy");
        map.put("place", "China");
        map.put("time", System.currentTimeMillis());

        String res = replaceV2(text, map);
        System.out.println(res); // 输出 : hello Lucy, welcome to China! now 2stamp is: 1,490,619,291,742 !
    }

根本原因替换时, 对数字进行了格式化,没三个加一个,解决方法也比较简单,不传数字就可以了(就是这么粗暴)

更新后的代码

代码语言:javascript
复制
 public String replaceV2(String text, Map<String, Object> map) {
        List<String> keys = new ArrayList<>();

        // 把文本中的所有需要替换的变量捞出来, 丢进keys
        Matcher matcher = pattern.matcher(text);
        while (matcher.find()) {
            String key = matcher.group();
            if (!keys.contains(key)) {
                keys.add(key);
            }
        }

        // 开始替换, 将变量替换成数字,  并从map中将对应的值丢入 params 数组
        Object[] params = new Object[keys.size()];
        for (int i = 0; i < keys.size(); i++) {
            text = text.replaceAll(keys.get(i), i + "");
            params[i] = map.get(keys.get(i) + "");
        }


        return replace(text, params);
    }

如果你硬是要扣细节,要实现第二节里面定义的格式,不想传map,这个时候可以怎么玩?

--- 我也不知道怎么玩... 用反射后去的参数名是定义的参数名,如果你的接口定义的是可变参数,实际使用的时候就是一个数组了,这个时候想获取实际传入的参数名就无能为力了


并不完美,在正则获取结果之后,直接替换结果就好了,干嘛还要重复多次一举!!!,下面这样不也可以实现要求么

代码语言:javascript
复制
 public String replaceV3(String text, Map<String, Object> map) {
        Matcher matcher = pattern.matcher(text);
        while (matcher.find()) {
            String key = matcher.group();
            text = text.replaceAll("\\{" + key + "\\}", map.get(key) + "");
        }

        return text;
    }


    @Test
    public void testReplaceV3() {
        String text = "hello {user}, welcome to {place}! {place} is a beautiful place!";

        Map<String, Object> map = new HashMap<>(2);
        map.put("user", "Lucy");
        map.put("place", "China");

        String res = replaceV3(text, map);
        System.out.println(res); // hello Lucy, welcome to China! China is a beautiful place!
    }

上面这种看起来比起前面的正则,捞出来后又去调用 MessageFormat.format 要好多了, 但是也有点问题

  • 替换的数据量大时, replaceAll 的性能不咋的
  • 如果是对一个模板进行批量替换时,改怎么做?

对于批量替换,显然采用前面的方案实现起来简单且高效多了, 简单的改造下即可

代码语言:javascript
复制
public List<String> replaceV4(String text, List<Map<String, Object>> mapList) {
        List<String> keys = new ArrayList<>();

        // 把文本中的所有需要替换的变量捞出来, 丢进keys
        Matcher matcher = pattern.matcher(text);
        int index = 0;
        while (matcher.find()) {
            String key = matcher.group();
            if (!keys.contains(key)) {
                keys.add(key);
                // 开始替换, 将变量替换成数字, 
                text = text.replaceAll(keys.get(index), index + "");
                index ++;
            }
        }

        
        List<String> result = new ArrayList<>();
        //  从map中将对应的值丢入 params 数组
        Object[] params = new Object[keys.size()];
        for (Map<String, Object> map: mapList) {
            for (int i = 0; i < keys.size(); i++) {
                params[i] = map.get(keys.get(i) + "");
            }
            
            result.add(replace(text, params));
        }

        return result;
    }

4. 进阶++

对于上面的实现还是不满意,要求既高效、还可以选择并发替换、还能支持批量

需求会越来越高级,想一想该怎么实现上面的需求呢!

详情静待下一篇,主要是借鉴 MessageFormat的实现原理, 想实现这样的功能当然是自己动手写才是真理

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Java 一步一步实现高逼格的字符串替换工具(一)
    • 2. 通用的工具怎么玩
      • 3. 进阶
        • 4. 进阶++
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档