[Leetcode刷题]——链表
一、找出两个链表的交点
160.相交链表(easy)2021-01-05
编写一个程序,找到两个单链表相交的起始节点
如下面的两个链表,在c1 处相交:
- public class Solution {
- public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
- ListNode l1 = headA;
- ListNode l2 = headB;
- while(l1 != l2){
- l1 = (l1 == null) ? headB : l1.next;
- l2 = (l2 == null) ? headA : l2.next;
- }
- return l1;
- }
- }
解法:如果两个链表有相交,设第一个链表长度为 a+c, 第二个链表长度为b+c。
让两个链表指针从头开始遍历,第一个链表遍历a+c后接着从头开始遍历第二个链表的b,第二个链表遍历b+c后从头开始遍历第二个链表的a。
最后两个指针会同时到达相交的地方。
注意,条件判断时判断的是(l1 == null)而不是 (l1.next == null) 。其目的是为了如果两个链表没有相交,那么在遍历两个链表后连个指针均指向了链表最后的null,而非陷入死循环。
此解法的复杂度为O(n)。
二、反转链表
206.反转链表(easy)2021-01-06
反转一个单链表。
示例:
输入: 1->2->3->4->5->NULL
输出: 5->4->3->2->1->NULL
进阶:
你可以迭代或递归地反转链表。你能否用两种方法解决这道题?
我的原始解法,错误示范:
- class Solution {
- public ListNode reverseList(ListNode head) {
- ListNode newHead = head;
- newHead.next = null;
- ListNode traverseNode = head;
- while(traverseNode.next != null){
- traverseNode = traverseNode.next;
- ListNode temp = traverseNode;
- temp.next = newHead;
- newHead = temp;
- }
- return newHead;
- }
- }
错误分析:上述代码理论推导似乎可行,但是编译器运行不通过。
debug后发现,第2-5行,将head结点赋值给newhead结点,然后将newhead结点的next指针改为null,发现原始的head结点的next指针自动也变成了null。
在idea中可以看到,虽然head的值赋值给newHead 和traverseNode, 但是并没有开辟新空间,所以这里不能当一般变量处理。
标准解法:
1.递归,yyds
- public static ListNode reverseList(ListNode head) {
- if(head == null || head.next == null){
- return head;
- }
- ListNode next = head.next;
- ListNode newHead = reverseList(next);
- next.next = head;
- head.next = null;
- return newHead;
- }
2.头插法
- public ListNode reverseList(ListNode head) {
- ListNode newHead = new ListNode(-1);
- while (head != null) {
- ListNode next = head.next;
- head.next = newHead.next;
- newHead.next = head;
- head = next;
- }
- return newHead.next;
- }
三、合并两个有序链表
21、合并两个有序链表(easy) 2021-01-07
将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
示例:
输入:1->2->4, 1->3->4
输出:1->1->2->3->4->4
我的代码:
常规思路,需要注意的是在进行循环后的第二个if 语句时,又重新判断了( l1 != null )。 如果不加判断条件有可能为null就直接写l1.val会产生NullPointerException异常。
- class Solution {
- public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
- ListNode newHead = new ListNode(-1);
- ListNode temp = newHead;
- while(l1 != null && l2 != null){
- if(l1.val < l2.val){
- temp.next = l1;
- temp = l1;
- l1 = l1.next;
- }
- if(l1 != null && l2.val <= l1.val){
- temp.next = l2;
- temp = l2;
- l2 = l2.next;
- }
- }
- if(l1 == null){
- temp.next = l2;
- }
- if(l2 == null){
- temp.next = l1;
- }
- return newHead.next;
- }
- }
递归的写法,真简洁:
- class Solution {
- public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
- if (l1 == null) return l2;
- if (l2 == null) return l1;
- if (l1.val < l2.val) {
- l1.next = mergeTwoLists(l1.next, l2);
- return l1;
- } else {
- l2.next = mergeTwoLists(l1, l2.next);
- return l2;
- }
- }
- }
四、从有序链表中删除重复结点
83、删除排序链表中的重复元素 (easy) 2021-01-08
给定一个排序链表,删除所有重复的元素,使得每个元素只出现一次。
示例 1:
输入: 1->1->2
输出: 1->2
示例 2:输入: 1->1->2->3->3
输出: 1->2->3
我的解法:需要注意的一点是在没有判断结点是否为空的情况下之直接就使用.val 调用结点的数值会报NullPointerException异常
- class Solution {
- public ListNode deleteDuplicates(ListNode head) {
- ListNode tempNode = head;
- while(tempNode != null){
- if(tempNode.next != null && tempNode.val == tempNode.next.val){
- tempNode.next = tempNode.next.next;
- }else{
- tempNode = tempNode.next;
- }
- }
- return head;
- }
- }
递归的解法,妙呀,三行就搞定!唯一不足的就是内存消耗较多。
- class Solution {
- public ListNode deleteDuplicates(ListNode head) {
- if (head == null || head.next == null) return head;
- head.next = deleteDuplicates(head.next);
- return head.val == head.next.val ? head.next : head;
- }
- }
五、删除链表的倒数第N个结点
19、给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点。(medium) 2021-01-09
示例:
给定一个链表: 1->2->3->4->5, 和 n = 2.
当删除了倒数第二个节点后,链表变为 1->2->3->5.
说明:给定的 n 保证是有效的。
进阶:
你能尝试使用一趟扫描实现吗?
我的解法,遍历两次,内存击败29.8%,还是太菜
- class Solution {
- public ListNode removeNthFromEnd(ListNode head, int n) {
- ListNode tempNode = head;
- int len = 1;
- while(tempNode.next != null){
- len++;
- tempNode = tempNode.next;
- }
- tempNode = head;
- if(n == len){
- return head.next;
- }else{
- while(tempNode != null && tempNode.next != null){
- if(len == n + 1){
- tempNode.next = tempNode.next.next;
- }
- len--;
- tempNode = tempNode.next;
- }
- return head;
- }
- }
- }
遍历一次的原理,设置两个指针,让一个指针先走n长度,然后两个指针一起走,当快的指针走到链表结尾时,慢的指针就在倒数第n个位置
- class Solution {
- public ListNode removeNthFromEnd(ListNode head, int n) {
- ListNode fast = head;
- while (n-- > 0) {
- fast = fast.next;
- }
- if (fast == null) return head.next;
- ListNode slow = head;
- while (fast.next != null) {
- fast = fast.next;
- slow = slow.next;
- }
- slow.next = slow.next.next;
- return head;
- }
- }
六、两两交换链表中的相邻结点
24、给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。(medium) 2021-01-10
你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。
示例 1:
输入:head = [1,2,3,4]
输出:[2,1,4,3]
解法:力扣里的链表的头节点都是第一个结点,在这个题中,设置第一个结点的前一个结点为头结点,在判断后两个结点是否为空,然后进行交换即可。
- class Solution {
- public ListNode swapPairs(ListNode head) {
- ListNode preHeadNode = new ListNode(-1);
- preHeadNode.next = head;
- ListNode traverseNode = preHeadNode;
- while(traverseNode.next != null && traverseNode.next.next != null){
- ListNode n1 = traverseNode.next;
- ListNode n2 = traverseNode.next.next;
- traverseNode.next = n2;
- ListNode temp = n2.next;
- n2.next = n1;
- n1.next = temp;
- traverseNode = n1;
- }
- return preHeadNode.next;
- }
- }
递归的写法,需要系统的学习递归,链表和树的大多数题目太适合递归了。
参考博客https://lyl0724.github.io/2020/01/25/1/
递归三部曲:1.找终止条件 2.找返回值 3.单次过程
- class Solution {
- public ListNode swapPairs(ListNode head) {
- if(head == null || head.next == null){
- return head;
- }
- ListNode next = head.next;
- head.next = swapPairs(next.next);
- next.next = head;
- return next;
- }
- }
[Leetcode刷题]——链表的更多相关文章
- C#LeetCode刷题-链表
链表篇 # 题名 刷题 通过率 难度 2 两数相加 29.0% 中等 19 删除链表的倒数第N个节点 29.4% 中等 21 合并两个有序链表 C#LeetCode刷题之#21-合并两个有序链 ...
- LeetCode刷题 链表专题
链表专题 链表题目的一般做法 单链表的结构类型 删除节点 方法一 方法二 增加节点 LeedCode实战 LC19.删除链表的倒数第N个结点 解法思路 LC24.两两交换链表中的节点 解法思路 LC6 ...
- LeetCode刷题总结-链表
LeetCode刷题总结-链表 一.链表 链表分为单向链表.单向循环链表和双向链表,一下以单向链表为例实现单向链表的节点实现和单链表的基本操作. 单向链表 单向链表也叫单链表,是链表中最简单的 ...
- LeetCode刷题专栏第一篇--思维导图&时间安排
昨天是元宵节,过完元宵节相当于这个年正式过完了.不知道大家有没有投入继续投入紧张的学习工作中.年前我想开一个Leetcode刷题专栏,于是发了一个投票想了解大家的需求征集意见.投票于2019年2月1日 ...
- LeetCode刷题总结之双指针法
Leetcode刷题总结 目前已经刷了50道题,从零开始刷题学到了很多精妙的解法和深刻的思想,因此想按方法对写过的题做一个总结 双指针法 双指针法有时也叫快慢指针,在数组里是用两个整型值代表下标,在链 ...
- LeetCode刷题总结-数组篇(上)
数组是算法中最常用的一种数据结构,也是面试中最常考的考点.在LeetCode题库中,标记为数组类型的习题到目前为止,已累计到了202题.然而,这202道习题并不是每道题只标记为数组一个考点,大部分习题 ...
- LeetCode刷题总结-树篇(下)
本文讲解有关树的习题中子树问题和新概念定义问题,也是有关树习题的最后一篇总结.前两篇请参考: LeetCode刷题总结-树篇(上) LeetCode刷题总结-树篇(中) 本文共收录9道题,7道中等题, ...
- LeetCode刷题笔记和想法(C++)
主要用于记录在LeetCode刷题的过程中学习到的一些思想和自己的想法,希望通过leetcode提升自己的编程素养 :p 高效leetcode刷题小诀窍(这只是目前对我自己而言的小方法,之后会根据自己 ...
- C#LeetCode刷题-设计
设计篇 # 题名 刷题 通过率 难度 146 LRU缓存机制 33.1% 困难 155 最小栈 C#LeetCode刷题之#155-最小栈(Min Stack) 44.9% 简单 173 二叉搜索 ...
随机推荐
- Nday漏洞组合拳修改全校师生密码
很久以前写的文章了,发一下:) 本文是我真实的挖洞经历.撰写本文时相关学校已修复漏洞,相关漏洞也提交给了教育漏洞平台.纯粹是挖洞经验的总结和技术分享,由于敏感信息比较多,所以文章里面很多图片已经面目全 ...
- pandas 删除列
ddf = pd.DataFrame({"id":[1,2,3], "name":[4,5,6],"age":[7,8,9]})ddf = ...
- 百度前端技术学院-基础-day17-18
JavaScript小练习 task 1 基于上一个任务中,关于加减乘除的任务,加上对于特殊情况的判断,比如判断两个输入框是否都是正常输入了数字类型的内容,比如除法的时候除数是否为0,当判断到输入有异 ...
- 转:minhash
Minhash算法及其应用 一.引言 MinHash算法属于Locality Sensitive Hashing,用于快速估计两个集合的相似度.最早由Broder Andrei Z. 在1997年提出 ...
- MySQL技术内幕InnoDB存储引擎(七)——事务
什么是数据库的事务? 事务是访问并更新数据库中各种数据的一个程序执行单元.事务也是数据库区别于文件系统的一个重要特性. 事务需要满足的特性 1.原子性 原子性就是指数据库中的一个完整的事务是不可分割的 ...
- [JDK8]Map接口与Dictionary抽象类
package java.util; 一.Map接口 接口定义 public interface Map<K,V> Map是存放键值对的数据结构.map中没有重复的key,每个key最多只 ...
- 【Django admin 中文配置】
打开settings.py文件,找到语言编码.时区的设置项,将内容改为如下: [其中 zh-Hans是简体中文 zh-Hant是繁体中文] LANGUAGE_CODE = 'zh-Hans' # LA ...
- 我是如何用go-zero 实现一个中台系统的
最近发现golang社区里出了一个新星的微服务框架,来自好未来,光看这个名字,就很有奔头,之前,也只是玩过go-micro,其实真正的还没有在项目中运用过,只是觉得 微服务,grpc 这些很高大尚,还 ...
- Flink读写Redis(三)-读取redis数据
自定义flink的RedisSource,实现从redis中读取数据,这里借鉴了flink-connector-redis_2.11的实现逻辑,实现对redis读取的逻辑封装,flink-connec ...
- DRF类视图让你的代码DRY起来
刚开始写views.py模块的代码,一般都是用def定义的函数视图,不过DRF更推荐使用class定义的类视图,这能让我们的代码更符合DRY(Don't Repeat Yourself)设计原则: 使 ...