Сортування однієї струни на Java


131

Чи існує рідний спосіб сортування рядка за його вмістом у java? Напр

String s = "edcba"  ->  "abcde"

Відповіді:


215

toCharArrayз Arrays.sortподальшим викликом конструктора String:

import java.util.Arrays;

public class Test
{
    public static void main(String[] args)
    {
        String original = "edcba";
        char[] chars = original.toCharArray();
        Arrays.sort(chars);
        String sorted = new String(chars);
        System.out.println(sorted);
    }
}

EDIT: Як вказує лінія, це не вдасться, якщо рядок містить сурогатні пари або справді складені символи (наголос + e як окремі символи) тощо. У цей момент це стає набагато складніше ... сподіваємось, вам це не потрібно :) Крім того, це просто замовлення порядковим порядком, не беручи до уваги великі літери, акценти чи що-небудь інше.


2
Правильним способом було б сортування точок коду. На жаль, немає String.toCodePointArray. (Який порядок ми повинні сортувати, btw?)
Том Хоутін - tackline

1
Проект ICU описує метод сортування UTF-16 у порядку коду: icu-project.org/docs/papers/utf16_code_point_order.html . Я не думаю, що Arrays.sort знищить будь-які додаткові символи через спосіб визначення діапазонів, але не цитуйте мене.
МакДауелл

1
Це може нічого не зруйнувати, але порядок сортування не є оптимальним, якщо ви хочете взяти до уваги великі регістри та акценти, наприклад. Цей алгоритм буде сортувати "éDedCBcbAàa" як "ABCDabcdeàé", тоді як, наприклад, в англійській (американській) мові, більш бажано отримати "aAàbBcCdDeé".
eljenso

1
@YiweiG І в цьому випадку відповідь така: точно ні. sortedце вже String... що ви очікували б закликати toString()це зробити?
Джон Скіт

1
@Hengameh: Ви сортуєте масив символів, але потім ігноруєте його. Ви хочетеchar[] c = s.toCharArray(); Arrays.sort(c); String sorted = new String(c);
Джон Скіт

49

Ні, немає вбудованого методу String. Ви можете перетворити його в масив char, сортувати його за допомогою Arrays.sort та перетворити його назад у String.

String test= "edcba";
char[] ar = test.toCharArray();
Arrays.sort(ar);
String sorted = String.valueOf(ar);

Або якщо ви хочете правильно попрацювати з певними локальними речами, такими як великі регістри та наголошені символи:

import java.text.Collator;
import java.util.Arrays;
import java.util.Comparator;
import java.util.Locale;

public class Test
{
  public static void main(String[] args)
  {
    Collator collator = Collator.getInstance(new Locale("fr", "FR"));
    String original = "éDedCBcbAàa";
    String[] split = original.split("");
    Arrays.sort(split, collator);
    String sorted = "";
    for (int i = 0; i < split.length; i++)
    {
      sorted += split[i];
    }
    System.out.println(sorted); // "aAàbBcCdDeé"
  }
}

FYI: цей метод розділить 32-бітові кодові точки на два - символи Unicode зі значенням більше 0xFFFF, створюючи рядки з недійсними значеннями. Не є проблемою для французької мови, але може спричинити проблеми для деяких місцевостей.
МакДоуелл

Дивіться Character.isHighSurrogate (char)
McDowell

1
Я думаю, що це станеться ... якщо тільки він не захоче сортувати рядки, що містять суахілі чи що-небудь :)
eljenso

3
"Я думаю, що це вдасться ... якщо тільки він не захоче сортувати рядки, що містять суахілі" - я бачу гасло - Unicode: коли ви хочете простий спосіб локалізації та перекладу ваших додатків на деякі мови. Bzzt. Збій. Зробити це майже правильно, ви майже не маєте помилок, щоб їх виправити пізніше.
Йонас Келькер

@Jonas Я сказав, що думаю , якщо ОП не хоче вказати, що абсолютно необхідно підтримувати суахілі. Я навіть віддаю перевагу простому рішенню без локальної точки, знову ж таки, якщо в ОП не зазначено, що це недостатньо. Ви коли-небудь чули про принцип YAGNI?
eljenso

32

У Java 8 це можна зробити за допомогою:

String s = "edcba".chars()
    .sorted()
    .collect(StringBuilder::new, StringBuilder::appendCodePoint, StringBuilder::append)
    .toString();

Трохи коротша альтернатива, яка працює з потоком струн довжиною один (кожен символ у несортованій рядку перетворюється на рядок у потоці):

String sorted =
    Stream.of("edcba".split(""))
        .sorted()
        .collect(Collectors.joining());

@Dennis на якому саме і чому?
Марцін

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

18

Перетворити у масив символівСортуватиПеретворити назад у рядок :

String s = "edcba";
char[] c = s.toCharArray();        // convert to array of chars 
java.util.Arrays.sort(c);          // sort
String newString = new String(c);  // convert back to String
System.out.println(newString);     // "abcde"

Який сенс було додавати цю відповідь через 4 роки після того, як інші люди опублікували принаймні 3 однакові відповіді
Нік Кардосо

1
@ NickCardoso я насправді не пам'ятаю, ви запитуєте мене про відповідь, яку я розмістив на своїх ранніх стадіях на Stack Overflow. Ви справді чекаєте пояснень з цього приводу?
Марун

@NickCardoso Я взагалі НЕ намагаюся дати тобі виправдання. Ваш коментар та голосова заява наразі нічого не змінить. Ось що я вирішив ЧЕТВОРИХ років тому, я не бачу сенсу піднімати це зараз :)
Maroun

16

Більш необгрунтований підхід без використання сортування Arrays.sort метод. Для цього використовується сортування вставки.

public static void main(String[] args){
    String wordSt="watch";
    char[] word=wordSt.toCharArray();

    for(int i=0;i<(word.length-1);i++){
        for(int j=i+1;j>0;j--){
            if(word[j]<word[j-1]){
                char temp=word[j-1];
                word[j-1]=word[j];
                word[j]=temp;
            }
        }
    }
    wordSt=String.valueOf(word);
    System.out.println(wordSt);
}

1
запитання задали Native way в java, не використовуючи жодного іншого алгоритму сортування.
Vikrant Goel

1
Голосували за те, що це було корисне рішення, а не тому, що це була запитувана відповідь.
Кріс

1
@VikrantGoel "Рідний шлях у Яві" - що не є рідним у цьому підході? я не бачу жодних сторонніх вимог. Чи ти?
Нік Кардосо

Це має бути правильна відповідь. Як сказав @NickCardoso ... ніщо не є "рідним способом у Java", ніж це.
Маріано Зоррілья

14
    String a ="dgfa";
    char [] c = a.toCharArray();
    Arrays.sort(c);
    return new String(c);

Зауважте, що це не буде працювати, як очікувалося, якщо це змішаний регістр String (Він поставить верхній регістр перед малим регістром). Ви можете передати порівняльник до методу Сортування, щоб змінити це.


1
вам потрібно імпортувати java.util.Arrays; інакше це не працюватиме
бункер

3

Порядок:

  1. Спочатку перетворять рядок у масив char
  2. Потім сортуйте масив символів
  3. Перетворити масив символів у рядок
  4. Роздрукуйте рядок

Фрагмент коду:

    String input = "world";
    char[] arr = input.toCharArray();
    Arrays.sort(arr);
    String sorted = new String(arr);
    System.out.println(sorted);

Це жарт? Опублікував однакові відповіді на запитання 8 років? Як ледачий.
Нік Кардосо

Я невіглас. Пробач мені.
rahedcs

2

Питання: сортуйте рядок у Java

public class SortAStringInJava {
    public static void main(String[] args) {

        String str = "Protijayi";
// Method 1
        str = str.chars() // IntStream
                .sorted().collect(StringBuilder::new, StringBuilder::appendCodePoint, StringBuilder::append).toString();

        System.out.println(str);
        // Method 2
        str = Stream.of(str.split(" ")).sorted().collect(Collectors.joining());
        System.out.println(str);
    }
}

-1
public static void main(String[] args) {
    String str = "helloword";   
    char[] arr;
    List<Character> l = new ArrayList<Character>();
    for (int i = 0; i < str.length(); i++) {
        arr = str.toCharArray();
        l.add(arr[i]);

    }
    Collections.sort(l);
    str = l.toString();
    System.out.println(str);
    str = str.replaceAll("\\[", "").replaceAll("\\]", "")
            .replaceAll("[,]", "");
    System.out.println(str);

}

-2

Без використання колекцій на Java:

import java.util.Scanner;

public class SortingaString {
    public static String Sort(String s1)
    {
        char ch[]=s1.toCharArray();         
        String res=" ";
        
        for(int i=0; i<ch.length ; i++)
        {
            for(int j=i+1;j<ch.length; j++)
            {
                if(ch[i]>=ch[j])
                {
                    char m=ch[i];
                    ch[i]=ch[j];
                    ch[j]=m;
                }
            }
            
            res=res+ch[i];
            
        }

        return res;
    }

    public static void main(String[] args) {
        Scanner sc=new Scanner(System.in);
        System.out.println("enter the string");
        
        String s1=sc.next();
        String ans=Sort( s1);
        
        System.out.println("after sorting=="+ans);
    }
}

Вихід:

введіть рядок ==

сортування

після сортування == ginorst


Слід переглянути ще раз> =. Що відбувається з "mnmnmnm" як вашим рядком?
Нік Кардосо
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.