Як отримати рядок між двома символами?


93

У мене є рядок,

String s = "test string (67)";

Я хочу отримати номер 67, який є рядком між (і).

Хто-небудь може підказати мені, як це зробити?


1
Є кілька способів - можна перебирати на символи в рядку , поки ви не досягнете (або знайти індекс першого (і )і зробити це з підрядка або, що більшість людей буде робити, використовувати регулярний вираз.
Andreas Dolk,

Відповіді:


102

Напевно, є справді акуратний RegExp, але я не в цій області, тому замість цього ...

String s = "test string (67)";

s = s.substring(s.indexOf("(") + 1);
s = s.substring(0, s.indexOf(")"));

System.out.println(s);

4
Не розглядаючи дивацтва синтаксичного аналізу регулярних виразів, я вважаю, що це найкращий спосіб витягти необхідний рядок.
правдоподібність

3
регулярний вираз набагато потужніший і може зайняти більше випадків, але для простоти це працює ...
MadProgrammer

2
Серйозно, чому це залучить голос проти? Це не працює? Чи не відповідає він на операційне запитання?
MadProgrammer

якщо у мене є кілька значень, то як я можу використовувати підрядок, вважаю, що я маю такий рядок, 'this is an example of <how><i have it>'і мені потрібно знайти значення між '<' та '>' this
Vignesh,

@Vignesh Використовуйте регулярний вираз
MadProgrammer

74

Дуже корисним рішенням цього питання, яке не вимагає від вас робити indexOf, є використання бібліотек Apache Commons .

 StringUtils.substringBetween(s, "(", ")");

Цей метод дозволить вам навіть обробляти, навіть якщо є кілька випадків закриття рядка, що нелегко буде здійснити, шукаючи indexOf закривального рядка.

Ви можете завантажити цю бібліотеку тут: https://mvnrepository.com/artifact/org.apache.commons/commons-lang3/3.4


7
Є також, substringsBetween(...)якщо ви очікуєте декількох результатів, саме цього я і шукав. Дякую
cahen


72

Спробуйте так

String s="test string(67)";
String requiredString = s.substring(s.indexOf("(") + 1, s.indexOf(")"));

Підпис методу для підрядка:

s.substring(int start, int end);

30

Використовуючи регулярний вираз:

 String s = "test string (67)";
 Pattern p = Pattern.compile("\\(.*?\\)");
 Matcher m = p.matcher(s);
 if(m.find())
    System.out.println(m.group().subSequence(1, m.group().length()-1)); 

2
Я думаю, що вам слід зробити цей ненажерливий матч, використовуючи ". *?" замість цього. В іншому випадку, якщо рядок схожий на "тестовий рядок (67) та (68), це поверне" 67) та (68 ".
Chthonic Project,

18

Java підтримує регулярні вирази , але вони начебто громіздкі, якщо ви насправді хочете використовувати їх для вилучення збігів. Я думаю, що найпростіший спосіб отримати той рядок, який потрібно у вашому прикладі, - це просто використати підтримку регулярного виразу в методі Stringкласу replaceAll:

String x = "test string (67)".replaceAll(".*\\(|\\).*", "");
// x is now the String "67"

Це просто видаляє все, включаючи перше (, і те саме, що стосується )і всього подальшого. Це просто залишає речі між дужками.

Однак результат цього все ще є String. Якщо замість цього вам потрібен цілочисельний результат, вам потрібно виконати ще одне перетворення:

int n = Integer.parseInt(x);
// n is now the integer 67

10

В одному рядку я пропоную:

String input = "test string (67)";
input = input.subString(input.indexOf("(")+1, input.lastIndexOf(")"));
System.out.println(input);`

7
String s = "test string (67)";

int start = 0; // '(' position in string
int end = 0; // ')' position in string
for(int i = 0; i < s.length(); i++) { 
    if(s.charAt(i) == '(') // Looking for '(' position in string
       start = i;
    else if(s.charAt(i) == ')') // Looking for ')' position in  string
       end = i;
}
String number = s.substring(start+1, end); // you take value between start and end

7

Для цього ви можете використовувати StringUtils загальної бібліотеки apache.

import org.apache.commons.lang3.StringUtils;
...
String s = "test string (67)";
s = StringUtils.substringBetween(s, "(", ")");
....

7
String result = s.substring(s.indexOf("(") + 1, s.indexOf(")"));

1
Будь ласка, відформатуйте код, відступивши 4 пробіли. Також я б трохи .substringдоповнив вашу відповідь, пояснивши, що робить ваш код для тих, хто відвідує, хто не впевнений у тому, що робить .indexOf`.
Багс

6

Тестовий рядок, test string (67)з якого потрібно отримати рядок, вкладений між двома рядками.

String str = "test string (67) and (77)", open = "(", close = ")";

Перелічено кілька можливих способів : Просте загальне рішення:

String subStr = str.substring(str.indexOf( open ) + 1, str.indexOf( close ));
System.out.format("String[%s] Parsed IntValue[%d]\n", subStr, Integer.parseInt( subStr ));

Фонд програмного забезпечення Apache commons.lang3.

StringUtilssubstringBetween()Функція class отримує рядок, вкладений між двома рядками. Повертається лише перший матч.

String substringBetween = StringUtils.substringBetween(subStr, open, close);
System.out.println("Commons Lang3 : "+ substringBetween);

Замінює заданий рядок на рядок, вкладений між двома рядками. #395


Візерунок із регулярними виразами: (\()(.*?)(\)).*

У Dot матчі (Майже) Будь-який символ .? = .{0,1}, .* = .{0,}, .+ = .{1,}

String patternMatch = patternMatch(generateRegex(open, close), str);
System.out.println("Regular expression Value : "+ patternMatch);

Regular-Expression з класом корисності RegexUtilsта деякими функціями.
      Pattern.DOTALL: Відповідає будь-якому символу, включаючи закінчувач рядка.
      Pattern.MULTILINE: Відповідає цілому рядку від початку ^до кінця $вхідної послідовності.

public static String generateRegex(String open, String close) {
    return "(" + RegexUtils.escapeQuotes(open) + ")(.*?)(" + RegexUtils.escapeQuotes(close) + ").*";
}

public static String patternMatch(String regex, CharSequence string) {
    final Pattern pattern  = Pattern.compile(regex, Pattern.DOTALL);
    final Matcher matcher = pattern .matcher(string);

    String returnGroupValue = null;
    if (matcher.find()) { // while() { Pattern.MULTILINE }
        System.out.println("Full match: " + matcher.group(0));
        System.out.format("Character Index [Start:End]«[%d:%d]\n",matcher.start(),matcher.end());
        for (int i = 1; i <= matcher.groupCount(); i++) {
            System.out.println("Group " + i + ": " + matcher.group(i));
            if( i == 2 ) returnGroupValue = matcher.group( 2 );
        }
    }
    return returnGroupValue;
}

StringUtils дуже корисний
TuGordoBello

5
public String getStringBetweenTwoChars(String input, String startChar, String endChar) {
    try {
        int start = input.indexOf(startChar);
        if (start != -1) {
            int end = input.indexOf(endChar, start + startChar.length());
            if (end != -1) {
                return input.substring(start + startChar.length(), end);
            }
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
    return input; // return null; || return "" ;
}

Використання:

String input = "test string (67)";
String startChar = "(";
String endChar   = ")";
String output = getStringBetweenTwoChars(input, startChar, endChar);
System.out.println(output);
// Output: "67"

4

Використовуйте Pattern and Matcher

public class Chk {

    public static void main(String[] args) {

        String s = "test string (67)";
        ArrayList<String> arL = new ArrayList<String>();
        ArrayList<String> inL = new ArrayList<String>();

        Pattern pat = Pattern.compile("\\(\\w+\\)");
        Matcher mat = pat.matcher(s);

        while (mat.find()) {

            arL.add(mat.group());
            System.out.println(mat.group());

        }

        for (String sx : arL) {

            Pattern p = Pattern.compile("(\\w+)");
            Matcher m = p.matcher(sx);

            while (m.find()) {

                inL.add(m.group());
                System.out.println(m.group());
            }
        }

        System.out.println(inL);

    }

}

2
Вимовляння імен змінних може зробити метод більш дружнім.
Zon

3

Інший спосіб використання методу спліт

public static void main(String[] args) {


    String s = "test string (67)";
    String[] ss;
    ss= s.split("\\(");
    ss = ss[1].split("\\)");

    System.out.println(ss[0]);
}

3

Найменш загальний спосіб, який я знайшов, щоб зробити це за допомогою Regex та Pattern / Matcher класів:

String text = "test string (67)";

String START = "\\(";  // A literal "(" character in regex
String END   = "\\)";  // A literal ")" character in regex

// Captures the word(s) between the above two character(s)
String pattern = START + "(\w+)" + END;

Pattern pattern = Pattern.compile(pattern);
Matcher matcher = pattern.matcher(text);

while(matcher.find()) {
    System.out.println(matcher.group()
        .replace(START, "").replace(END, ""));
}

Це може допомогти для більш складних проблем регулярного виразу, коли ви хочете отримати текст між двома наборами символів.


2

"Загальний" спосіб цього полягає в синтаксичному аналізі рядка з самого початку, викиданні всіх символів перед першою дужкою, запису символів після першої дужки та викиданні символів після другої дужки.

Я впевнений, що існує бібліотека регулярних виразів або щось інше, щоб це зробити.


Java підтримує регулярні вирази. Не потрібно бібліотеки regexp4j;)
Андреас Долк,

2
String s = "test string (67)";

System.out.println(s.substring(s.indexOf("(")+1,s.indexOf(")")));

2

Іншим можливим рішенням є використання lastIndexOfтам, де він буде шукати символ або рядок назад.

У моєму сценарії я мав наступні дії, Stringі мені довелося витягувати<<UserName>>

1QAJK-WKJSH_MyApplication_Extract_<<UserName>>.arc

Отже, indexOfі StringUtils.substringBetweenне було корисно, оскільки вони починають шукати персонажа з самого початку.

Отже, я використовував lastIndexOf

String str = "1QAJK-WKJSH_MyApplication_Extract_<<UserName>>.arc";
String userName = str.substring(str.lastIndexOf("_") + 1, str.lastIndexOf("."));

І це дає мені

<<UserName>>

1

Щось на зразок цього:

public static String innerSubString(String txt, char prefix, char suffix) {

    if(txt != null && txt.length() > 1) {

        int start = 0, end = 0;
        char token;
        for(int i = 0; i < txt.length(); i++) {
            token = txt.charAt(i);
            if(token == prefix)
                start = i;
            else if(token == suffix)
                end = i;
        }

        if(start + 1 < end)
            return txt.substring(start+1, end);

    }

    return null;
}

1

Це просте використання \D+регулярного виразу та виконана робота.
Буде вибрано всі символи, крім цифр, не потрібно ускладнювати

/\D+/

1

він поверне початковий рядок, якщо не буде збігатися регулярний вираз

var iAm67 = "test string (67)".replaceFirst("test string \\((.*)\\)", "$1");

додати збіги до коду

String str = "test string (67)";
String regx = "test string \\((.*)\\)";
if (str.matches(regx)) {
    var iAm67 = str.replaceFirst(regx, "$1");
}

--- РЕДАКТУВАТИ ---

я використовую https://www.freeformatter.com/java-regex-tester.html#ad-output для тестування регулярних виразів.

виявляється краще додати? після * для меншої відповідності. щось на зразок цього:

String str = "test string (67)(69)";
String regx1 = "test string \\((.*)\\).*";
String regx2 = "test string \\((.*?)\\).*";
String ans1 = str.replaceFirst(regx1, "$1");
String ans2 = str.replaceFirst(regx2, "$1");
System.out.println("ans1:"+ans1+"\nans2:"+ans2); 
// ans1:67)(69
// ans2:67
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.