慎用ArrayList的contains方法,使用HashSet的contains方法代替

ArrayList的contains到了十万级别,就基本上很慢了,HashSet到了百万级的数据也特别快,而且是快得特别多。

ArrayList的contains方法与HashSet的contains方法的区别?

​ ArrayList的contains方法会直接判断元素的内存地址是否相同,HashSet的contains方法会先判断元素的hashCode是否相同,如果不同则返回false,如果相同再用equals判断,因此HashSet的contains方法的方法效率会更高。

ArrayList.contains()

ArrayList的contains()方法会调用其indexOf()方法,在indexOf()方法里边,有一个for循环。

所以,ArrayList的contains()方法的时间复杂度是O(n)

该方法通过遍历数据和比较元素的方式来判断是否存在给定元素。当ArrayList中存放的元素非常多时,这种实现方式来判断效率将非常低。

public boolean contains(Object o) {
 return indexOf(o) >= 0;
}

public int indexOf(Object o) {
 if (o == null) {
     for (int i = 0; i < size; i++)
         if (elementData[i]==null)
             return i;
 } else {
     for (int i = 0; i < size; i++)
         if (o.equals(elementData[i]))
             return i;
 }
 return -1;
}    

HashSet.contains()

既然ArrayList的contains()方法存在性能问题,那么就应该寻找改进的办法。这里推荐使用HashSet来代替ArrayList。

下面介绍HashSet的contains()方法的实现过程:

注:HashSet将元素存放在HashMap中(HashMap的key)

public boolean contains(Object o) {
    return map.containsKey(o);
}

contains()方法调用HashMap的containsKey()方法

/**
* 检查是否包含key
* 如果key有对应的节点对象,则返回ture,不关心节点对象的值是否为空
*/
public boolean containsKey(Object key) {
    // 调用getNode方法来获取键值对,如果没有找到返回false,找到了就返回ture
    return getNode(hash(key), key) != null; //真正的查找过程都是通过getNode方法实现的
}

containsKey()方法调用getNode()方法。在该方法中,首先根据key计算hash值,然后从HashMap中取出该hash值对应的链表(链表的元素个数将很少),再通过变量该链表判断是否存在给定值。这种实现方式效率将比ArrayList的实现方法效率高非常多。

/**
* 该方法是Map.get方法的具体实现
* 接收两个参数
* @param hash key的hash值,根据hash值在节点数组中寻址,该hash值是通过hash(key)得到的,可参见:hash方法解析
* @param key key对象,当存在hash碰撞时,要逐个比对是否相等
* @return 查找到则返回键值对节点对象,否则返回null
*/
final Node<K,V> getNode(int hash, Object key) {
    Node<K,V>[] tab; Node<K,V> first, e; int n; K k; // 声明节点数组对象、链表的第一个节点对象、循环遍历时的当前节点对象、数组长度、节点的键对象
    // 节点数组赋值、数组长度赋值、通过位运算得到求模结果确定链表的首节点
    if ((tab = table) != null && (n = tab.length) > 0 &&
        (first = tab[(n - 1) & hash]) != null) {
        if (first.hash == hash && // 首先比对首节点,如果首节点的hash值和key的hash值相同 并且 首节点的键对象和key相同(地址相同或equals相等),则返回该节点
            ((k = first.key) == key || (key != null && key.equals(k))))
            return first; // 返回首节点

        // 如果首节点比对不相同、那么看看是否存在下一个节点,如果存在的话,可以继续比对,如果不存在就意味着key没有匹配的键值对    
        if ((e = first.next) != null) {
            // 如果存在下一个节点 e,那么先看看这个首节点是否是个树节点
            if (first instanceof TreeNode)
                // 如果是首节点是树节点,那么遍历树来查找
                return ((TreeNode<K,V>)first).getTreeNode(hash, key); 

            // 如果首节点不是树节点,就说明还是个普通的链表,那么逐个遍历比对即可    
            do {
                if (e.hash == hash &&
                    ((k = e.key) == key || (key != null && key.equals(k)))) // 比对时还是先看hash值是否相同、再看地址或equals
                    return e; // 如果当前节点e的键对象和key相同,那么返回e
            } while ((e = e.next) != null); // 看看是否还有下一个节点,如果有,继续下一轮比对,否则跳出循环
        }
    }
    return null; // 在比对完了应该比对的树节点 或者全部的链表节点 都没能匹配到key,那么就返回null
}

实例验证

测试ArrayList
public static void main(String[] args) {
    ArrayList<String> arrayList = new ArrayList<>();
    // 存入100000个数据
    for (int i = 0; i < 100000; i++) {
        arrayList.add("test" + i);
    }

    // 验证300000个数据(其中200000不存在)		
    long beginTime = System.currentTimeMillis();		for (int i = 0; i < 300000; i++) {
        arrayList.contains("test" + i);
    }
    long endTime = System.currentTimeMillis();

    System.out.println("cost time: " + (endTime - beginTime) + "ms");
}

打印结果:
cost time: 182210ms

测试HashSet
public static void main(String[] args) {

    Set<String> hashSet = new HashSet<>();
    // 存入100000个数据
    for (int i = 0; i < 100000; i++) {
        hashSet.add("test" + i);
    }

    // 验证300000个数据(其中200000不存在)
    long beginTime = System.currentTimeMillis();
    for (int i = 0; i < 300000; i++) {
        hashSet.contains("test" + i);
    }
    long endTime = System.currentTimeMillis();

    System.out.println("cost time: " + (endTime - beginTime) + "ms");
}

打印结果:
cost time: 49ms

总结

通过以上实例可以看出,使用ArrayList的contains()耗时是使用HashSet的contains()方法的3000多倍。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
### 回答1: Java中的ArrayList类提供了contains()方法,用于判断列表中是否包含指定的元素。该方法返回一个布尔值,如果列表中包含指定元素,则返回true,否则返回false。 使用方法如下: 1. 创建一个ArrayList对象,例如: ArrayList<String> list = new ArrayList<String>(); 2. 向列表中添加元素,例如: list.add("apple"); list.add("banana"); list.add("orange"); 3. 使用contains()方法判断列表中是否包含指定元素,例如: boolean result = list.contains("banana"); if (result) { System.out.println("列表中包含banana"); } else { System.out.println("列表中不包含banana"); } 输出结果为: 列表中包含banana 注意:contains()方法判断元素是否相等是通过equals()方法实现的,因此如果列表中包含自定义对象,需要重写equals()方法。 ### 回答2: Java中的ArrayList类是一种可变数组,支持动态添加、删除和修改元素,并且内置了许多实用的方法来操作数组。其中,contains()方法ArrayList类中非常常用的方法之一。 contains()方法用于判断ArrayList中是否包含某个元素。它的语法如下: ``` public boolean contains(Object o) ``` 其中,Object o是需要判断的元素。如果ArrayList中包含o,那么会返回true,否则返回false。 下面是一个简单的例子: ``` import java.util.ArrayList; public class Main { public static void main(String[] args) { ArrayList<String> list = new ArrayList<String>(); list.add("apple"); list.add("banana"); list.add("orange"); System.out.println("List contains apple: " + list.contains("apple")); System.out.println("List contains kiwi: " + list.contains("kiwi")); } } ``` 运行结果: ``` List contains apple: true List contains kiwi: false ``` 从上面的例子中可以看出,contains()方法首先需要ArrayList中包含元素,然后才能判断是否相等。在判断相等时,contains()方法会自动使用equals()方法来进行比较。 可以将contains()方法应用于其他类型的ArrayList,如Integer、Double等。 但是需要注意,对于自定义对象,需要正确实现equals()方法。 在实际开发中,contains()方法可以用来判断ArrayList是否包含指定元素,从而避免重复元素的添加,同时也可以应用于集合查找等问题。 ### 回答3: Java中的ArrayList是一个动态数组,它可以存储任意类型的对象。contains方法ArrayList提供的一个常用方法,它用于判断指定元素是否包含在ArrayList中。 contains方法的返回值为boolean类型,如果ArrayList包含指定元素,返回true,否则返回false。contains方法的实现原理是遍历整个ArrayList,逐个比较元素是否相等,因此其时间复杂度为O(n),不适合对大规模数据进行查找。 contains方法可以用于判断对象是否存在于ArrayList中,其比较过程是使用对象的equals方法来进行比较。如果需要自定义对象比较的方式,可以重写对象的equals方法。 需要注意的是,如果ArrayList中存储的是基本数据类型,如int、double等,contains方法将无法进行比较。此时可以使用包装类,如Integer、Double等来存储基本数据类型,然后使用contains方法进行比较。 同时,如果ArrayList中存储的对象是可变对象,如List或Map等,contains方法比较的是对象的引用,而不是对象中的内容。此时需要重写对象的equals方法来进行比较。 最后,需要特别注意的是,在使用contains方法时需要确保对象正确实现了equals方法和hashCode方法,否则可能会出现意想不到的结果,例如当对象作为HashMap中的key时,无法正常获取对应的value。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

itmengge

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值