Java escape-скобка

У меня этот маленький класс, чтобы сделать несколько замен на строку:

import java.util.HashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import org.apache.commons.lang.StringUtils;
public class MultipleReplace {
    public static void main(String[] args) {
        Map<String,String> tokens = new HashMap<String,String>();
        tokens.put(":asd:", "<img src=asd.gif>");
        tokens.put(":)", "<img src=sorriso.gif>");
        String template = ":asd: bravo! :)";
        String patternString = "(" + StringUtils.join(tokens.keySet(), "|") + ")";
        Pattern pattern = Pattern.compile(patternString);
        Matcher matcher = pattern.matcher(template);
        StringBuffer sb = new StringBuffer();
        while(matcher.find()) {
            matcher.appendReplacement(sb, tokens.get(matcher.group(1)));
        }
        matcher.appendTail(sb);

        System.out.println(sb.toString());
    }
}

Проблема заключается во второй замене, где у меня есть скобка, которая приводит к:

Исключение в потоке "main" java.util.regex.PatternSyntaxException: Непревзойденное закрытие ')' рядом с индексом 8 (:) |: ASD:)

Как я могу избежать скобок? Или вы можете предложить альтернативу для замены этого числа?

Большое спасибо и извините за мой английский:)

EDIT:

Скрытие с обратной косой чертой ')' тоже не работает, оно не будет компилироваться:

"Неверная escape-последовательность (допустимые -\b\t\n\f\r \" \'\) "

НОВЫЙ РЕДАКТИРОВАТЬ

с использованием двух компромиссов обратной смены, но не делает замены.

ПОСЛЕДНИЙ РЕДАКТИРОВАТЬ

Наконец, нашел решение, используя Pattern.quote при построении шаблона. Необходимо использовать итератор для выполнения цикла.

Здесь правильный код:

package string;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class MultipleReplace {
    @SuppressWarnings("rawtypes")
    public static void main(String[] args) {
        Map<String,String> tokens = new HashMap<String,String>();
        tokens.put(":asd:", "<img src=asd.gif>");
        tokens.put(":)", "<img src=sorriso.gif>");
        String template = ":asd: bravo! :)";
        Iterator it = tokens.entrySet().iterator();
        String patternString = "(";
        while (it.hasNext()) {
            Map.Entry pairs = (Map.Entry)it.next();
            System.out.println(pairs.getKey() + " = " + pairs.getValue());
            patternString = patternString +Pattern.quote((String) pairs.getKey());
            if (it.hasNext())
            {
                patternString = patternString + "|";
            }
        }
        patternString = patternString + ")";
        System.out.println(patternString);
        Pattern pattern = Pattern.compile(patternString);
        Matcher matcher = pattern.matcher(template);
        StringBuffer sb = new StringBuffer();
        while(matcher.find()) {
            matcher.appendReplacement(sb, tokens.get(matcher.group(1)));
        }
        matcher.appendTail(sb);
        System.out.println(sb.toString());
    }
}

Прокомментируйте это, если я могу улучшить работу! Большое вам спасибо!

Ответ 1

Используйте Pattern.quote, как я писал в комментарии. Он работает с каждой строкой и для длинных строк, содержащих много не буквенно-цифровых символов, менее подвержен ошибкам.

Ответ 2

Используйте обратную косую черту: \). Parens должны быть экранированы, потому что их можно использовать для группировки частей регулярного выражения.

    String template = ":asd: bravo\\! :\\)";

Ответ 3

Я приведу лучший пример, чем Тим Н.

Скажем, у вас есть строковое слово.

word = "http://www.randomwebsite.com/images/That Image (English)";

Если вы хотите заменить скобки пробелами, просто выполните:

word.replaceAll("\\(", " ");

Вам нужно сделать 2 двойных слэша, чтобы заставить компилятор заткнуться и не скулить.

Также имейте в виду, что функция возвращает строку. Таким образом, вы сделали бы

word = word.replaceAll("\\(", " ");

Если вы не хотите его видеть, просто распечатайте его.