Znajdź i wyodrębnij liczbę z ciągu


319

Mam wymaganie, aby znaleźć i wyodrębnić liczbę zawartą w ciągu.

Na przykład z tych ciągów:

string test = "1 test"
string test1 = " 1 test"
string test2 = "test 99"

W jaki sposób mogę to zrobić?


2
Czy liczba może być ujemna? Jeśli tak, w jaki sposób poradziłbyś sobie z „Cześć - jak się masz? -30”?
Jon Skeet,

Cześć John, Brak liczb ujemnych w danych
van

4
Liczby dziesiętne jak 1.5? Notacja wykładnicza jak 1.5E45?
Tim Pietzcker,

Podobne (ale nie identyczne): stackoverflow.com/questions/1561273/...
finnw

3
Dlaczego nie akceptuje się tutaj odpowiedzi?
Wiktor Stribiżew

Odpowiedzi:


63

przejdź przez ciąg i użyj Char.IsDigit

string a = "str123";
string b = string.Empty;
int val;

for (int i=0; i< a.Length; i++)
{
    if (Char.IsDigit(a[i]))
        b += a[i];
}

if (b.Length>0)
    val = int.Parse(b);

15
@ Thomas: Ten kod nie działa, wynika z b == "System.Linq.Enumerable..". Prawidłowe (a nawet prostsze) byłobyb = String.Join("", a.Where(char.IsDigit))
BlueRaja - Danny Pflughoeft

1
Dobrze, że nauczy mnie to nie testować kodu, który piszę w komentarzu! Możesz również utworzyć ciąg z tablicy char za pomocą new string(char[])konstruktora.
Thomas

1
Regex wykonuje znacznie lepszą pracę.
Jason Kelley,

@BlueRaja - Danny Pflughoeft Dlaczego nie poprawisz komentarza, abym mógł głosować :-)
SteveC

UWAGA: Jeśli ciąg znaków zawiera wiele liczb, ta odpowiedź spowoduje, że wszystkie zostaną połączone razem w jedną liczbę. Np. „A12bcd345” powoduje „12345”. (Które może być pożądane lub nie, w zależności od celu.) Jest to inne niż najlepiej głosowane rozwiązanie Regex, które w przypadku powyżej zwróci „12”. Ma to znaczenie w przypadkach takich jak numery telefonów „555-111-2222”.
ToolmakerSteve

546

\d+jest wyrażeniem regularnym liczby całkowitej. Więc

//System.Text.RegularExpressions.Regex
resultString = Regex.Match(subjectString, @"\d+").Value;

zwraca ciąg zawierający pierwsze wystąpienie liczby w subjectString.

Int32.Parse(resultString) poda ci numer.


10
do obsługi liczb ujemnych można użyć Regex.Match(subjectString, @"-?\d+").Valuezamiast tego
Jon List

45
Ta odpowiedź nie jest kompletna (w języku C #). Pobiera tylko pierwszą liczbę w ciągu. Musisz itarować nad dopasowaniami: resultString = string.Join (string.Empty, Regex.Matches (subjectString, @ "\ d +"). OfType <Match> () .Select (m => m.Value));
Markus

8
@Markus: Państwa pytanie „I trzeba wyodrębnić numer zawarty wewnątrz łańcucha”, a wszystkie przykłady pokazują jeden numer jest obecny w ciągu. Iterowanie po jednym elemencie nie jest przydatne.
Tim Pietzcker

2
@ ayman: Och, przecinki to tysiące separatorów? Będzie to wymagało znacznie bardziej złożonego wyrażenia regularnego - na co należy odpowiedzieć w osobnym pytaniu. Doskonałym punktem wyjścia jest Regular-Expressions.info, który zawiera również sekcje dotyczące silnika wyrażeń regularnych .NET.
Tim Pietzcker,

5
@DavidSopko: O czym ty mówisz? Pierwotne pytanie dotyczyło sposobu wyodrębnienia pojedynczej liczby z ciągu zarówno w tytule, jak i w treści pytania. Kolejne zmiany w pytaniu (rok po mojej odpowiedzi i później) przez osoby inne niż oryginalny autor zmieniły tytuł na „liczby”. Jeśli cokolwiek, ta błędna edycja powinna zostać wycofana.
Tim Pietzcker,

180

Oto jak oczyszczam numery telefonów, aby uzyskać tylko cyfry:

string numericPhone = new String(phone.Where(Char.IsDigit).ToArray());

31
string numericPhone =new String(phone.Where(Char.IsDigit).ToArray());
Damith,

1
bardzo eleganckie rozwiązanie .. Lubię użycie linq
Leo Gurdian

1
Ładne rozwiązanie dla liczb całkowitych! Pamiętaj, że to nie zadziała, jeśli spróbujesz parsować liczbę dziesiętną, ponieważ przecinek dziesiętny nie jest cyfrą.
Elijah Lofgren

40

użyj wyrażenia regularnego ...

Regex re = new Regex(@"\d+");
Match m = re.Match("test 66");

if (m.Success)
{
    Console.WriteLine(string.Format("RegEx found " + m.Value + " at position " + m.Index.ToString()));
}
else
{
    Console.WriteLine("You didn't enter a string containing a number!");
}

33

Czego używam, aby uzyskać numery telefonów bez interpunkcji ...

var phone = "(787) 763-6511";

string.Join("", phone.ToCharArray().Where(Char.IsDigit));

// result: 7877636511

18

Regex.Split może wyodrębniać liczby z ciągów. Otrzymasz wszystkie liczby znalezione w ciągu.

string input = "There are 4 numbers in this string: 40, 30, and 10.";
// Split on one or more non-digit characters.
string[] numbers = Regex.Split(input, @"\D+");
foreach (string value in numbers)
{
    if (!string.IsNullOrEmpty(value))
    {
    int i = int.Parse(value);
    Console.WriteLine("Number: {0}", i);
    }
}

Wynik:

Liczba: 4 Liczba: 40 Liczba: 30 Liczba: 10


16

Oto Linqwersja:

string s = "123iuow45ss";
var getNumbers = (from t in s
                  where char.IsDigit(t)
                  select t).ToArray();
Console.WriteLine(new string(getNumbers));

14
a co po prostu "123iuow45ss".AsEnumerable().Where(char.IsDigit)?
Ilja Iwanow

2
Po prostu nie lubię from t .. select tredundancji, ale i tak na zdrowie.
Ilja Iwanow

14

Kolejne proste rozwiązanie wykorzystujące Regex Powinieneś tego użyć

using System.Text.RegularExpressions;

a kod to

string var = "Hello3453232wor705Ld";
string mystr = Regex.Replace(var, @"\d", "");
string mynumber = Regex.Replace(var, @"\D", "");
Console.WriteLine(mystr);
Console.WriteLine(mynumber);

12

Możesz także spróbować tego

string.Join(null,System.Text.RegularExpressions.Regex.Split(expr, "[^\\d]"));

ładne, ale jeśli masz spacji między cyframi w oryginalnym ciąg wtedy to daje 1 duże łączone łańcuch z obu numerów dołączył (bez spacji)
Mohammad Zekrallah

11

Wystarczy użyć RegEx, aby dopasować ciąg, a następnie przekonwertować:

Match match = Regex.Match(test , @"(\d+)");
if (match.Success) {
   return int.Parse(match.Groups[1].Value);
}

9

Oto inne Linqpodejście, które wyodrębnia pierwszą liczbę z ciągu.

string input = "123 foo 456";
int result = 0;
bool success = int.TryParse(new string(input
                     .SkipWhile(x => !char.IsDigit(x))
                     .TakeWhile(x => char.IsDigit(x))
                     .ToArray()), out result);

Przykłady:

string input = "123 foo 456"; // 123
string input = "foo 456";     // 456
string input = "123 foo";     // 123

9

Dla tych, którzy chcą liczb dziesiętnych z ciągu z wyrażeniem regularnym w DWÓCH wierszach:

decimal result = 0;
decimal.TryParse(Regex.Match(s, @"\d+").Value, out result);

To samo dotyczy pływaka , długiego itp.


9
 string input = "Hello 20, I am 30 and he is 40";
 var numbers = Regex.Matches(input, @"\d+").OfType<Match>().Select(m => int.Parse(m.Value)).ToArray();

1
To najlepsza odpowiedź, która dała mi to, czego chciałem, czyli tablicę wielu liczb w ciągu. Gdyby tylko mógł zignorować przecinki w liczbach (separator tysięcy), byłoby to idealne! :-)
Sagar,

9

Możesz to zrobić za pomocą Stringwłaściwości takich jak poniżej:

 return new String(input.Where(Char.IsDigit).ToArray()); 

która daje tylko liczbę z ciągu.


7
var match=Regex.Match(@"a99b",@"\d+");
if(match.Success)
{
    int val;
    if(int.TryParse(match.Value,out val))
    {
        //val is set
    }
}

7

Pytanie nie mówi wprost, że chcesz tylko znaków od 0 do 9, ale nie byłoby łatwo uwierzyć, że to prawda z twojego zestawu przykładów i komentarzy. Oto kod, który to robi.

        string digitsOnly = String.Empty;
        foreach (char c in s)
        {
            // Do not use IsDigit as it will include more than the characters 0 through to 9
            if (c >= '0' && c <= '9') digitsOnly += c;
        }

Dlaczego nie chcesz używać Char.IsDigit () - Liczby obejmują znaki, takie jak ułamki, indeksy dolne, indeks górny, cyfry rzymskie, liczniki walut, liczby otoczone i cyfry specyficzne dla skryptu.


6
var outputString = String.Join("", inputString.Where(Char.IsDigit));

Zbierz wszystkie liczby z ciągu. Więc jeśli użyjesz dla przykładu „1 plus 2”, otrzyma „12”.


5

Metoda rozszerzenia, aby uzyskać wszystkie liczby dodatnie zawarte w ciągu:

    public static List<long> Numbers(this string str)
    {
        var nums = new List<long>();
        var start = -1;
        for (int i = 0; i < str.Length; i++)
        {
            if (start < 0 && Char.IsDigit(str[i]))
            {
                start = i;
            }
            else if (start >= 0 && !Char.IsDigit(str[i]))
            {
                nums.Add(long.Parse(str.Substring(start, i - start)));
                start = -1;
            }
        }
        if (start >= 0)
            nums.Add(long.Parse(str.Substring(start, str.Length - start)));
        return nums;
    }

Jeśli chcesz również liczb ujemnych, po prostu zmodyfikuj ten kod, aby obsługiwał znak minus ( -)

Biorąc pod uwagę ten wkład:

"I was born in 1989, 27 years ago from now (2016)"

Wynikowa lista numerów będzie:

[1989, 27, 2016]

5

jeśli liczba ma kropki dziesiętne, możesz użyć poniżej

using System;
using System.Text.RegularExpressions;

namespace Rextester
{
    public class Program
    {
        public static void Main(string[] args)
        {
            //Your code goes here
            Console.WriteLine(Regex.Match("anything 876.8 anything", @"\d+\.*\d*").Value);
            Console.WriteLine(Regex.Match("anything 876 anything", @"\d+\.*\d*").Value);
            Console.WriteLine(Regex.Match("$876435", @"\d+\.*\d*").Value);
            Console.WriteLine(Regex.Match("$876.435", @"\d+\.*\d*").Value);
        }
    }
}

wyniki:

„cokolwiek 876,8 cokolwiek” ==> 876,8

„cokolwiek 876 cokolwiek” ==> 876

„876435 $” ==> 876435

„876,435 $” ==> 876,435

Przykład: https://dotnetfiddle.net/IrtqVt


1
Dziękuję bardzo za tę odpowiedź. Używam C # i VS2017 i próbowałem dowiedzieć się, jak znaleźć wartość. Jeszcze raz dziękuję za odpowiedź.
Bubbles,


2
  string verificationCode ="dmdsnjds5344gfgk65585";
            string code = "";
            Regex r1 = new Regex("\\d+");
          Match m1 = r1.Match(verificationCode);
           while (m1.Success)
            {
                code += m1.Value;
                m1 = m1.NextMatch();
            }

Ten kod służy do znajdowania wszystkich wartości całkowitych w ciągu.
Manoj Gupta

Lepiej byłoby dodać opis bezpośrednio do odpowiedzi, niż opublikować go osobno jako komentarz. Komentarze nie zawsze są natychmiast widoczne.
John Dvorak

2

Interesujące podejście zapewnia tutaj Ahmad Mageed, wykorzystuje Regex i program budujący ciągi, aby wyodrębnić liczby całkowite w kolejności, w jakiej występują w ciągu.

Przykład użycia Regex.Split na podstawie postu Ahmada Mageeda jest następujący:

var dateText = "MARCH-14-Tue";
string splitPattern = @"[^\d]";
string[] result = Regex.Split(dateText, splitPattern);
var finalresult = string.Join("", result.Where(e => !String.IsNullOrEmpty(e)));
int DayDateInt = 0;

int.TryParse(finalresult, out DayDateInt);

2

Oto mój algorytm

    //Fast, C Language friendly
    public static int GetNumber(string Text)
    {
        int val = 0;
        for(int i = 0; i < Text.Length; i++)
        {
            char c = Text[i];
            if (c >= '0' && c <= '9')
            {
                val *= 10;
                //(ASCII code reference)
                val += c - 48;
            }
        }
        return val;
    }

1

oto moje rozwiązanie

string var = "Hello345wor705Ld";
string alpha = string.Empty;
string numer = string.Empty;
foreach (char str in var)
{
    if (char.IsDigit(str))
        numer += str.ToString();
    else
        alpha += str.ToString();
}
Console.WriteLine("String is: " + alpha);
Console.WriteLine("Numeric character is: " + numer);
Console.Read();


0
static string GetdigitFromString(string str)
    {
        char[] refArray = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' };
        char[] inputArray = str.ToCharArray();
        string ext = string.Empty;
        foreach (char item in inputArray)
        {
            if (refArray.Contains(item))
            {
                ext += item.ToString();
            }
        }
        return ext;
    }

0
string s = "kg g L000145.50\r\n";
        char theCharacter = '.';
        var getNumbers = (from t in s
                          where char.IsDigit(t) || t.Equals(theCharacter)
                          select t).ToArray();
        var _str = string.Empty;
        foreach (var item in getNumbers)
        {
            _str += item.ToString();
        }
        double _dou = Convert.ToDouble(_str);
        MessageBox.Show(_dou.ToString("#,##0.00"));

0

Korzystając z odpowiedzi @ tim-pietzcker z góry , następujące będą działać dla PowerShell.

PS C:\> $str = '1 test'
PS C:\> [regex]::match($str,'\d+').value
1

-3

Na podstawie ostatniej próbki stworzyłem metodę:

private string GetNumberFromString(string sLongString, int iLimitNumbers)
{
    string sReturn = "NA";
    int iNumbersCounter = 0;
    int iCharCounter = 0; 

    string sAlphaChars = string.Empty;
    string sNumbers = string.Empty;
    foreach (char str in sLongString)
    {
        if (char.IsDigit(str))
        {
            sNumbers += str.ToString();
            iNumbersCounter++;
            if (iNumbersCounter == iLimitNumbers)
            {
                return sReturn = sNumbers;
            }
        }
        else
        {
            sAlphaChars += str.ToString();
            iCharCounter++;
            // reset the counter 
            iNumbersCounter = 0; 
        }
    }
    return sReturn;
}
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.