5

我有一个文件名数组,需要按文件名的扩展名对该数组进行排序。是否有捷径可寻?

4

8 回答 8

21
Arrays.sort(filenames, new Comparator<String>() {
    @Override
    public int compare(String s1, String s2) {
        // the +1 is to avoid including the '.' in the extension and to avoid exceptions
        // EDIT:
        // We first need to make sure that either both files or neither file
        // has an extension (otherwise we'll end up comparing the extension of one
        // to the start of the other, or else throwing an exception)
        final int s1Dot = s1.lastIndexOf('.');
        final int s2Dot = s2.lastIndexOf('.');
        if ((s1Dot == -1) == (s2Dot == -1)) { // both or neither
            s1 = s1.substring(s1Dot + 1);
            s2 = s2.substring(s2Dot + 1);
            return s1.compareTo(s2);
        } else if (s1Dot == -1) { // only s2 has an extension, so s1 goes first
            return -1;
        } else { // only s1 has an extension, so s1 goes second
            return 1;
        }
    }
});

为了完整性:java.util.Arraysjava.util.Comparator

于 2009-01-13T19:39:17.303 回答
4

如果我没记错的话, Arrays.sort(...) 需要一个 Comparator<> ,它将用于进行排序。您可以提供一个查看字符串扩展部分的实现。

于 2009-01-13T19:35:06.953 回答
3

您可以实现自定义的字符串比较器。使其按'.'. 然后将比较器和您的数组传入

Arrays.sort(stringArray, yourComparator);

//  An implementation of the compare method
public int compare(String o1, String o2) {
    return o1.substring(o1.lastIndexOf('.')).compareTo(o2.substring(o2.lastIndexOf('.'));
}
于 2009-01-13T19:35:26.670 回答
3

比较器通常很难完全正确,并且必须为每个比较生成比较键,对于大多数排序算法来说,这意味着 O(n log n)。另一种方法是为您需要排序的每个项目创建(键,值)对,将它们放在 TreeMap 中,然后询问值,因为这些值是根据键排序的。

例如

import java.util.Arrays;
import java.util.TreeMap;

public class Bar {

    public static void main(String[] args) {
        TreeMap<String, String> m2 = new TreeMap<String, String>();
        for (String string : Arrays.asList(new String[] { "#3", "#2", "#1" })) {
            String key = string.substring(string.length() - 1);
            String value = string;
            m2.put(key, value);
        }
        System.out.println(m2.values());
    }
}

打印出来

[#1、#2、#3]

您应该能够轻松地使关键计算适应您的问题。

每个条目只计算一次密钥,因此 O(n) - (但排序仍然是 O(n log n))。如果密钥计算很昂贵或 n 很大,这可能是相当可衡量的。

于 2009-01-13T22:44:16.570 回答
1

创建一个比较器并比较字符串扩展名。看看以下

http://java.sun.com/j2se/1.4.2/docs/api/java/util/Comparator.html

然后将您的字符串列表传递给 Arrays.sort(List, Comparator)

于 2009-01-13T19:36:10.977 回答
1

创建您自己的比较器,将字符串视为文件名并根据扩展名进行比较。然后将Arrays.sort与 Comparator 参数一起使用。

于 2009-01-13T19:36:14.703 回答
1
    String DELIMETER = File.separator + ".";
    List<String> orginalList = new CopyOnWriteArrayList<>(Arrays.asList(listOfFileNames));
    Set<String> setOfuniqueExtension = new TreeSet<>();

    for (String item : listOfFileNames) {
        if (item.contains(".")) {
            String[] split = item.split(DELIMETER);
            String temp = "." + split[split.length - 1];
            setOfuniqueExtension.add(temp);
        }
    }

    List<String> finalListOfAllFiles = new LinkedList<>();
    setOfuniqueExtension.stream().forEach((s1) -> {
        for (int i = 0; i < orginalList.size(); i++) {
            if (orginalList.get(i).contains(s1)) {
                finalListOfAllFiles.add(orginalList.get(i));
                orginalList.remove(orginalList.get(i));
                i--;
            }
        }
    });

    orginalList.stream().filter((s1) -> (!finalListOfAllFiles.contains(s1))).forEach((s1) -> {
        finalListOfAllFiles.add(s1);
    });

    return finalListOfAllFiles;
于 2017-10-17T23:25:49.667 回答
-1

如果您只想按扩展名对文件进行分组而不关心实际的字母顺序,您可以使用以下命令:

我认为当文件名没有“。”时,您可以做的最简单的事情也适用。只是颠倒名称并比较它们。

Arrays.sort(ary, new Comparator<String>() {
    @Override
    public int compare(String o1, String o2) {
        String r1 = new StringBuffer(o1).reverse().toString();
        String r2 = new StringBuffer(o2).reverse().toString();
        return r1.compareTo(r2);
    }
});

可惜java的字符串甚至没有reverse()。

于 2009-01-13T21:13:34.103 回答