ฉันจะใช้ตัวเปรียบเทียบเพื่อกำหนดลำดับการจัดเรียงแบบกำหนดเองได้อย่างไร


92

ฉันต้องการพัฒนาการสาธิตการเรียงลำดับสำหรับรายการรถ ฉันใช้ตารางข้อมูลเพื่อแสดงรายการรถ ตอนนี้จริงๆแล้วฉันต้องการจัดเรียงรายการตามสีรถ ที่นี่ไม่ได้เรียงตามลำดับตัวอักษร ฉันต้องการใช้ลำดับการจัดเรียงที่กำหนดเองเช่นรถสีแดงมาก่อนแล้วสีน้ำเงินเป็นต้น

เพื่อที่ฉันพยายามที่จะใช้ Java ComparatorและComparableแต่มันจะช่วยให้การเรียงลำดับตามตัวอักษรเท่านั้น

มีใครช่วยแนะนำวิธีการนำเทคนิคไปใช้ได้ไหมเพื่อให้การเรียงลำดับเร็วขึ้น

class Car implements Comparable<Car>
{
    private String name;
    private String color;

    public Car(String name, String color){
        this.name = name;
        this.color = color;
    }

    //Implement the natural order for this class
    public int compareTo(Car c) {
        return name.compareTo(c.name);
    }

    static class ColorComparator implements Comparator<Car> {
        public int compare(Car c1, Car c2) {
            String a1 = c1.color;
            String a2 = c2.color;
            return a1.compareTo(a2);
        }
    }

    public static void main(String[] args) {
        List<Car> carList = new ArrayList<>();
        List<String> sortOrder = new ArrayList<>();

        carList.add(new Car("Ford","Silver"));
        carList.add(new Car("Tes","Blue"));
        carList.add(new Car("Honda","Magenta"));

        sortOrder.add("Silver");
        sortOrder.add("Magenta");
        sortOrder.add("Blue");

        // Now here I am confuse how to implement my custom sort             
    }
}

คำตอบ:


101

ผมแนะนำให้คุณสร้าง enum สำหรับสีรถของคุณแทนที่จะใช้ Strings และลำดับตามธรรมชาติของ enum จะเป็นลำดับที่คุณประกาศค่าคงที่

public enum PaintColors {
    SILVER, BLUE, MAGENTA, RED
}

และ

 static class ColorComparator implements Comparator<CarSort>
 {
     public int compare(CarSort c1, CarSort c2)
     {
         return c1.getColor().compareTo(c2.getColor());
     }
 }

คุณเปลี่ยน String เป็น PaintColor จากนั้นในรายการรถหลักของคุณจะกลายเป็น:

carList.add(new CarSort("Ford Figo",PaintColor.SILVER));

...

Collections.sort(carList, new ColorComparator());

ฉันจะรันตัวอย่างนี้ได้อย่างไร PaintColors ไม่สามารถเข้าถึงได้ใน ColorComparator คุณช่วยอธิบายได้ไหมว่าเมธอดหลักจะเป็นอย่างไร
Deepak

ผลลัพธ์คืออะไร? SILVER เป็นอันดับแรกเสมอหรือไม่?
Deepak

@ ดีพัค: ใช่ลำดับตามธรรมชาติของenumค่าคือลำดับที่กำหนดค่า
Joachim Sauer

1
@SeanPatrickFloyd ฉันไม่คิดว่าการพึ่งพาการเรียงลำดับตามธรรมชาติของ enums เป็นแนวทางปฏิบัติที่ดีที่สุด
ไมค์

1
@mike ฉันไม่เห็นด้วย แต่มีข้อโต้แย้งที่ถูกต้องสำหรับทั้งสองมุมมอง
Sean Patrick Floyd

57

แล้วสิ่งนี้ล่ะ:

List<String> definedOrder = // define your custom order
    Arrays.asList("Red", "Green", "Magenta", "Silver");

Comparator<Car> comparator = new Comparator<Car>(){

    @Override
    public int compare(final Car o1, final Car o2){
        // let your comparator look up your car's color in the custom order
        return Integer.valueOf(
            definedOrder.indexOf(o1.getColor()))
            .compareTo(
                Integer.valueOf(
                    definedOrder.indexOf(o2.getColor())));
    }
};

โดยหลักการแล้วฉันยอมรับว่าการใช้ an enumเป็นแนวทางที่ดียิ่งขึ้น แต่เวอร์ชันนี้มีความยืดหยุ่นมากกว่าเนื่องจากให้คุณกำหนดลำดับการจัดเรียงที่แตกต่างกันได้

อัปเดต

ฝรั่งมีฟังก์ชั่นนี้รวมอยู่ในOrderingคลาส:

List<String> colorOrder = ImmutableList.of("red","green","blue","yellow");
final Ordering<String> colorOrdering = Ordering.explicit(colorOrder);
Comparator<Car> comp = new Comparator<Car>() {
    @Override
    public int compare(Car o1, Car o2) {
        return colorOrdering.compare(o1.getColor(),o2.getColor());
    }
}; 

เวอร์ชันนี้มีความละเอียดน้อยกว่าเล็กน้อย


อัพเดทอีกครั้ง

Java 8 ทำให้ตัวเปรียบเทียบมีรายละเอียดน้อยลง:

Comparator<Car> carComparator = Comparator.comparing(
        c -> definedOrder.indexOf(c.getColor()));

คุณช่วยดูคำถามนี้ด้วย ขอบคุณ. stackoverflow.com/questions/61934313/…
user2048204

Comparator<Car> carComparator = Comparator.comparing( c -> definedOrder.indexOf(c.getColor())); กำลังแสดงข้อผิดพลาดในการคอมไพล์
Brooklynn99

@SantosshKumhar นี่คือตัวอย่างการทำงาน: ideone.com/frUbym
Sean Patrick Floyd

25

ตัวเปรียบเทียบในแนว ...

List<Object> objList = findObj(name);
Collections.sort(objList, new Comparator<Object>() {
    @Override
    public int compare(Object a1, Object a2) {
        return a1.getType().compareToIgnoreCase(a2.getType());
    }
});

12

ฉันคิดว่าสิ่งนี้สามารถทำได้ดังนี้:

class ColorComparator implements Comparator<CarSort>
{
    private List<String> sortOrder;
    public ColorComparator (List<String> sortOrder){
        this.sortOrder = sortOrder;
    }

    public int compare(CarSort c1, CarSort c2)
    {
        String a1 = c1.getColor();
        String a2 = c2.getColor();
        return sortOrder.indexOf(a1) - sortOrder.indexOf(a2);
     }
 }

สำหรับการเรียงลำดับให้ใช้สิ่งนี้:

Collections.sort(carList, new ColorComparator(sortOrder));

7

ฉันต้องทำบางอย่างที่คล้ายกับคำตอบของฌอนและอิลาเล็กซ์
แต่ฉันมีตัวเลือกมากเกินไปในการกำหนดลำดับการจัดเรียงอย่างชัดเจนและจำเป็นต้องลอยเฉพาะบางรายการที่ด้านหน้าของรายการ ... ตามลำดับที่ระบุ (ไม่เป็นธรรมชาติ)
หวังว่านี่จะเป็นประโยชน์กับคนอื่น ๆ

public class CarComparator implements Comparator<Car> {

    //sort these items in this order to the front of the list 
    private static List<String> ORDER = Arrays.asList("dd", "aa", "cc", "bb");

    public int compare(final Car o1, final Car o2) {
        int result = 0;
        int o1Index = ORDER.indexOf(o1.getName());
        int o2Index = ORDER.indexOf(o2.getName());
        //if neither are found in the order list, then do natural sort
        //if only one is found in the order list, float it above the other
        //if both are found in the order list, then do the index compare
        if (o1Index < 0 && o2Index < 0) result = o1.getName().compareTo(o2.getName());
        else if (o1Index < 0) result = 1;
        else if (o2Index < 0) result = -1;
        else result = o1Index - o2Index;
        return result;
    }

//Testing output: dd,aa,aa,cc,bb,bb,bb,a,aaa,ac,ac,ba,bd,ca,cb,cb,cd,da,db,dc,zz
}

4

ฉันจะทำสิ่งนี้:

List<String> order = List.of("Red", "Green", "Magenta", "Silver");

Comparator.comparing(Car::getColor(), Comparator.comparingInt(c -> order.indexOf(c)))

เครดิตทั้งหมดไปที่ @Sean Patrick Floyd :)


3

ใน Java 8 คุณสามารถทำสิ่งนี้:

ก่อนอื่นคุณต้องมี Enum:

public enum Color {
    BLUE, YELLOW, RED
}

ระดับรถยนต์:

public class Car {

    Color color;

    ....

    public Color getColor() {
        return color;
    }

    public void setColor(Color color) {
        this.color = color;
    }
}

จากนั้นใช้รายการรถของคุณคุณสามารถทำได้:

Collections.sort(carList, Comparator:comparing(CarSort::getColor));

นี่ไม่ใช่การจัดเรียงแบบกำหนดเอง
zygimantus

นอกจากนี้ยังไม่ใช่วิธีที่ "ใช้งานได้" ... สิ่งนี้มีผลกระทบด้าน !!
TriCore

2

กำหนด Enum Type เป็น

public enum Colors {
     BLUE, SILVER, MAGENTA, RED
}

เปลี่ยนชนิดข้อมูลcolorจากStringเป็นColors เปลี่ยนชนิดผลตอบแทนและประเภทอาร์กิวเมนต์ของ getter และ setter วิธีการของสีเป็นColors

กำหนดประเภทตัวเปรียบเทียบดังนี้

static class ColorComparator implements Comparator<CarSort>
{
    public int compare(CarSort c1, CarSort c2)
    {
        return c1.getColor().compareTo(c2.getColor());
    }
}

หลังจากเพิ่มองค์ประกอบในรายการแล้วให้เรียกวิธีการจัดเรียงของคอลเล็กชันโดยส่งผ่านรายการและวัตถุตัวเปรียบเทียบเป็นอาร์กิวเมนต์

เช่นนั้นพิมพ์โดยใช้Collections.sort(carList, new ColorComparator()); ListIterator

การใช้งานคลาสเต็มมีดังนี้:

package test;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;    
import java.util.ListIterator;

public class CarSort implements Comparable<CarSort>{

    String name;
    Colors color;

    public CarSort(String name, Colors color){
        this.name = name;
        this.color = color;
    } 

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Colors getColor() {
        return color;
    }
    public void setColor(Colors color) {
        this.color = color;
    }

    //Implement the natural order for this class
    public int compareTo(CarSort c)
    {
        return getName().compareTo(c.getName());
    }

    static class ColorComparator implements Comparator<CarSort>
    {
        public int compare(CarSort c1, CarSort c2)
        {
            return c1.getColor().compareTo(c2.getColor());
        }
    }

    public enum Colors {
         BLUE, SILVER, MAGENTA, RED
    }

     public static void main(String[] args)
     {
         List<CarSort> carList = new ArrayList<CarSort>();
         List<String> sortOrder = new ArrayList<String>();

         carList.add(new CarSort("Ford Figo",Colors.SILVER));
         carList.add(new CarSort("Santro",Colors.BLUE));
         carList.add(new CarSort("Honda Jazz",Colors.MAGENTA));
         carList.add(new CarSort("Indigo V2",Colors.RED));
         Collections.sort(carList, new ColorComparator());

         ListIterator<CarSort> itr=carList.listIterator();
         while (itr.hasNext()) {
            CarSort carSort = (CarSort) itr.next();
            System.out.println("Car colors: "+carSort.getColor());
        }
     }
}

0

ใช้ลูปง่ายๆ:

public static void compareSortOrder (List<String> sortOrder, List<String> listToCompare){
        int currentSortingLevel = 0;
        for (int i=0; i<listToCompare.size(); i++){
            System.out.println("Item from list: " + listToCompare.get(i));
            System.out.println("Sorting level: " + sortOrder.get(currentSortingLevel));
            if (listToCompare.get(i).equals(sortOrder.get(currentSortingLevel))){

            } else {
                try{
                    while (!listToCompare.get(i).equals(sortOrder.get(currentSortingLevel)))
                        currentSortingLevel++;
                    System.out.println("Changing sorting level to next value: " + sortOrder.get(currentSortingLevel));
                } catch (ArrayIndexOutOfBoundsException e){

                }

            }
        }
    }

และเรียงลำดับใน List

public static List<String> ALARMS_LIST = Arrays.asList(
            "CRITICAL",
            "MAJOR",
            "MINOR",
            "WARNING",
            "GOOD",
            "N/A");
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.