0

请查看代码:

/* Run1.java */
package test;

import java.util.Iterator;
import java.util.SortedSet;
import java.util.TreeSet;

public class Run1 
{
    static public void main(String[] args)
    {
        SortedSet<TestClass> s = new TreeSet<TestClass>(); 

        s.add( new TestClass("name1", 100) );
        s.add( new TestClass("name2", 10) );
        s.add( new TestClass("name3", 1) );
        s.add( new TestClass("name4", 10) );
        s.add( new TestClass("name5", 100) );

        Iterator<TestClass> it = s.iterator();

        while(it.hasNext())
        {
            TestClass t = it.next();
            System.out.println( t.name+' '+t.value );
        }
    }
}

/* TestClass.java */
package test;

public class TestClass implements Comparable<TestClass> 
{
    public String name;
    public int value;

    public TestClass(String name, int value) {
        this.name = name;
        this.value = value;
    }

    public int compareTo(TestClass o) 
    {
        return this.value - o.value;
    }

    public boolean equals(Object o) 
    {
        if (!(o instanceof TestClass))
            return false;
        TestClass n = (TestClass)o;

        return this.name.equals(n.name);
    }

    public int hashCode() 
    {
        return 31*name.hashCode();
    }

    public String toString() 
    {
        return name;
    }
}

打印出

name3 1
name2 10
name1 100

正如我所看到的,因为compareTo用于检查相等性(返回 0 时)。但我需要按字段检查唯一性,TestClass.name并且只按TestClass.value

4

4 回答 4

2

compareTo()和的结果equals()在这种情况下需要兼容,这意味着您需要在比较中考虑相等性。例如:

public int compareTo(TestClass o) 
{
    return (this.value == o.value) ? this.name.compareTo(o.name) : this.value - o.value;
}

它为具有相同值的对象引入了名称子顺序,使结果与您的equals()实现兼容。

于 2011-11-16T12:25:45.683 回答
2

如何破解 compareTo 方法如下:

public int compareTo(TestClass o) 
{
    if (this.name != null && this.name.equals(o.name)) {
        return 0;
    }


    return this.value - o.value;
}

这应该在对值进行排序时对名称进行相等检查(删除重复项)

于 2011-11-16T12:25:53.827 回答
1

如果我理解正确,那么您想要的是 compareTo 始终为班级实现“自然顺序”。这意味着类的客户期望类的行为方式。合同上,compareTo 应该与 equals 一致,这就是为什么我总是将 equals 实现为:

return compareTo(obj)==0;

这保证了一致性。

然后,如果你想要另一个排序顺序,你应该实现另一个实现 Comparable 的类。通过这种方式,您可以具有类一致性和单独的排序顺序。

于 2011-11-16T12:34:50.660 回答
0

编写比较 TestClass 对象的比较器。

public class TVComparator implements Comparator<TestClass> {
    public int compare(TestClass o1, TestClass o2) {
        if (o1.name.equals(o2.name)) return 0;
        return o1.value - o2.value;
    }
}

为简单起见,我省略了对空值的任何检查。

于 2011-11-16T12:34:02.293 回答