皮皮网

【168彩源码】【南山对讲源码】【js星空源码】comparator源码

2024-11-18 22:45:27 来源:平滑kd指标源码

1.TreeMap就这么简单【源码剖析】
2.Comparable和Comparator的区别
3.java中compareToIgnoreCase大
4.java的源码Comparable和Comparator区别
5.java-增强版Comparator进行集合排序null异常处理
6.public static <T> int binarySearch(List<? extends T> list, T key, Comparator<? super T> c)

comparator源码

TreeMap就这么简单【源码剖析】

       本文主要讲解TreeMap的实现原理,使用的源码是JDK1.8版本。

       在开始之前,源码建议读者具备一定的源码数据结构基础知识。

       TreeMap的源码实现主要通过红黑树和比较器Comparator来保证元素的有序性。如果构造时传入了Comparator对象,源码168彩源码则使用Comparator的源码compare方法进行元素比较。否则,源码使用Comparable接口的源码compareTo方法实现自然排序。

       TreeMap的源码核心方法有put、get和remove等。源码put方法用于插入元素,源码同时会根据Comparator或Comparable对元素进行排序。源码get方法用于查找指定键的源码值,remove方法则用于删除指定键的源码南山对讲源码元素。

       遍历TreeMap通常使用EntryIterator类,该类提供了按顺序遍历元素的方法。TreeMap的遍历过程基于红黑树的结构,通过查找、比较和调整节点来实现。

       总之,TreeMap是一个基于红黑树的有序映射集合,其主要特性包括元素的有序性、高效的时间复杂度以及灵活的比较方式。在设计和实现需要有序映射的数据结构时,TreeMap是一个不错的选择。

       如有错误或疑问,欢迎在评论区指出,让我们共同进步。js星空源码

       请注意,上述HTML代码片段经过了精简和格式调整,保留了原文的主要内容和结构,但为了适应HTML格式并删除了不相关的内容(如标题、关注转发等),在字数控制上也有所调整。

Comparable和Comparator的区别

       comparator接口与Comparable接口的区别

       1. Comparator 和 Comparable 相同的地方

       ä»–们都是java的一个接口, 并且是用来对自定义的class比较大小的,

       ä»€ä¹ˆæ˜¯è‡ªå®šä¹‰class: 如 public class Person{ String name; int age }.

       å½“我们有这么一个personList,里面包含了person1, person2, persion3....., 我们用Collections.sort( personList ), 是得不到预期的结果的. 这时肯定有人要问, 那为什么可以排序一个字符串list呢:

       å¦‚ StringList{ "hello1" , "hello3" , "hello2"}, Collections.sort( stringList ) 能够得到正确的排序, 那是因为 String 这个对象已经帮我们实现了 Comparable接口 , 所以我们的 Person 如果想排序, 也要实现一个比较器。

       2. Comparator 和 Comparable 的区别

       Comparable

       Comparable 定义在 Person类的内部:

       public class Persion implements Comparable { ..比较Person的大小..},

        因为已经实现了比较器,那么我们的Person现在是一个可以比较大小的对象了,它的比较功能和String完全一样,可以随时随地的拿来比较大小,因为Person现在自身就是有大小之分的。Collections.sort(personList)可以得到正确的结果。

       Comparator

       Comparator 是定义在Person的外部的, 此时我们的Person类的结构不需要有任何变化,如

       public class Person{ String name; int age },

       ç„¶åŽæˆ‘们另外定义一个比较器:

       public PersonComparator implements Comparator() { ..比较Person的大小..},

       åœ¨PersonComparator里面实现了怎么比较两个Person的大小. 所以,用这种方法,当我们要对一个 personList进行排序的时候, 我们除了了要传递personList过去, 还需要把PersonComparator传递过去,因为怎么比较Person的大小是在PersonComparator里面实现的, 如:

       Collections.sort( personList , new PersonComparator() ).

       3. Comparator 和 Comparable 的实例

       Comparable:

       å®žçŽ°Comparable接口要覆盖compareTo方法, 在compareTo方法里面实现比较:

       public class Person implements Comparable {

        String name;

        int age

        public int compareTo(Person another) {

        int i = 0;

        i = name.compareTo(another.name); // 使用字符串的比较

        if(i == 0) { // 如果名字一样,比较年龄, 返回比较年龄结果

        return age - another.age;

        } else {

        return i; // 名字不一样, 返回比较名字的结果.

        }

        }

       }

        这时我们可以直接用 Collections.sort( personList ) 对其排序了.

       Comparator:

       å®žçŽ°Comparator需要覆盖 compare 方法:

       public class Person{

        String name;

        int age

       }

       class PersonComparator implements Comparator {

        public int compare(Person one, Person another) {

        int i = 0;

        i = one.name.compareTo(another.name); // 使用字符串的比较

        if(i == 0) { // 如果名字一样,比较年龄,返回比较年龄结果

        return one.age - another.age;

        } else {

        return i; // 名字不一样, 返回比较名字的结果.

        }

        }

       }

        Collections.sort( personList , new PersonComparator()) 可以对其排序

        4:总结

       ä¸¤ç§æ–¹æ³•å„有优劣, 用Comparable 简单, 只要实现Comparable 接口的对象直接就成为一个可以比较的对象,但是需要修改源代码, 用Comparator 的好处是不需要修改源代码, 而是另外实现一个比较器, 当某个自定义的对象需要作比较的时候,把比较器和对象一起传递过去就可以比大小了, 并且在Comparator 里面用户可以自己实现复杂的可以通用的逻辑,使其可以匹配一些比较简单的对象,那样就可以节省很多重复劳动了。

java中compareToIgnoreCase大

       在Java中,`String`类的`compareToIgnoreCase`方法提供了一种忽略大小写的字符串比较方式。这个方法的核心逻辑在于,它首先将两个字符串中的字符逐个进行比较,如果遇到不同字符,会将它们转换为大写或小写后再进行比较。只有当所有字符都相同或者都转换为同一形式后,才会继续下一个字符。江湖指标源码如果在整个字符串长度范围内找不到相同的字符,那么比较将基于字符串的长度差异。这个方法在`CaseInsensitiveComparator`类中实现,其源码如下:

       String类中的compareToIgnoreCase方法实现是通过逐字符比较,忽略大小写的。它首先计算两个字符串的长度,然后从第一个字符开始比较,如果字符不同,先将两个字符转换为大写或小写,再次比较。如果仍然不同,再转换为小写,然后比较ASCII值。若在字符串长度范围内未找到相同的架设和源码字符,则以字符串长度的差异作为结果返回。这种方法的代码结构清晰,易于理解。

       总的来说,`compareToIgnoreCase`方法是通过字符级别的转换和比较,实现了在比较字符串时忽略大小写的逻辑。

java的Comparable和Comparator区别

       Java编程时,有时会用到`Comparable`和`Comparator`来实现数据的排序。它们都有比较的用途,但有明显的差异,我们先从基础和使用场景开始探讨。

       ### 基础对比

       `Comparable`与`Comparator`都属于接口类型,但前者属于`java.lang`包下,后者则位于`java.util`包中。

       ### 方法对比

       `Comparable`提供了`compareTo`方法,用以比较对象之间的大小关系。而`Comparator`提供了`compare`方法,通常用于对比对象进行排序。`compareTo`和`compare`方法的基本用途相似,但后者更通用,可以为不同类型的对象提供比较逻辑。

       ### 内部与外部比较器

       `Comparable`可以被视为一个对象内部的比较机制,它主要用于实现数据类型的自然排序规则。反之,`Comparator`作为外部比较器,通常用于更复杂或特定的比较逻辑,如排序算法。

       ### 实现与使用场景

       使用`Comparable`时,通常需将实现此接口的类与排序方法结合,直接使用该类创建的实例来进行比较。而借助`Comparator`,可以灵活处理不同类型的数据或定制化的比较逻辑,例如在`Arrays.sort`或`Collections.sort`方法中传入`Comparator`对象。

       在代码设计中,选择`Comparable`还是`Comparator`取决于比较规则的固定性和灵活性需要。若规则易变动,推荐使用`Comparator`,因为它提供独立的比较逻辑,简化了维护和扩展。如`String`和`Integer`等类自带的比较规则比较稳定且符合自然顺序,故往往选择实现`Comparable`。

       深入源码分析,你将了解到接口的具体实现逻辑和行为,这对理解Java编程的核心逻辑至关重要。

java-增强版Comparator进行集合排序null异常处理

       在Java开发中,我们有时会遇到使用增强版Comparator对集合进行排序时遇到null异常的问题。例如,当尝试对一个Student对象列表按照no、age、name和money进行排序时,如果列表中存在未赋值的no或money,就可能导致NullPointerException。为了解决这个问题,我们可以通过查阅Comparator的源码来找到解决策略。

       Comparator类中的Objects.requireNonNull方法确保了比较对象和字段不能为空。针对null值,Comparator提供了两个有用的静态方法:nullsFirst和nullsLast。这两个方法分别将null视为小于或大于非null值,从而避免了排序时的null异常。例如,可以将代码修改为:

       dataList.sort(Comparator.comparing(Student::getNo, Comparator.nullsFirst(String::compareTo).reversed()).thenComparing(Student::getAge)

       .thenComparing(Student::getName).thenComparing(Student::getMoney));

       在这个修改后的代码中,我们首先对no字段使用了nullsFirst,确保了null值的排序位置。Comparator.comparing方法允许我们指定比较函数,而Comparator.thenComparing则支持多字段排序。

       除了这些,Comparator还提供了其他一些方法,如reversed、reverseOrder、naturalOrder等,用于调整排序顺序。比如,Comparator.reverseOrder()会反转排序顺序,Comparator.naturalOrder()则按照对象的自然顺序进行排序。这些方法可以根据实际需求灵活运用。

       总的来说,通过使用Comparator的nullsFirst和nullsLast方法,我们可以有效地处理Java集合排序时的null异常问题,使得代码更加健壮。更多关于Comparator的方法可以参考Java 8的官方开发文档。

public static <T> int binarySearch(List<? extends T> list, T key, Comparator<? super T> c)

       é—®é¢˜1,2其实可以一起讲,就是:你没完全理解==,equals和自动装箱(auto boxing)。仔细看你的comparator的实现return i<j?-1:(i==j?0:1);这句话,注意,你的参数是Integer类型,不是基本类型int,你对类的对象做==操作,是比较它们是否是同一个引用,而不是他们的值是否相等。所以,你用这个比较器做二叉搜索,结果肯定是不可预知的。当<判断失败的时候,后面的==基本不可能是true,这也是为什么if(integers.get(1)==new Integer(1)){ System.out.println("<3>");}这句话不能打印出<3>来。你后面那个实现是正确的,因为类的对象没有<操作和>操作,所以编译器会把Integer对象自动转成int再执行比较。

       é—®é¢˜3:api写的很清楚,当找不到的时候,会返回【-(插入值)-1】,在这个程序里,因为你的比较器写的不对,所以搜索失败,并且它认为插入值是1,这没有问题啊。其实,因为上面说的问题,他返回任何奇奇怪怪的数都有可能。

       å¦‚果你想对两个Integer类的对象比较值,你可以调用i.eqauls(j);或者i.intValue()==j.intValue();intValue方法返回的是Integer对象包装的基本类型的int值

       å¦å¤–,补充一点,不知道是你描述的不好,还是我理解的不对。你说的“根据API文档解释, Comparator只是在search之前对list进行排序。”不对,binarysearch不会排序,而是api要求你在调用binarySearch之前,必须保证传进去的list是已排序了的,你可以调用Collections.sort来实现,也可以自己写排序算法

       =========================================================

       comparator的作用就是比较器啊,告诉排序方法怎样比较两个值谁大谁小或相等,对于Integer,当然比较规则很直观,但是对别的类呢,比如你有一个一个Student类,你希望根据学号来排序,查找,你就需要定义自己的Comparator,当然你也可以让你的类实现Comparable接口,这样你就可以直接调用binarySearch的另外一个不需要比较器的版本

       =========================================================

       æœ‰ç‚¹ä¸æ˜Žç™½ï¼Œä½ æ–°è¡¥å……的问题是针对我哪句话讲的?你贴的API是没错啦,但理解不对。binarySearch不会排序,排序的是sort,binarySearch实现中不会调用sort,这点你自己看源代码就知道了,事实上api表达的也是这个意思(注意,原话是“在进行此调用之前”,意思是用户自己调用sort再调用binarysearch)。你知道pre-condition说法吗?列表已排序状态就是binarySearch方法成功的pre-condition(前置条件),你可以传一个乱序的list进去,没问题,编译器不会报错,但是你得不到自己想要的结果,这个你自己试就知道了。

       =========================================================

       æ±—。。你的意思难道不还是search会调用sort么?否则,如果你承认search不调用sort,那为什么还怀疑comparator会否参与搜索过程?如果search既不排序,又不使用comparator,那这个参数不就没有用了吗?

       æ‰€ä»¥ï¼Œsearch需要comparator,只用来搜索,和排序毫无关系。(你应该知道二叉搜索是怎么执行的吧,那就很清楚为什么需要一个comparator了)