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);

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


目录
打赏
0
4
4
1
31
分享
相关文章
Java也能快速搭建AI应用?一文带你玩转Spring AI可落地性
Java语言凭借其成熟的生态与解决方案,特别是通过 Spring AI 框架,正迅速成为 AI 应用开发的新选择。本文将探讨如何利用 Spring AI Alibaba 构建在线聊天 AI 应用,并实现对其性能的全面可观测性。
622 17
Java 也能快速搭建 AI 应用?一文带你玩转 Spring AI 可观测性
Java 也能快速搭建 AI 应用?一文带你玩转 Spring AI 可观测性
103 4
CRaC技术助力ACS上的Java应用启动加速
容器计算服务借助ACS的柔性算力特性并搭配CRaC技术极致地提升Java类应用的启动速度。
Java 也能快速搭建 AI 应用?一文带你玩转 Spring AI 可观测性
Java 也能快速搭建 AI 应用?一文带你玩转 Spring AI 可观测性
Java中的this关键字详解:深入理解与应用
本文深入解析了Java中`this`关键字的多种用法
480 9
|
2月前
|
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
198 60
【Java并发】【线程池】带你从0-1入门线程池
|
13天前
|
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
本文涉及InheritableThreadLocal和TTL,从源码的角度,分别分析它们是怎么实现父子线程传递的。建议先了解ThreadLocal。
49 4
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
86 23
|
1月前
|
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
152 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等