Подтвердить что ты не робот

Различные регулярные выражения приводят к платформе Java SE и Android

У меня есть следующий код Java SE, который работает на ПК

public static void main(String[] args) {
    // stringCommaPattern will change
    // ","abc,def","
    // to
    // ","abcdef","        
    Pattern stringCommaPattern = Pattern.compile("(\",\")|,(?=[^\"[,]]*\",\")");
    String data = "\"SAN\",\"Banco Santander, \",\"NYSE\"";
    System.out.println(data);
    final String result = stringCommaPattern.matcher(data).replaceAll("$1");
    System.out.println(result);
}

Я получаю ожидаемый результат

"SAN","Banco Santander, ","NYSE"
"SAN","Banco Santander ","NYSE"

Однако, когда приходит Android.

Pattern stringCommaPattern = Pattern.compile("(\",\")|,(?=[^\"[,]]*\",\")");
String data = "\"SAN\",\"Banco Santander, \",\"NYSE\"";
Log.i("CHEOK", data);
final String result = stringCommaPattern.matcher(data).replaceAll("$1");
Log.i("CHEOK", result);

Я получаю

"SAN","Banco Santander, ","NYSE"
"SAN","Banco Santandernull ","NYSE"

Любое предложение и обходное решение, как я могу заставить этот код вести себя так же, как на Java SE?


Дополнительное примечание:

Другие шаблоны дают одинаковый результат. Похоже, что Android использует пустую строку для неповторимой группы, а Java SE использует пустую строку для неповторимой группы.

Возьмите следующий код.

public static void main(String[] args) {
    // Used to remove the comma within an integer digit. The digit must be located
    // in between two string. Replaced with $1.
    //
    // digitPattern will change
    // ",100,000,"
    // to
    // ",100000,"        
    final Pattern digitPattern = Pattern.compile("(\",)|,(?=[\\d,]+,\")");
    String data = "\",100,000,000,\"";
    System.out.println(data);
    final String result = digitPattern.matcher(data).replaceAll("$1");
    System.out.println(result);
}

Java SE

",100,000,000,"
",100000000,"

Android

",100,000,000,"
",100null000null000,"
4b9b3361

Ответ 1

Не причина, но в качестве обходного пути вы можете сделать цикл appendReplacement самостоятельно, а не использовать replaceAll

StringBuffer result = new StringBuffer();
Matcher m = digitPattern.matcher(data);
while(m.find()) {
  m.appendReplacement(result, (m.group(1) == null ? "" : "$1"));
}
m.appendTail(result);

Это должно работать как на JavaSE, так и на Android.

Или обойти проблему полностью, изменив регулярное выражение

Pattern commaNotBetweenQuotes = Pattern.compile("(?<!\"),(?!\")");
String result = commaNotBetweenQuotes.matcher(data).replaceAll("");

Здесь регулярное выражение соответствует только запятым, которые вы хотите изменить, а не тем, которые вы хотите оставить без изменений, поэтому вы можете просто заменить их на "", не требуя захвата групп.