豆知识
伪头!!
1 ListNode phead = new ListNode(-1 , head);
java链表基本结构
1 2 3 4 5 6 7 8 class ListNode { int val; ListNode next; ListNode(int val){ this .val=val; } }
范型写法:使用范型可以兼容不同的数据类型
1 2 3 4 5 6 7 8 class ListNode <E > { E val; ListNode<E> next; ListNode(E val){ this .val=val; } }
leetcode203中的定义
1 2 3 4 5 6 7 8 9 10 11 12 13 public class ListNode { int val; ListNode next; ListNode() {} ListNode(int val) { this .val = val; } ListNode(int val, ListNode next) { this .val = val; this .next = next; } }
双链表
1 2 3 4 5 6 7 public class ListNode { int val; ListNode next; ListNode prev; ListNode(int x) { val = x; } }
回收删除
在对节点进行替换或删除的时候,被替换或被删节点的next引用需不需要设置为null?
答案是: 不需要 ,因为一个对象被回收的前提是因为没有任何地方持有这个对象的引用(引用计数器为0)也就是说它不在被引用,那么那么它将被回收,至于它引用什么对象无关紧要,因为对于它所引用的对象来说依然是看引用计数器是否为0;
链表的递归遍历
剑指 offer 06. 从尾到头打印链表
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 class Solution { ArrayList<Integer> tmp = new ArrayList<Integer>(); public int [] reversePrint(ListNode head) { recur(head); int [] res = new int [tmp.size()]; for (int i = 0 ; i < res.length; i++) res[i] = tmp.get(i); return res; } void recur (ListNode head) { if (head == null ) return ; recur(head.next); tmp.add(head.val); } }
这题可以直接用循环+栈做,本质一样
给你一个链表的头节点 head
和一个整数 val
,请你删除链表中所有满足 Node.val == val
的节点,并返回 新的头节点 。
思路:设置虚拟节点dummyNode
由于链表的头节点head
有可能需要被删除,因此创建哑节点dummyNode
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 class Solution { public ListNode removeElements (ListNode head, int val) { if (head == null ) { return head; } ListNode dummyNode = new ListNode(-1 , head); ListNode pre = dummyNode; ListNode cur = head; while (cur != null ) { if (cur.val == val) { pre.next = cur.next; } else { pre = cur; } cur = cur.next; } return dummyNode.next; } }
基础操作:707.设计链表
设计链表的实现。您可以选择使用单链表或双链表。单链表中的节点应该具有两个属性:val
和 next
。val
是当前节点的值,next
是指向下一个节点的指针/引用。如果要使用双向链表,则还需要一个属性 prev
以指示链表中的上一个节点。假设链表中的所有节点都是 0-index 的。
在链表类中实现这些功能:
get(index):获取链表中第 index
个节点的值。如果索引无效,则返回-1
。
addAtHead(val):在链表的第一个元素之前添加一个值为 val
的节点。插入后,新节点将成为链表的第一个节点。
addAtTail(val):将值为 val
的节点追加到链表的最后一个元素。
addAtIndex(index,val):在链表中的第 index
个节点之前添加值为 val
的节点。如果 index
等于链表的长度,则该节点将附加到链表的末尾。如果 index
大于链表长度,则不会插入节点。如果index
小于0,则在头部插入节点。
deleteAtIndex(index):如果索引 index
有效,则删除链表中的第 index
个节点。
示例:
1 2 3 4 5 6 7 MyLinkedList linkedList = new MyLinkedList(); linkedList.addAtHead(1 ); linkedList.addAtTail(3 ); linkedList.addAtIndex(1 ,2 ); linkedList.get(1 ); linkedList.deleteAtIndex(1 ); linkedList.get(1 );
思路
单链表
有关ListNode的定义见leetcode203中的定义
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 class MyLinkedList { public ListNode phead; public int size; public MyLinkedList () { this .size = 0 ; this .phead = new ListNode(-1 ); } public int get (int index) { if (index < 0 || index >= size) return -1 ; ListNode cur = this .phead; for (int i = 0 ; i <= index; i++) { cur = cur.next; } return cur.val; } public void addAtHead (int val) { addAtIndex(0 , val); } public void addAtTail (int val) { ListNode tmp = this .phead; while (tmp.next != null ) { tmp = tmp.next; } tmp.next = new ListNode(val); this .size++; } public void addAtIndex (int index, int val) { if (index > this .size) return ; if (index < 0 ) index = 0 ; this .size++; ListNode pre = phead; for (int i = 0 ; i < index; i++) { pre = pre.next; } ListNode newNode = new ListNode(val, pre.next); pre.next = newNode; } public void deleteAtIndex (int index) { if (index < 0 || index >= this .size) return ; this .size--; ListNode pre = phead; for (int i = 0 ; i < index; i++) { pre = pre.next; } pre.next = pre.next.next; } }
双链表
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 public class ListNode { int val; ListNode next; ListNode prev; ListNode(int x) { val = x; } }class MyLinkedList { int size; ListNode head, tail; public MyLinkedList () { size = 0 ; head = new ListNode(0 ); tail = new ListNode(0 ); head.next = tail; tail.prev = head; } public int get (int index) { if (index < 0 || index >= size) return -1 ; ListNode curr = head; if (index + 1 < size - index) for (int i = 0 ; i < index + 1 ; ++i) curr = curr.next; else { curr = tail; for (int i = 0 ; i < size - index; ++i) curr = curr.prev; } return curr.val; } public void addAtHead (int val) { ListNode pred = head, succ = head.next; ++size; ListNode toAdd = new ListNode(val); toAdd.prev = pred; toAdd.next = succ; pred.next = toAdd; succ.prev = toAdd; } public void addAtTail (int val) { ListNode succ = tail, pred = tail.prev; ++size; ListNode toAdd = new ListNode(val); toAdd.prev = pred; toAdd.next = succ; pred.next = toAdd; succ.prev = toAdd; } public void addAtIndex (int index, int val) { if (index > size) return ; if (index < 0 ) index = 0 ; ListNode pred, succ; if (index < size - index) { pred = head; for (int i = 0 ; i < index; ++i) pred = pred.next; succ = pred.next; } else { succ = tail; for (int i = 0 ; i < size - index; ++i) succ = succ.prev; pred = succ.prev; } ++size; ListNode toAdd = new ListNode(val); toAdd.prev = pred; toAdd.next = succ; pred.next = toAdd; succ.prev = toAdd; } public void deleteAtIndex (int index) { if (index < 0 || index >= size) return ; ListNode pred, succ; if (index < size - index) { pred = head; for (int i = 0 ; i < index; ++i) pred = pred.next; succ = pred.next.next; } else { succ = tail; for (int i = 0 ; i < size - index - 1 ; ++i) succ = succ.prev; pred = succ.prev.prev; } --size; pred.next = succ; succ.prev = pred; } }
给你两个 非空 的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。
请你将两个数相加,并以相同形式返回一个表示和的链表。
你可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例 1:
1 2 3 输入:l1 = [2,4,3], l2 = [5,6,4] 输出:[7,0,8] 解释:342 + 465 = 807.
思路/试错
一开始用int sum来存储数据,通过sum+=val*i;i*=10
类似的方法来更新sum;然后再通过不断%10
来把每一位加入链表
这样会有问题,当数据太大的时候,因为int类型长度有限,会发生溢出
Testcase
1 2 [9] [1,9,9,9,9,9,9,9,9,9]
Answer
Expected Answer
1 [0,0,0,0,0,0,0,0,0,0,1]
而且要多写一次循环=。=
在加的过程中就建链表
注意最后的进位要处理
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 class Solution { public ListNode addTwoNumbers (ListNode l1, ListNode l2) { ListNode newPHead = new ListNode(); ListNode pre = newPHead; int c = 0 ; int sum = 0 ; while (l1 != null || l2 != null ) { int n1 = l1 != null ? l1.val : 0 ; int n2 = l2 != null ? l2.val : 0 ; sum = n1 + n2 + c; if (sum >= 10 ) { c = 1 ; sum %= 10 ; } else { c = 0 ; } ListNode newNode = new ListNode(sum); pre.next = newNode; pre = newNode; if (l1 != null ) { l1 = l1.next; } if (l2 != null ) { l2 = l2.next; } } if (c == 1 ) { ListNode newNode = new ListNode(1 ); pre.next = newNode; pre = newNode; } return newPHead.next; } }
给你一个链表,删除链表的倒数第 n
个结点,并且返回链表的头结点。
思路: 快慢指针
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 class Solution { public ListNode removeNthFromEnd (ListNode head, int n) { ListNode phead = new ListNode(-1 , head); ListNode slow = phead; ListNode fast = phead; for (int i = 0 ; i < n; i++) { fast = fast.next; } while (fast.next != null ) { fast = fast.next; slow = slow.next; } slow.next = slow.next.next; return phead.next; } }
给你一个链表数组,每个链表都已经按升序排列。
请你将所有链表合并到一个升序链表中,返回合并后的链表。
思路一: 在21.合并两个有序链表 的基础上改进,两两合并
可以依次遍历,但更好的方法是 分治(类二分法)
按[left,mid],[mid+1,right]进行划分
left==right时,返回list[left]
left>right时,返回null,递归终止
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 class Solution { public ListNode mergeKLists (ListNode[] lists) { if (lists.length == 0 ) return null ; return merge(lists, 0 , lists.length - 1 ); } public ListNode merge (ListNode[] lists, int l, int r) { if (l < r) { int mid = (r + l) >> 1 ; return mergeTwoLists(merge(lists, l, mid), merge(lists, mid + 1 , r)); } else if (l == r) { return lists[l]; } return null ; } public ListNode mergeTwoLists (ListNode list1, ListNode list2) { ListNode pnewList = new ListNode(); ListNode cur = pnewList; while (list1 != null && list2 != null ) { int n1 = list1.val; int n2 = list2.val; if (n1 < n2) { cur.next = list1; list1 = list1.next; } else { cur.next = list2; list2 = list2.next; } cur = cur.next; } cur.next = list1 == null ? list2 : list1; return pnewList.next; } }
即每个链表没有被合并的元素的最前面一个,每次在这些元素里面选取 val
属性最小的元素合并到答案中。
用优先队列优化 合并K个排序链表 - 合并K个升序链表 - 力扣(LeetCode) (leetcode-cn.com)
PriorityQueue
和Queue
的区别在于,它的出队顺序与元素的优先级有关,对PriorityQueue
调用remove()
或poll()
方法,返回的总是优先级最高的元素。
要使用PriorityQueue
,我们就必须给每个元素定义“优先级”。
使用PriorityQueue - 廖雪峰的官方网站 (liaoxuefeng.com)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 class Solution { class Status implements Comparable <Status > { int val; ListNode ptr; Status(int val, ListNode ptr) { this .val = val; this .ptr = ptr; } public int compareTo (Status status2) { return this .val - status2.val; } } PriorityQueue<Status> queue = new PriorityQueue<Status>(); public ListNode mergeKLists (ListNode[] lists) { for (ListNode node: lists) { if (node != null ) { queue.offer(new Status(node.val, node)); } } ListNode head = new ListNode(0 ); ListNode tail = head; while (!queue.isEmpty()) { Status f = queue.poll(); tail.next = f.ptr; tail = tail.next; if (f.ptr.next != null ) { queue.offer(new Status(f.ptr.next.val, f.ptr.next)); } } return head.next; } }
给你一个链表的头节点 head
,判断链表中是否有环。
如果链表中有某个节点,可以通过连续跟踪 next
指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos
来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos
是 -1
,则在该链表中没有环。注意:pos
不作为参数进行传递 ,仅仅是为了标识链表的实际情况。
如果链表中存在环,则返回 true
。 否则,返回 false
。
// pos 不作为参数进行传递,仅仅是为了标识链表的实际情况。
// pos 是给我们自定义测试案例的
思路: 快慢指针
慢指针步长1,快指针步长2,如果是环形链表,快指针一定能追上慢指针
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 public class Solution { public boolean hasCycle (ListNode head) { if (head == null || head.next == null ) return false ; ListNode slowp = head; ListNode fastp = head; while (fastp != null && fastp.next != null ) { slowp = slowp.next; fastp = fastp.next.next; if (slowp == fastp) { return true ; } } return false ; } }
给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null
。
如果链表中有某个节点,可以通过连续跟踪 next
指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos
来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos
是 -1
,则在该链表中没有环。注意:pos
不作为参数进行传递 ,仅仅是为了标识链表的实际情况。
不允许修改 链表。
思路一:用Set存节点
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 public class Solution { public ListNode detectCycle (ListNode head) { Set<ListNode> nodes = new HashSet<>(); while (head != null ) { if (!nodes.add(head)) { return head; } head = head.next; } return null ; } }
思路二:快慢指针
在141的基础上,通过计算来得到相应的节点
慢指针和快指针相遇时:慢指针走了$a+b$,快指针走了$a+n(b+c)+b$,则
$$2(a+b)=a+n(b+c)+b$$
即
$$a=(n-1)(b+c)+c$$
新设一个指针newp,从head开始和慢指针一起移动,则最后一定会在入口处相遇
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 public class Solution { public ListNode detectCycle (ListNode head) { if (head == null || head.next == null ) { return null ; } ListNode slowp = head; ListNode fastp = head; while (fastp != null && fastp.next != null ) { slowp = slowp.next; fastp = fastp.next.next; if (slowp == fastp) { ListNode newp = head; while (slowp != newp) { slowp = slowp.next; newp = newp.next; } return newp; } } return null ; } }