Java 8查找并替换匹配的字符串

时间:2015-04-10 21:03:25

标签: java regex string java-8

我正在尝试在messages.properties中找到来自errorMessage的字符串,如果errorMessage有字符串我需要将其替换为相应的值

我有messages.properties如下

inv_date=INVOICE DATE
inv_id=INVOICE NUMBER
cl_id=CLIENT ID
lf_matter_id=LAW FIRM MATTER ID
inv_total_net_due=INVOICE TOTAL
(inv_start_date|invoice start date)=BILLING START DATE
(inv_end_date|invoice end date)=BILLING END DATE
inv_desc=INVOICE DESCRIPTION
units=LINE ITEM NUMBER OF UNITS
discount_amount=LINE ITEM ADJUSTMENT AMOUNT
total_amount=LINE ITEM TOTAL
(charge_date|charge date)= LINE ITEM DATE
acca_task=LINE ITEM TASK CODE
acca_expense=LINE ITEM EXPENSE CODE
acca_activity= LINE ITEM ACTIVITY CODE
(tk_id|time keeper id)=TIMEKEEPER ID
charge_desc=LINE ITEM DESCRIPTION
lf_id=LAW FIRM ID
(BaseRate|Rate|tk_rate)=LINE ITEM UNIT COST
tk_level=TIMEKEEPER CLASSIFICATION
cl_matter_id=CLIENT MATTER ID

我的errorMesage可以有任何(左侧)字符串,我需要用右侧字符串值替换它

以下是几个示例错误消息

String errorMesage1 = "Line 3 : Could not parse inv_date value" 
String errorMesage2 = "Line : 1 BaseRate is a required field"
下面的

是我的方法,它会回复错误信息

public static String toUserFriendlyErrorMessage(String message) {
    ResourceBundle rb = ResourceBundle.getBundle("messages");
    for(String key : rb.keySet()){
        String header = rb.getString(key);
        if(message.contains(key)) {
          return message.replaceAll(key, rb.getString(key));           
        }
    }
    return message;

}

以下是预期输出: 对于errorMessage1,它可以正常工作

System.out.println(toUserFriendlyErrorMessage(errorMessage1)); ==> Line 3 : Could not parse INVOICE DATE value 

但是对于errorMessage2,它无法正常工作。它不会将BaseRate替换为LINE ITEM UNIT COST

System.out.println(toUserFriendlyErrorMessage(errorMessage2)); ==> Line : 1 BaseRate is a required field

有没有办法找到多个字符串的出现并用相应的值替换它?

例如:查找(BaseRate|Rate|tk_rate)并将字符串替换为LINE ITEM UNIT COST

还想知道这种方法可以在java 8中进一步简化吗?

1 个答案:

答案 0 :(得分:4)

我认为你应该重新考虑你的设计并使用单独的密钥来获得几个“别名” - 或者甚至可能更好:根本没有别名,每个替换只有一个密钥。问题是这些属性文件中的键不应该包含空格 - 是否包含parantheses - 因此文件未正确解析。如果您打印密钥,您会看到它们在第一个空格处被截断,例如(inv_start_date|invoice start date)变为(inv_start_date|invoice

当然,这也意味着,即使您将这些“别名”拆分为单独的键,也不能拥有invoice start date之类的键,因为它仍然包含空格并且无法正确解析。

可以将这些替换放入Java源代码中的regualr Map中:

static Map<String, String> replacements = new HashMap<>();
static {
    replacements.put("inv_date", "INVOICE DATE");
    replacements.put("(BaseRate|Rate|tk_rate)", "LINE ITEM UNIT COST");
    // ... more stuff ...
}

或手动解析文件,将字符串拆分为=并将其放入Map

由于(BaseRate|Rate|tk_rate)之类的密钥实际上是有效的正则表达式,因此您可以使用replaceAll替换它们的所有变体。如果它们不包含在字符串中,replaceAll将不执行任何操作,因此contains检查不是必需的。

public static String toUserFriendlyErrorMessage(String message) {
    for (String key : replacements.keySet()) {
        message = message.replaceAll(key, replacements.get(key));
    }
    return message;
}

示例输出:

Line 3 : Could not parse INVOICE DATE value
Line : 1 LINE ITEM UNIT COST is a required field

或者,如果你想使用一些“Java 8 magic”,你可以使用reduce,但就个人而言,我认为循环更具可读性。

    return replacements.keySet().stream()
            .reduce(message, (s, k) -> s.replaceAll(k, replacements.get(k)))
            .toString();