C# - Как отсортировать строки, содержащие текст и числа

У меня есть список, который заполнен именами компьютеров, именами пользователей и числом (в виде строки). Я уже создал свой собственный ListViewItemComparer, используя интерфейс IComparer. Но он не сортирует предметы так, как я хочу. Вот как он должен отсортировать компьютеры, например: Компьютер-1 Компьютер-2 Компьютер-3 ... Компьютер-15

и вот как он их сортирует: Компьютер-1 Компьютер-10 Компьютер-11 ... Компьютер-2 Компьютер-3

Проблема в том, что я не могу просто вырезать «компьютерную» часть и сравнить следующие числа, поскольку это был всего лишь пример, а имена компьютеров могут быть чем угодно (aaa393bbb333, ccccvvvv, 222hhhdh, Computer-01, Computer-02, ....)

Это мой код:

 private bool isNumeric(String pInput)
    {
        int o;
        return int.TryParse(pInput, out o);
    }

    public int Compare(object x, object y)
    {
        ListViewItem itemX = x as ListViewItem;
        ListViewItem itemY = y as ListViewItem;
        //
        int returnVal = -1;
        if (itemX == null && itemY == null) returnVal = 0;
        else if (itemX == null) returnVal = -1;
        else if (itemY == null) returnVal = 1;
        else if (itemX.SubItems.Count - 1 < col && itemY.SubItems.Count - 1 < col) returnVal = 0;
        else if (itemX.SubItems.Count - 1 < col) returnVal = -1;
        else if (itemY.SubItems.Count - 1 < col) returnVal = 1;
        else if (isNumeric(itemX.SubItems[col].Text) && isNumeric(itemY.SubItems[col].Text))
        {
            //used for number comparison
            int value1 = int.Parse(itemX.SubItems[col].Text);
            int value2 = int.Parse(itemY.SubItems[col].Text);
            if (value1 == value2) returnVal = 0;
            else if (value1 < value2) returnVal = -1;
            else if (value1 > value2) returnVal = 1;
        }
        else returnVal = String.Compare(itemX.SubItems[col].Text, itemY.SubItems[col].Text);
        if (order == SortOrder.Descending)
            returnVal *= -1;
        return returnVal;
    }

Конечно, вы можете «вырезать» имя компьютера из номера, только не забывайте о них ;-) При сравнении двух имен компьютеров вы сначала сравниваете часть имени компьютера без номера. Если они разные (неравные), то теперь вы получили желаемый результат сравнения. Если части имени компьютера идентичны (равны), вы затем сравните числовые части. Итак, теперь вы также получили желаемый результат сравнения, даже если части имени компьютера совпадают ...

user2819245 10.11.2018 16:54

В окна встроен метод "естественной сортировки". Проводник файлов использует его для сортировки буквенно-цифровых имен файлов

Ňɏssa Pøngjǣrdenlarp 10.11.2018 16:59

Спасибо за Ваш ответ. Но вы меня не поняли. «Компьютер» был только примером названия компьютера. Имена компьютеров могут быть чем угодно. Таким образом, имя компьютера может быть даже aaaa100aa2. Но как я могу сравнить это с другими компьютерными именами?

C.M. 10.11.2018 17:00

@Disaffected 1070452 да, это именно то, что я ищу. Но как мне это реализовать?

C.M. 10.11.2018 17:05

Зайдите в свою любимую поисковую систему и исследуйте «естественный вид». Здесь уже есть десятки решений.

Ňɏssa Pøngjǣrdenlarp 10.11.2018 17:10
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
0
5
141
1

Ответы 1

Вы можете использовать выражение регулярного выражения

(?<=\D)(?=\d)|(?<=\d)(?=\D)

чтобы разбить строку на числа и текст.

string[] parts = Regex.Split(theString, @"(?<=\D)(?=\d)|(?<=\d)(?=\D)");

Это дает поочередно числа и текст. Как это работает:

(?<=exp)pos      Match any position pos following a prefix exp. 
pos(?=exp)       Match any position pos preceding a suffix exp. 

Таким образом, регулярное выражение означает разделение в позиции между текстом и числом или в позиции между числом и текстом, где \D означает нецифровой символ, \d - цифровой символ и | - OR.

Вам нужно будет проверить, является ли первая часть текстом или числом с

bool firstIsNumber = Char.IsDigit(parts[0][0]);

Если первая часть является числом, все части с четными индексами являются числами, а все части с нечетными индексами являются текстами и наоборот.

private static readonly Regex numTextSplitRegex =
    new Regex(@"(?<=\D)(?=\d)|(?<=\d)(?=\D)", RegexOptions.Compiled);

public int Compare(string x, string y)
{
    x = x ?? "";
    y = y ?? "";
    string[] xParts = numTextSplitRegex.Split(x);
    string[] yParts = numTextSplitRegex.Split(y);

    bool firstXIsNumber = xParts[0].Length > 0 && Char.IsDigit(xParts[0][0]);
    bool firstYIsNumber = yParts[0].Length > 0 && Char.IsDigit(yParts[0][0]);

    if (firstXIsNumber != firstYIsNumber) {
        return x.CompareTo(y);
    }

    for (int i = 0; i < Math.Min(xParts.Length, yParts.Length); i++) {
        int result;
        if (firstXIsNumber == (i % 2 == 0)) { // Compare numbers.
            long a = Int64.Parse(xParts[i]);
            long b = Int64.Parse(yParts[i]);
            result = a.CompareTo(b);
        } else { // Compare texts.
            result = xParts[i].CompareTo(yParts[i]);
        }
        if (result != 0) {
            return result;
        }
    }
    return xParts.Length.CompareTo(yParts.Length);
}

Другие вопросы по теме