电子书 – Spring Cloud 指南 – NPI EA (分类=Spring Cloud)
announcement - icon

让我们开始使用 Spring Cloud 的微服务架构

>> 加入 Pro 并下载电子书

电子书 – Mockito – NPI EA (标签 = Mockito)
announcement - icon

模拟是单元测试的重要组成部分,Mockito 库使编写 清晰直观的单元测试 变得容易,用于您的 Java 代码。

通过我们的 Mockito 指南 开始模拟,并改进您的应用程序测试

下载电子书

电子书 – Java 并发 – NPI EA (分类=Java 并发)
announcement - icon

在应用程序中处理并发可能是一个棘手的过程,其中包含许多 潜在的陷阱。 扎实的掌握基本知识将有助于最大程度地减少这些问题。

通过我们的 Java 并发 指南开始了解多线程应用程序

>> 下载电子书

电子书 – 响应式 – NPI EA (分类=响应式)
announcement - icon

Spring 5 增加了对使用 Spring WebFlux 模块进行响应式编程的支持,此支持自那时起不断改进。 开始使用 Reactor 项目基础知识和 Spring Boot 中的响应式编程

>> 加入 Pro 并下载电子书

电子书 – Java Streams – NPI EA (分类=Java Streams)
announcement - icon

自从 Java 8 引入以来,Stream API 已成为 Java 开发的基础。 基本操作,例如迭代、过滤、映射元素序列,使用起来看似很简单。

但这些也可能被过度使用并陷入一些常见陷阱。

更好地了解 Stream 的工作方式 以及如何将其与其他语言功能结合使用,请查看我们关于 Java Streams 的指南

>> 加入 Pro 并下载电子书

电子书 – Jackson – NPI EA (分类=Jackson)
announcement - icon

用 Jackson 正确处理 JSON

下载电子书

电子书 – HTTP 客户端 – NPI EA (分类=Http 客户端)
announcement - icon

充分利用 Apache HTTP 客户端

下载电子书

电子书 – Maven – NPI EA (分类 = Maven)
announcement - icon

开始使用 Apache Maven

下载电子书

电子书 – 持久化 – NPI EA (分类=持久化)
announcement - icon

您在努力实现正确的持久化层 Spring 吗?

探索电子书

电子书 – RwS – NPI EA (分类=Spring MVC)
announcement - icon

使用 Spring 构建 REST API 吗?

下载电子书

课程 – LS – NPI EA (分类=Jackson)
announcement - icon

通过 Learn Spring 课程开始学习 Spring 和 Spring Boot

>> 学习 SPRING
课程 – RWSB – NPI EA (分类=REST)
announcement - icon

通过构建一个完整的 REST API,深入了解 Spring Boot 3 和 Spring 6,使用该框架

>> 全新的“REST With Spring Boot”

课程 – LSS – NPI EA (分类=Spring Security)
announcement - icon

是的,Spring Security 可能很复杂,从核心内的更高级功能到框架中深入的 OAuth 支持。

我将安全材料构建为 两个完整的课程 - 核心和 OAuth,以针对这些更复杂的场景进行实践。 我们探索何时以及如何使用每个功能,并 在后台项目中对其进行编码

您可以在这里探索该课程

>> 学习 Spring Security

课程 – LSD – NPI EA (标签=Spring Data JPA)
announcement - icon

Spring Data JPA 是处理 JPA 复杂性的绝佳方式,它具有 Spring Boot 的强大简洁性

通过引导式参考课程开始使用 Spring Data JPA

>> 查看课程

合作伙伴 – Moderne – NPI EA (类别=Spring Boot)
announcement - icon

使用 OpenRewrite 安全且自动地重构 Java 代码。

手动重构大型代码库既缓慢、有风险,又容易拖延。OpenRewrite 应运而生。这个用于大规模、自动化代码转换的开源框架可以帮助团队安全、一致地进行现代化改造。

每个月,OpenRewrite 的创建者和维护者 Moderne 都会举办现场、实践培训课程——一个面向初学者,一个面向经验丰富的用户。您将了解配方的运作方式、如何将其应用于项目,以及如何自信地进行代码现代化改造。

参加下一次课程,带来您的问题,并学习如何自动化通常会占用您 sprint 时间的工作。

合作伙伴 – LambdaTest – NPI EA (类别=测试)
announcement - icon

回归测试是发布流程中的重要步骤,以确保新代码不会破坏现有功能。随着代码库的不断发展,我们希望频繁运行这些测试,以便尽早发现任何问题。

确保这些测试以自动化的方式频繁运行的最佳方法当然是将其包含在 CI/CD 管道中。 这样,每次向仓库提交代码时,回归测试将自动执行。

在本教程中,我们将学习如何使用 Selenium 创建回归测试,然后使用 GitHub Actions 将它们包含在我们的管道中,在 LambdaTest 云网格上运行

>> 如何使用 GitHub Actions 运行 Selenium 回归测试

课程 – LJB – NPI EA (类别 = Core Java)
announcement - icon

通过编码方式构建 Java 的坚实、实用的基础

>> 学习 Java 基础

1. 简介

LinkedListListDeque 接口的双向链表实现。它实现了所有可选的列表操作,并允许所有元素(包括 null)。

2. 特性

以下是 LinkedList 的重要属性

  • 对列表进行索引的操作将从列表的开头或结尾遍历,具体取决于哪个更靠近指定的索引
  • 它不是 同步的
  • 它的 IteratorListIterator 迭代器是 快速失败的(这意味着在迭代器的创建之后,如果列表被修改,则会抛出 ConcurrentModificationException
  • 每个元素都是一个节点,它保留对下一个和前一个节点的引用
  • 它维护插入顺序

虽然 LinkedList 不是同步的,但我们可以通过调用 Collections.synchronizedList 方法来获取其同步版本,例如

List list = Collections.synchronizedList(new LinkedList(...));

3. 与 ArrayList 的比较

虽然它们都实现了 List 接口,但它们具有不同的语义 - 这将肯定影响选择使用哪个接口的决定。

3.1. 结构

ArrayList 是一个基于索引的数据结构,由 Array 支持。它以 O(1) 的性能提供对其元素的随机访问。

另一方面,LinkedList 将其数据存储为元素列表,并且每个元素都链接到其前一个和下一个元素。在这种情况下,查找项目的操作执行时间等于 O(n)。

3.2. 操作

在一个 LinkedList 中插入、添加和删除项目更快,因为不需要调整数组大小或更新索引,当在集合内的某个任意位置添加元素时,只需要更改周围元素的引用。

3.3. 内存使用

LinkedListArrayList 消耗更多的内存,因为 LinkedList 中的每个节点都存储两个引用,一个用于其前一个元素,一个用于其下一个元素,而 ArrayList 仅持有数据及其索引。

4. 初始化 LinkedList 的不同方法

我们可以使用两个构造函数之一初始化 LinkedList。此外,我们使用构造函数来初始化类的新的实例。

4.1. 初始化一个空的 LinkedList

我们使用空的 LinkedList() 构造函数来初始化一个空的 LinkedList

让我们演示一个使用 泛型String 参数化的 LinkedList 的示例。

让我们使用 JUnit 5 测试来验证空的 LinkedList

@Test
public void whenInitializingLinkedList_ShouldReturnEmptyList() throws Exception {
    LinkedList<String> linkedList=new LinkedList<String>();     
    Assertions.assertTrue(linkedList.isEmpty());
}

相应地,我们可以将 String 元素添加到此列表中

linkedList.addFirst("one");
linkedList.add("two");
linkedList.add("three");

同样,我们可以初始化任何类类型的 LinkedList

4.2. 从集合初始化 LinkedList

当我们需要初始化一个 LinkedList 时,其元素是给定 Collection 的元素时,我们也可以使用 LinkedList(Collection<? extends E> c) 构造函数。它以 Collection 的迭代器返回它们的顺序添加元素。

让我们演示一个使用泛型参数化为 IntegerLinkedList 的例子。 但是,首先,我们创建一个 Collection,用作我们在调用构造函数时使用的参数。 让我们创建一个参数化为 IntegerArrayList 类型的 Collection,并使用泛型

ArrayList<Integer> arrayList = new ArrayList<Integer>(3);

 arrayList.add(Integer.valueOf(1));
 arrayList.add(Integer.valueOf(2));
 arrayList.add(Integer.valueOf(3));

随后,让我们调用构造函数来初始化一个 LinkedList

LinkedList<Integer> linkedList=new LinkedList<Integer>(arrayList);

再次,让我们使用 JUnit 5 测试来验证 LinkedList 从其构造的 ArrayList 中派生其元素

@Test
public void whenInitializingListFromCollection_ShouldReturnCollectionsElements() throws Exception {
    ArrayList<Integer> arrayList=new ArrayList<Integer>(3);
        
    arrayList.add(Integer.valueOf(1));
    arrayList.add(Integer.valueOf(2));
    arrayList.add(Integer.valueOf(3));
 
    LinkedList<Integer> linkedList=new LinkedList<Integer>(arrayList);

    Object[] linkedListElements = linkedList.toArray();
    Object[] collectionElements = arrayList.toArray();

    Assertions.assertArrayEquals(linkedListElements,collectionElements);
}

同样,我们可以使用任何 Java Collection 来初始化 LinkedList。 因此,从 Collection 初始化 LinkedList 的元素是 Collection 元素的类型。

5. 用法

以下是一些代码示例,展示了如何使用 LinkedList

5.1. 创建

LinkedList<Object> linkedList = new LinkedList<>();

5.2. 添加元素

除了标准的 add()addAll() 方法外,LinkedList 还实现了 ListDeque 接口,您还可以找到 addFirst()addLast(),它们分别在开头或结尾添加一个元素。

5.3. 移除元素

与元素添加类似,此列表实现提供了 removeFirst()removeLast()。

此外,还有便捷的方法,例如 removeFirstOccurence()removeLastOccurence(),它们返回一个 boolean(如果集合包含指定的元素,则返回 true)。

5.4. 队列操作

Deque 接口提供类似队列的行为(实际上,Deque 扩展了 Queue 接口)

linkedList.poll();
linkedList.pop();

这些方法检索第一个元素并将其从列表中删除。

poll()pop() 的区别在于,pop 在空列表上会抛出 NoSuchElementException(),而 poll 返回 null。 pollFirst()pollLast() API 也可用。

例如,以下是如何使用 push API 的

linkedList.push(Object o);

它将元素作为集合的头部插入。

LinkedList 还有许多其他方法,其中大部分对于已经使用过 Lists 的用户来说应该很熟悉。 Deque 提供的其他方法可能是“标准”方法的便捷替代方案。

完整的文档可以在 这里 找到。

6. 在 LinkedList 的特定位置插入元素

当我们想在 LinkedList 的特定位置添加元素时,我们有几种方法选择

方法 描述
addFirst(E e) 在列表的开头添加一个元素
addLast(E e) 在列表的末尾添加一个元素
add(E e) 在列表的末尾添加一个元素
add(int index, E element) 在列表的索引位置 i 添加一个元素

让我们演示如何使用这些方法中的每一个。 此外,让我们初始化一个空列表

LinkedList<String> linkedList=new LinkedList<String>();

随后,让我们使用方法 addFirst(E e) 添加第一个元素

linkedList.addFirst("one");

虽然我们可以通过重复调用 add(E e) 方法来构建列表,但我们想演示在特定位置添加元素的各种选项。 因此,让我们使用方法 addLast(E e) 在列表末尾添加最后一个元素

linkedList.addLast("three");

让我们调用 add(E e) 方法将一个元素添加到到目前为止构建的列表的末尾

linkedList.add("four");

LinkedList 使用基于 0 的索引,这意味着第一个元素位于索引 0,第二个元素位于索引 1,第三个元素位于索引 2,依此类推。为了创建一个序列,让我们使用 add(int index, E element) 在索引位置 1 处添加元素“two”。

linkedList.add(1,"two");

让我们使用 JUnit 5 测试来验证我们是否在 LinkedList 中添加了元素,且位于特定的各自位置。

@Test
public void whenAddingElementsInLinkedListAtSpecificPosition_ShouldReturnElementsInProperSequence() throws Exception {
    LinkedList<String> linkedList=new LinkedList<String>();
        
    linkedList.addFirst("one");
    linkedList.addLast("three");
    linkedList.add("four");
    linkedList.add(1,"two");

    Object[] linkedListElements = linkedList.toArray();
    Object[] expectedListElements = {"one","two","three","four"};

    Assertions.assertArrayEquals(linkedListElements,expectedListElements);
}

JUnit 测试应该通过,从而验证我们在 LinkedList 中添加了元素到特定的位置。

7. 将数组转换为 LinkedList

在某些情况下,我们可能以数组的形式拥有数据,并且需要将其作为 LinkedList 来处理,以便进行高效的插入或利用 LinkedList 提供的 Deque 操作。在 Java 中,将数组转换为 LinkedList 非常简单。

首先,让我们从一个包含元素的示例数组开始。

String[] array = { "apple", "banana", "cherry", "date" };

我们可以通过 **将数组传递给 Arrays.asList() 方法,然后从结果 List 创建一个新的 LinkedList** 来将此数组转换为 LinkedList

List<String> list = Arrays.asList(array);  // Convert array to List
LinkedList<String> linkedList = new LinkedList<>(list);

现在 linkedList 包含来自原始数组的所有元素,我们可以对其使用任何 LinkedList 操作。

将数组转换为 LinkedList 的另一种方法是 **使用 Collections.addAll() 方法**。此方法提供了一种简单的方法,可直接将数组中的所有元素添加到 LinkedList 中。

首先,我们创建一个空的 LinkedList。然后,我们使用 Collections.addAll() 方法将数组中的每个元素添加到 LinkedList 中。

LinkedList<String> linkedList = new LinkedList<>();
Collections.addAll(linkedList, array);

完成此操作后,linkedList 将包含来自数组的所有元素,并且它会保留顺序。

8. 结论

ArrayList 通常是默认的 List 实现。

然而,在某些用例中,使用 LinkedList 会更合适,例如优先考虑恒定插入/删除时间(例如,频繁的插入/删除/更新),而不是恒定访问时间和有效的内存使用。

支持本文的代码可在 GitHub 上获取。 一旦你Baeldung Pro 会员 身份登录,就开始学习并在项目上进行编码。
Baeldung Pro – NPI EA (类别 = Baeldung)
announcement - icon

Baeldung Pro 具有完全无广告以及最终具有深色模式,提供干净的学习体验

>> 探索干净的 Baeldung

一旦早期采用者的席位全部用完,价格将上涨并保持在每年 33 美元。

电子书 – HTTP 客户端 – NPI EA (类别=HTTP 客户端)
announcement - icon

Apache HTTP Client 是一个非常强大的库,适用于简单和高级用例,在测试 HTTP 端点时尤其适用。 查看我们的指南,涵盖基本请求和响应处理,以及安全性、Cookie、超时等。

>> 下载电子书

电子书 – Java 并发 – NPI EA (分类=Java 并发)
announcement - icon

在应用程序中处理并发可能是一个棘手的过程,其中包含许多 潜在的陷阱。 扎实的掌握基本知识将有助于最大程度地减少这些问题。

通过我们的 Java 并发 指南开始了解多线程应用程序

>> 下载电子书

电子书 – Java Streams – NPI EA (分类=Java Streams)
announcement - icon

自从 Java 8 引入以来,Stream API 已成为 Java 开发的基础。 基本操作,例如迭代、过滤、映射元素序列,使用起来看似很简单。

但这些也可能被过度使用并陷入一些常见陷阱。

更好地了解 Stream 的工作方式 以及如何将其与其他语言功能结合使用,请查看我们关于 Java Streams 的指南

>> 加入 Pro 并下载电子书

电子书 – 持久化 – NPI EA (分类=持久化)
announcement - icon

您在努力实现正确的持久化层 Spring 吗?

探索电子书

课程 – LS – NPI EA (类别=REST)

announcement - icon

从 Spring Boot 开始,通过 Learn Spring 课程了解核心 Spring。

>> 查看课程

合作伙伴 – Moderne – NPI EA (标签=重构)
announcement - icon

现代 Java 团队行动迅速——但代码库并不总是跟上。 框架会发生变化,依赖关系会漂移,技术债务会累积,直到它开始拖慢交付速度。 OpenRewrite 就是为此而构建的:一个开源重构引擎,可在保持开发人员意图不变的同时自动化重复的代码更改。

由 Moderne 的 OpenRewrite 创建者和维护者领导的每月培训系列,将介绍实际的迁移和现代化模式。 无论您是重构配方的新手,还是准备编写自己的配方,您都将学习以安全且可扩展的方式进行重构的实用方法。

如果您曾经希望重构感觉像编写代码一样自然——并且一样快速——这是一个很好的起点

电子书 Jackson – NPI EA – 3 (类别 = Jackson)
© .