What is the best way to get min and max value from a list of Comparables that main contain null values?

42,684

Solution 1

What's wrong with Collections.max?

And why do you care about null safety? Are you sure you want to allow nulls to be in your Collection?

Solution 2

If you really need to exclude "null" from the result, and you can't prevent it from being in your array, then maybe you should just iterate through the array with a simple loop and keep track of the "min" and "max" in separate variables. You can still use the "compare()" method on each object to compare it with your current "min" and "max" values. This way, you can add your own code for checking for nulls and ignoring them.

EDIT: here's some code to illustrate what I'm talking about. Unfortunately there is an edge case you need to consider - what if all of the arguments passed in are null? What does your method return?

public static <T extends Comparable<T>> T minOf(T...ts){
    T min = null;
    for (T t : ts) {
        if (t != null && (min == null || t.compareTo(min) < 0)) {
            min = t;
        }
    }
    return min;
}

public static <T extends Comparable<T>> T maxOf(T...ts){
    T max = null;
    for (T t : ts) {
        if (t != null && (max == null || t.compareTo(max) > 0)) {
            max = t;
        }
    }
    return max;
}

Solution 3

You should not implement Comparable to accept null, as it breaks the interface's contract.

From https://docs.oracle.com/javase/7/docs/api/java/lang/Comparable.html :

Note that null is not an instance of any class, and e.compareTo(null) should throw a NullPointerException even though e.equals(null) returns false.

You must instead create a new interface, e.g. ComparableNull instead.

See also:

Share:
42,684
user2427
Author by

user2427

Updated on August 11, 2022

Comments

  • user2427
    user2427 almost 2 years

    I am thinking about something like this:

    public static <T extends Comparable<T>> T minOf(T...ts){        
        SortedSet<T> set = new TreeSet<T>(Arrays.asList(ts));
        return set.first();
    }
    
    public static <T extends Comparable<T>> T maxOf(T...ts){
        SortedSet<T> set = new TreeSet<T>(Arrays.asList(ts));
        return set.last();
    }
    

    But is not null safe, which is something I want too.

    Do you know a better way to solve this problem?

    EDIT:

    After the comments I have also tried min():

    public static <T extends Comparable<T>> T minOf(T...ts){        
        return Collections.min(Arrays.asList(ts), new Comparator<T>(){
    
            public int compare(T o1, T o2) {
                if(o1!=null && o2!=null){
                    return o1.compareTo(o2);
                }else if(o1!=null){
                    return 1;
                }else{
                    return -1;  
                }
            }});
    }
    

    What do you think of that?

  • Yetti99
    Yetti99 about 9 years
    The question was for max and min. Collections.max() will look at every element. Then doing Collections.min() will again look at every element. So maybe sort just once is faster.
  • Ciro Santilli OurBigBook.com
    Ciro Santilli OurBigBook.com almost 9 years
    @Yetti99 sorting will be slower because it is O(n log n) and max + min is O(2n). What would be a little better is to do a single loop and update both max and min in it. And the OP does not seem to require both in a single function.
  • Yetti99
    Yetti99 almost 9 years
    @Ciro Santilli 六四事件 法轮功 纳米比亚胡海峰 You are right about average complicity of a sort. But if a list is nearly in order the complexity will be O(n) best case. And then its quite possible that the application may need to have the list sorted for some other reason.