Java集合类ArrayList应用 | 二维数组的集合类表示与杨辉三角实现

简介: 这是一个关于LeetCode第118题“杨辉三角”的问题解答摘要。题目要求生成一个杨辉三角的前n行,其中每一行都是由前一行的元素按规则生成的。杨辉三角的规律是:每一行的第一个和最后一个数是1,其他数是其上方两数之和。

一、题干

🔗力扣:118. 杨辉三角



二、题解


1. 思路


我们知道杨辉三角的规律是:


  1. 每一行的第一列和它的最后一列上的数均为1.


  1. 除此之外,每个数是它的左上方与右上方的数之和。如果用 i 代表行, j 代表列,则有:

      [ i ][ j ] 等于 [ i-1 ][ j-1 ] + [ i-1 ][ j ]




本题的关键在于对ArrayList集合类的理解。在力扣上,该题的默认代码模板是这样的:


class Solution {
    public List<List<Integer>> generate(int numRows) {
        ...
    }
}


因此,要完成这道题,我们需要先理清该函数的返回值 List<List<Integer>> 代表什么含义。


  • 从外层的List可以看出,有一个List集合类容器,且容器中的每个元素的数据类型仍是List。本文中我们采用ArrayList集合类实现,因此有以下示意图:



  • 而里层List中的元素类型是Integer,则有如下示意图:



由此可以看出,List<List<Integer>> 所代表的正是一个二维数组。用 i 表示横坐标(行),j 表示纵坐标(列),我们可以通过ArrayList中的get()方法获取二维数组 [i][j] 位置处的元素值。我们将步骤分解,得到具体操作如下:


首先,我们需要创建出外层(即上图中左侧)的ArrayList,用ret表示:



List<List<Integer>> ret = new ArrayList<>();


ret中的每一个元素也是ArrayList类型。我们接下来要做的,就是依照规律逐行将杨辉三角中的数值放入“二维数组”中。第一行的1没有前一行或前一列,它是特殊的,需要单独先把1放进去:


1.//创建第一行
List<Integer> firstRow = new ArrayList<>();
//将第一行的元素1放入第一行的第一个
firstRow.add(1);
//将第一行作为ret的首元素加入ret中
ret.add(firstRow);



然后再插入剩下的数值。用for循环控制每行要完成的事,i 代表行数,由于第一行已经提前单独插入过数值,因此 i 从第二行开始,也就是下标为1处开始。


每一行,都创建一个ArrayList顺序表,用curRow表示。每一行的第一个元素和最后一个元素一定是1,直接通过curRow.add(1)插入即可。两个1中间部分的元素通过插入上方元素之和计算。可以新创建一个变量prevRow表示curRow的前一行:



List<Integer> prevRow = ret.get(i-1);
//获取元素并将结果插入curRow
curRow.add(prevRow.get(j-1) + prevRow.get(j));


也可以不用prevRow,直接插入:


curRow.add(ret.get(i-1).get(j-1) + ret.get(i-1).get(j));


for循环的最后,还需要把当前行插入ret中。

ret.add(curRow);


示意图及合并代码如下:





        for (int i = 1; i < numRows; i++) {
            List<Integer> curRow = new ArrayList<>();
            curRow.add(1);
            for (int j = 1; j < i; j++) {
                curRow.add(ret.get(i-1).get(j-1) + ret.get(i-1).get(j));
            }
            curRow.add(1);
            ret.add(curRow);
        }

2. 完整代码


class Solution {
    public List<List<Integer>> generate(int numRows) {
        List<List<Integer>> ret = new ArrayList<>();
        
        List<Integer> firstRow = new ArrayList<>();
        firstRow.add(1);
        ret.add(firstRow);
        for (int i = 1; i < numRows; i++) {
            List<Integer> curRow = new ArrayList<>();
            curRow.add(1);
            for (int j = 1; j < i; j++) {
                curRow.add(ret.get(i-1).get(j-1) + ret.get(i-1).get(j));
            }
            curRow.add(1);
            ret.add(curRow);
        }
        return ret;
    }
}


三、总结


  1. 杨辉三角的数值规律。


  1. Link<Link<Integer>>可以看作是一个二维数组,且该“二维数组”中的每个元素类型是Integer。分解来看即是一个ArrayList中的每个元素类型仍是一个ArrayList,且后者中的元素类型是Integer。




  1.  通过集合类的get方法获取元素。ret.get(i)指获取ret中第i个元素(ret中的元素类型仍是ArrayList,所以也就是获取第i行),ret.get(i).get(j)指获取行中第j个的元素。


  1. 关键的插入方法是,每一行都创建新的ArrayList对象curRow,先add顺序表curRow中的内容,最后再将该行作为ret的一个元素add到顺序表ret中。通过:




curRow.add(...);

ret.add(curRow);

即可搞定向该“二维数组”中插入元素。


相关文章
|
27天前
|
人工智能 安全 Java
Java和Python在企业中的应用情况
Java和Python在企业中的应用情况
48 7
|
22天前
|
缓存 Java 开发者
Java多线程并发编程:同步机制与实践应用
本文深入探讨Java多线程中的同步机制,分析了多线程并发带来的数据不一致等问题,详细介绍了`synchronized`关键字、`ReentrantLock`显式锁及`ReentrantReadWriteLock`读写锁的应用,结合代码示例展示了如何有效解决竞态条件,提升程序性能与稳定性。
83 6
|
20天前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
28 2
|
28天前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
44 8
|
27天前
|
关系型数据库 MySQL Java
MySQL索引优化与Java应用实践
【11月更文挑战第25天】在大数据量和高并发的业务场景下,MySQL数据库的索引优化是提升查询性能的关键。本文将深入探讨MySQL索引的多种类型、优化策略及其在Java应用中的实践,通过历史背景、业务场景、底层原理的介绍,并结合Java示例代码,帮助Java架构师更好地理解并应用这些技术。
26 2
|
29天前
|
存储 安全 Java
Java多线程编程中的并发容器:深入解析与实战应用####
在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
|
28天前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
61 1
|
4月前
|
存储 安全 Java
【Java集合类面试二十五】、有哪些线程安全的List?
线程安全的List包括Vector、Collections.SynchronizedList和CopyOnWriteArrayList,其中CopyOnWriteArrayList通过复制底层数组实现写操作,提供了最优的线程安全性能。
|
4月前
|
Java
【Java集合类面试二十三】、List和Set有什么区别?
List和Set的主要区别在于List是一个有序且允许元素重复的集合,而Set是一个无序且元素不重复的集合。
|
2月前
|
安全 Java 程序员
深入Java集合框架:解密List的Fail-Fast与Fail-Safe机制
本文介绍了 Java 中 List 的遍历和删除操作,重点讨论了快速失败(fail-fast)和安全失败(fail-safe)机制。通过普通 for 循环、迭代器和 foreach 循环的对比,详细解释了各种方法的优缺点及适用场景,特别是在多线程环境下的表现。最后推荐了适合高并发场景的 fail-safe 容器,如 CopyOnWriteArrayList 和 ConcurrentHashMap。
65 5
下一篇
DataWorks