【Java刷题】初始化List应该选择ArrayList还是LinkedList
文章目录
前言
记录java刷题的一个大坑,ArrayList和LinkedList效率差别
题目
解题思路
层次遍历,对每一层,按照排序的结果对数组进行交换,记录交换次数即可。
遇到的坑:ArrayList和LinkedList
这是第319场周赛T3,这题被卡了好久,看了题解以后发现和我的思路一样,但我的代码就是有两个测试用例超时。
经过反复提交测试,发现是初始化List时用ArrayList和LinkedList的区别。
未通过代码
class Solution {
public int minimumOperations(TreeNode root) {
//层次遍历结果
List<List<Integer>> list = levelOrder(root);
int res = 0;
//加上每层按照排序结果交换的次数
for(int i = 0; i < list.size(); i++){
res += getMinswap(list.get(i));
}
return res;
}
List<List<Integer>> levelOrder(TreeNode root){
List<List<Integer>> list = new LinkedList<>();
Queue<TreeNode> queue = new LinkedList<>();
queue.offer(root);
while (!queue.isEmpty()){
int size = queue.size();
List<Integer> tmp = new LinkedList<>();
for(int i = 0; i < size; i++){
TreeNode now = queue.poll();
tmp.add(now.val);
if(now.left != null) queue.offer(now.left);
if(now.right != null) queue.offer(now.right);
}
list.add(tmp);
}
return list;
}
int getMinswap(List<Integer> nums){
if(nums.size() == 1) return 0;
int n = nums.size();
int res = 0;
List<Integer> sortedNums = new LinkedList<>(nums);
Collections.sort(sortedNums);
HashMap<Integer, Integer> map = new HashMap<>();
for(int i = 0; i < sortedNums.size(); i++){
map.put(sortedNums.get(i), i);
}
for(int i = 0; i < n; ){
if(sortedNums.get(i).equals(nums.get(i))){
i++;
continue;
}
Collections.swap(nums, i, map.get(nums.get(i)));
res++;
}
return res;
}
}
通过代码
class Solution {
public int minimumOperations(TreeNode root) {
//层次遍历结果
List<List<Integer>> list = levelOrder(root);
int res = 0;
//加上每层按照排序结果交换的次数
for(int i = 0; i < list.size(); i++){
res += getMinswap(list.get(i));
}
return res;
}
List<List<Integer>> levelOrder(TreeNode root){
List<List<Integer>> list = new ArrayList<>();
Queue<TreeNode> queue = new LinkedList<>();
queue.offer(root);
while (!queue.isEmpty()){
int size = queue.size();
List<Integer> tmp = new ArrayList<>();
for(int i = 0; i < size; i++){
TreeNode now = queue.poll();
tmp.add(now.val);
if(now.left != null) queue.offer(now.left);
if(now.right != null) queue.offer(now.right);
}
list.add(tmp);
}
return list;
}
int getMinswap(List<Integer> nums){
if(nums.size() == 1) return 0;
int n = nums.size();
int res = 0;
List<Integer> sortedNums = new ArrayList<>(nums);
Collections.sort(sortedNums);
HashMap<Integer, Integer> map = new HashMap<>();
for(int i = 0; i < sortedNums.size(); i++){
map.put(sortedNums.get(i), i);
}
for(int i = 0; i < n; ){
if(sortedNums.get(i).equals(nums.get(i))){
i++;
continue;
}
Collections.swap(nums, i, map.get(nums.get(i)));
res++;
}
return res;
}
}
小结
把我原来的LinkedList改成ArrayList就过了。
ArrayList和LinkedList的区别
ArrayList和LinkedList有什么区别:
- ArrayList底层结构是顺序表(基于数组);
LinkList是链表; - ArrayList数据存放在内存空间上;
LinkList不是存放在连续的内存空间上; - ArrayList能够高效的进行“随机访问” ,时间复杂度是O(1);
- LinkList能够高效的进行插入删除,时间复杂度为O(1)。
在刷题的时候应该如何选择
大部分应该选择ArrayList,因为很多时候我们都要遍历数组,此时就要调用get方法访问元素,对于随机访问,ArrayList比LinkedList快。
由于LinkList能够高效的进行插入删除,在任意位置插入操作对应add(int index, E element),删除操作对应remove(int index),所以遇到这两种操作比较多的时候应该用LinkList。这种情况在我刷题的过程中遇到得比较少。
排序效率
Collections.sort()内部是归并排序,下面对ArrayList和LinkedList实现的List做一个简单的排序效率测试。
测试代码
import java.util.*;
public class Main {
public static void main(String[] args) {
int count = 10000000;
List<Integer> listArray = new ArrayList<>();
List<Integer> listLinked = new LinkedList<>();
Random random = new Random();
//生成count个随机值
for (int i = 0; i < count; i++) {
int rand = random.nextInt();
listArray.add(rand);
listLinked.add(rand);
}
//开始计时
long startTime1 = System.currentTimeMillis();
//对ArrayList排序
Collections.sort(listArray);
long endTime1 = System.currentTimeMillis();
long usedTime1 = (endTime1 - startTime1);
//开始计时
long startTime2 = System.currentTimeMillis();
//对LinkedList排序
Collections.sort(listLinked);
long endTime2 = System.currentTimeMillis();
long usedTime2 = (endTime2 - startTime2);
//打印排序时间
System.out.println("ArrayList排序用时:" + usedTime1);
System.out.println("LinkedList排序用时:" + usedTime2);
}
}
测试结果

单从排序效率来看LinkList是比ArrayList高的,然而根据我上面那道题的例子,里面也有用到排序,当我把排序的那个List改为LinkList时仍然无法通过,应该是排完序之后又对数组进行了遍历,而遍历ArrayList是快的。
遍历效率
测试代码
import java.util.*;
public class Main {
public static void main(String[] args) {
int count = 100000;
List<Integer> listArray = new ArrayList<>();
List<Integer> listLinked = new LinkedList<>();
Random random = new Random();
//生成count个随机值
for (int i = 0; i < count; i++) {
int rand = random.nextInt();
listArray.add(rand);
listLinked.add(rand);
}
//开始计时
long startTime1 = System.currentTimeMillis();
//对ArrayList遍历
for(int i = 0; i < listArray.size(); i++){
int a = listArray.get(i);
}
long endTime1 = System.currentTimeMillis();
long usedTime1 = (endTime1 - startTime1);
//开始计时
long startTime2 = System.currentTimeMillis();
//对LinkedList遍历
for(int i = 0; i < listLinked.size(); i++){
int a = listLinked.get(i);
}
long endTime2 = System.currentTimeMillis();
long usedTime2 = (endTime2 - startTime2);
//打印遍历时间
System.out.println("ArrayList遍历用时:" + usedTime1);
System.out.println("LinkedList遍历用时:" + usedTime2);
}
}
测试结果

由此可见,ArrayList的遍历效率比LinkList高,而且超出LinkList的部分大于LinkList排序超出ArrayList的部分。因此在排序和遍历操作都存在的情况,还是应该选择ArrayList。
结论
刷题的时候首选ArrayList。遇到add(int index, E element)和remove(int index)操作频繁的时候,再改用LinkList。
【Java刷题】初始化List应该选择ArrayList还是LinkedList的更多相关文章
- 牛客网Java刷题知识点之ArrayList 、LinkedList 、Vector 的底层实现和区别
不多说,直接上干货! 这篇我是从整体出发去写的. 牛客网Java刷题知识点之Java 集合框架的构成.集合框架中的迭代器Iterator.集合框架中的集合接口Collection(List和Set). ...
- 牛客网Java刷题知识点之Map的两种取值方式keySet和entrySet、HashMap 、Hashtable、TreeMap、LinkedHashMap、ConcurrentHashMap 、WeakHashMap
不多说,直接上干货! 这篇我是从整体出发去写的. 牛客网Java刷题知识点之Java 集合框架的构成.集合框架中的迭代器Iterator.集合框架中的集合接口Collection(List和Set). ...
- 牛客网Java刷题知识点之Java 集合框架的构成、集合框架中的迭代器Iterator、集合框架中的集合接口Collection(List和Set)、集合框架中的Map集合
不多说,直接上干货! 集合框架中包含了大量集合接口.这些接口的实现类和操作它们的算法. 集合容器因为内部的数据结构不同,有多种具体容器. 不断的向上抽取,就形成了集合框架. Map是一次添加一对元素. ...
- 牛客网Java刷题知识点之为什么HashMap不支持线程的同步,不是线程安全的?如何实现HashMap的同步?
不多说,直接上干货! 这篇我是从整体出发去写的. 牛客网Java刷题知识点之Java 集合框架的构成.集合框架中的迭代器Iterator.集合框架中的集合接口Collection(List和Set). ...
- 牛客网Java刷题知识点之HashMap的实现原理、HashMap的存储结构、HashMap在JDK1.6、JDK1.7、JDK1.8之间的差异以及带来的性能影响
不多说,直接上干货! 福利 => 每天都推送 欢迎大家,关注微信扫码并加入我的4个微信公众号: 大数据躺过的坑 Java从入门到架构师 人工智能躺过的坑 ...
- 牛客网Java刷题知识点之TCP、UDP、TCP和UDP的区别、socket、TCP编程的客户端一般步骤、TCP编程的服务器端一般步骤、UDP编程的客户端一般步骤、UDP编程的服务器端一般步骤
福利 => 每天都推送 欢迎大家,关注微信扫码并加入我的4个微信公众号: 大数据躺过的坑 Java从入门到架构师 人工智能躺过的坑 Java全栈大联盟 ...
- 牛客网Java刷题知识点之泛型概念的提出、什么是泛型、泛型在集合中的应用、泛型类、泛型方法、泛型接口、泛型限定上限、泛型限定下限、 什么时候使用上限?泛型限定通配符的体现
不多说,直接上干货! 先来看个泛型概念提出的背景的例子. GenericDemo.java package zhouls.bigdata.DataFeatureSelection; import ja ...
- 牛客网Java刷题知识点之构造函数可以调用一般函数,但是一般函数不可以直接调用构造函数
不多说,直接上干货! 通过 牛客网Java刷题知识点之构造函数是什么.一般函数和构造函数什么区别呢.构造函数的重载.构造函数的内存图解 我们对构造函数有了一个比较清楚的认识,当我们在创建对象时,我们会 ...
- 牛客网Java刷题知识点之构造函数与set方法、与类名同名的一般方法、构造函数中有return语句
不多说,直接上干货! 通过 牛客网Java刷题知识点之构造函数是什么.一般函数和构造函数什么区别呢.构造函数的重载.构造函数的内存图解 我们对构造函数有了一个比较清楚的认识,当我们在创建对象时,我们会 ...
- 牛客网Java刷题知识点之关键字static、static成员变量、static成员方法、static代码块和static内部类
不多说,直接上干货! 牛客网Java刷题知识点之关键字static static代表着什么 在Java中并不存在全局变量的概念,但是我们可以通过static来实现一个“伪全局”的概念,在Java中st ...
随机推荐
- 『现学现忘』Git分支 — 41、分支基本操作(二)
目录 6.新建一个分支并且使分支指向指定的提交对象 7.思考: 8.项目分叉历史的形成 9.分支的总结 提示:接上篇 6.新建一个分支并且使分支指向指定的提交对象 使用命令:git branch br ...
- 浅谈API和SDK的区别
首先了解一下他们的定义 API:application program interface 应用程序接口 通常表示一些事先定义好的函数,为了向外部提供一组功能的实现,实现和其他软件的交互 SDK:so ...
- 原生Ajax处理文件流
在通过Ajax处理请求时,可能会遇到需要下载文件的情况,这里简要的说明下处理方法. let downloadFile = document.getElementById("downloadI ...
- LAPM概述及配置
一.LAMP概述 1.1LAMP的概念 LAMP架构是目前成熟的企业网站应用模式之一,指的是协同工作的一整套系统和相关软件,能够提供动态web站点服务及其应用开发环境 LAMP是一个缩写词,具体包括L ...
- HTML躬行记(4)——Web音视频基础
公司目前的业务会接触比较多的音视频,所以有必要了解一些基本概念. 文章涉及的一些源码已上传至 Github,可随意下载. 一.基础概念 本节音视频的基础概念摘自书籍<FFmpeg入门详解 音视频 ...
- 记一次多个Java Agent同时使用的类增强冲突问题及分析
摘要:Java Agent技术常被用于加载class文件之前进行拦截并修改字节码,以实现对Java应用的无侵入式增强. 本文分享自华为云社区<记一次多个JavaAgent同时使用的类增强冲突问题 ...
- java学习之爬虫
0x00前言 对比与Python的爬虫机制和java的爬虫机制来详解一下java的爬虫,对于一般性的需求无论java还是python都可以胜任. 如需要模拟登陆.对抗防采集选择python更方便些,如 ...
- 基于SpERT的中文关系抽取
SpERT_chinese 基于论文SpERT: "Span-based Entity and Relation Transformer"的中文关系抽取,同时抽取实体.实体类别和关 ...
- Go语言核心36讲40
我相信,经过上一次的学习,你已经对strings.Builder和strings.Reader这两个类型足够熟悉了. 我上次还建议你去自行查阅strings代码包中的其他程序实体.如果你认真去看了,那 ...
- Spring Security(4)
您好,我是湘王,这是我的博客园,欢迎您来,欢迎您再来- 前面的方法中,除了login()方法能成功,另外两个都失败,并不是因为代码问题,而是Spring Security默认是通过Web页面来实现页面 ...