前言
###我做这类文章一个重要的目的还是给正在学习的大家提供方向(例如想要掌握基础用法,该刷哪些题?建议灵神的题单和代码随想录)和记录自己的学习过程,我的解析也不会做的非常详细,只会提供思路和一些关键点,力扣上的大佬们的题解质量是非常非常高滴!!!
这几天今天复习一下数据结构
习题
1.二进制链表转整数
题目链接:1290. 二进制链表转整数 - 力扣(LeetCode)
题面:
分析:遍历链表,然后开一个变量,每次左移一位再加上链表上的值即可
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public int getDecimalValue(ListNode head) {
int sum = 0;
for(ListNode i = head;i!=null;i=i.next){
sum<<=1;
sum+=i.val;
}
return sum;
}
}
2. 找出临界点之间的最小和最大距离
题目链接:2058. 找出临界点之间的最小和最大距离 - 力扣(LeetCode)
题面:
分析:我们可以定义三个变量分别记录最开始的节点索引,上一个的索引,和最后的索引,那么最小距离就是在找到一个索引后,取ans[0]和当前索引与记录的上一个节点的索引的最小值,最大值就是最后一个索引与第一个索引的超值
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
int[] ans = new int[]{Integer.MAX_VALUE,-1};
public int[] nodesBetweenCriticalPoints(ListNode head) {
int start = -1;
int end = -1;
int pre = -1;
int count = 0;
int index = 0;
for(ListNode i = head;i!=null&&i.next.next!=null;i=i.next){
index++;
int a = i.val;
int b = i.next.val;
int c = i.next.next.val;
if(b>a&&b>c){
count++;
if(count==1)start = index;
if(index>1)ans[0] = Math.min(ans[0],index - pre);
pre = index;
}
else if(b<a&&b<c){
count++;
if(count==1)start = index;
if(index>1)ans[0] = Math.min(ans[0],index - pre);
pre = index;
}
}
if(count<2)return new int[]{-1,-1};
ans[1] = pre - start;
return ans;
}
}
3.合并零之间的节点
题目链接:2181. 合并零之间的节点 - 力扣(LeetCode)
题面:
分析:这题遍历链表,定义一个变量sum来计算和值,如果下一个是0,就更新链表
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode mergeNodes(ListNode head) {
ListNode ans = new ListNode();
ListNode node = new ListNode();
ans.next = node;
int sum = 0;
int count= 0;
for(ListNode i = head;i!=null;i=i.next){
if(i.val==0)continue;
sum+=i.val;
if(i.next.val==0&&count==0){
node.val = sum;
sum = 0;
count++;
}else if(i.next.val==0){
ListNode node2 = new ListNode();
node2.val = sum;
sum = 0;
node.next = node2;
node = node2;
}
}
return ans.next;
}
}
4. 分割链表
题面:
分析:这题主要看链表中节点的个数与k之间的数学关系,例如如果count/k==0,说明肯定不够分,那么就是将链表全分为单节点,如果>0,记flag = count/k,如果count%k!=0,为了迎合题意:任意两部分的长度差距不能超过1,且前面的长度大于后面的长度,则让前count%k的长度为flag+1
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode[] splitListToParts(ListNode head, int k) {
ListNode[] ans = new ListNode[k];
int count = 0;
for(ListNode i = head;i!=null;i=i.next)count++;
// System.out.println(count);
int flag = count/k;
if(flag==0){
int index = 0;
for(ListNode i = head;i!=null;i=i.next){
ListNode node = new ListNode(i.val);
ans[index++] = node;
}
}else{
int fcount = count%k;
int index = 0;
int gcount = 0;
boolean su = false;
ListNode fhead = head;
for(ListNode i = head;i!=null||((i==null)&&su);i=i.next){
if(su){
i = fhead;
su = false;
}
gcount++;
if(fcount>0&&gcount%(flag+1)==0){
ListNode node = i;
i = i.next;
su = true;
node.next = null;
ans[index++] = fhead;
fhead = i;
gcount = 0;
fcount--;
}else if(fcount==0&&gcount%flag==0){
ListNode node = i;
i = i.next;
su = true;
node.next = null;
ans[index++] = fhead;
fhead = i;
gcount = 0;
}
if(i==null)break;
}
}
return ans;
}
}
5.链表组件
题面:
分析:哈希表
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public int numComponents(ListNode head, int[] nums) {
Map<Integer,Integer> map = new HashMap<>();
for(int a:nums){
map.put(a,1);
}
int ans = 0;
int count = 0;
for(ListNode i = head;i!=null;i = i.next){
if(map.getOrDefault(i.val,-1)==1){
count++;
continue;
}else{
if(count>0){
ans++;
count = 0;
}
}
}
if(count>0)ans++;
return ans;
}
}
6.移除链表元素
题目链接:203. 移除链表元素 - 力扣(LeetCode)
题面:
分析:定义一个虚拟头结点会简单些
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode removeElements(ListNode head, int val) {
ListNode fhead = new ListNode();
fhead.next = head;
ListNode pre = fhead;
for(ListNode i = head;i!=null;i = i.next){
if(i.val==val){
pre.next = i.next;
}else{
pre = i;
}
}
return fhead.next;
}
}
7.从链表中移除在数组中存在的节点
题目链接:3217. 从链表中移除在数组中存在的节点 - 力扣(LeetCode)
题面:
分析:在上一题的基础上加个哈希表
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode modifiedList(int[] nums, ListNode head) {
Map<Integer,Integer> map = new HashMap<>();
for(int a:nums){
map.put(a,1);
}
ListNode fhead = new ListNode();
fhead.next = head;
int count = 0;
ListNode pre = fhead;
for(ListNode i = head;i!=null;i = i.next){
if(map.getOrDefault(i.val,-1)==1){
pre.next = i.next;
}else{
pre = i;
}
}
return fhead.next;
}
}
8.删除排序链表中的重复元素
题目链接:83. 删除排序链表中的重复元素 - 力扣(LeetCode)
题面:
分析:和上面题类似
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode deleteDuplicates(ListNode head) {
ListNode fhead = new ListNode(Integer.MIN_VALUE);
fhead.next = head;
ListNode pre = fhead;
for(ListNode i = head;i!=null;i = i.next){
if(i.val == pre.val){
pre.next = i.next;
}else{
pre = i;
}
}
return fhead.next;
}
}
9.删除排序链表中的重复元素II
题目链接:82. 删除排序链表中的重复元素 II - 力扣(LeetCode)
题面:
分析:这题有许多小问题需要考虑,值得细做
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode deleteDuplicates(ListNode head) {
ListNode fhead = new ListNode(Integer.MAX_VALUE);
fhead.next = head;
ListNode pre = fhead;
ListNode flag = fhead;
int count = 1;
boolean su = false;
for(ListNode i = head;i!=null;i = i.next){
su = false;
if(i.val!=pre.val&&count>=2){
flag.next = i;
count = 1;
su = true;
}else if(i.val==pre.val){
count++;
}
if(i.val!=head.val&&i.val!=pre.val&&!su){
flag = pre;
}
pre = i;
}
if(count>=2)flag.next = null;
return fhead.next;
}
}
10.删除链表中的节点
题目链接:237. 删除链表中的节点 - 力扣(LeetCode)
题面:
分析:让node等于node的下一个节点的值,以此类推,但题目需要链表长度减一,那么倒数第二个节点应该指向null,如果i.next.next==null,说明此时遍历到的i就是倒数第二个节点
代码:
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode(int x) { val = x; }
* }
*/
class Solution {
public void deleteNode(ListNode node) {
for(ListNode i = node;i!=null;i=i.next){
i.val = i.next.val;
if(i.next.next==null){
i.next = null;
break;
}
}
}
}
11.合并两个链表
题目链接:1669. 合并两个链表 - 力扣(LeetCode)
题面:
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode mergeInBetween(ListNode list1, int a, int b, ListNode list2) {
ListNode fhead = new ListNode();
fhead.next = list1;
ListNode pre = fhead;
int index = 0;
boolean su = true;
for(ListNode i = list1;i!=null;i=i.next){
if(index==a){
su = false;
}
if(su)pre = i;
if(index==b){
pre.next = list2;
for(ListNode j = list2;j!=null;j = j.next){
if(j.next==null){
j.next = i.next;
break;
}
}
break;
}
index++;
}
return fhead.next;
}
}
12.从链表中移除节点
题目链接:2487. 从链表中移除节点 - 力扣(LeetCode)
题面:
分析:首先将链表中的值存到数组里去,然后从后往前遍历数组,并不断更新当前索引右边数字集合的最大值,如果当前索引所对应的值小于最大值,那么将当前位置标记成一个值(不在节点取值范围内),如果大于,就更新最大值,然后遍历链表,将标记的位置的节点删掉就行
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
int mu = 100005;
public ListNode removeNodes(ListNode head) {
int[] arr = new int[100005];
int count = 0;
for(ListNode i = head;i!=null;i=i.next){
arr[count++] = i.val;
}
int max = arr[count-1];
for(int i = count-2;i>=0;i--){
if(arr[i]<max){
arr[i] = mu;
}else{
max = arr[i];
arr[i] = 0;
}
}
ListNode fhead = new ListNode();
fhead.next = head;
ListNode pre = fhead;
int index = 0;
for(ListNode i = head;i!=null;i=i.next){
if(arr[index++]==mu){
pre.next = i.next;
}else{
pre = i;
}
}
return fhead.next;
}
}
13.在链表中插入最大公约数
题目链接:2807. 在链表中插入最大公约数 - 力扣(LeetCode)
题面:
分析:这里用到了辗转相除法快速求最大公约数
代码:
/**
* Definition for singly-linked list.
* 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; }
* }
*/
class Solution {
public ListNode insertGreatestCommonDivisors(ListNode head) {
if(head.next==null)return head;
for(ListNode i = head;i!=null;i=i.next.next){
int val = gcd(i.val,i.next.val);
ListNode node = new ListNode(val);
ListNode next = i.next;
i.next = node;
node.next = next;
if(i.next.next.next==null)break;
}
return head;
}
public int gcd(int a,int b){
return b==0?a:gcd(b,a%b);
}
}
14.循环有序列表的插入
题目链接: LCR 029. 循环有序列表的插入 - 力扣(LeetCode)
题面:
贴上大佬代码:
class Solution {
public Node insert(Node he, int x) {
Node t = new Node(x);
t.next = t;
if (he == null) return t;
Node ans = he;
int min = he.val, max = he.val;
while (he.next != ans) {
he = he.next;
min = Math.min(min, he.val);
max = Math.max(max, he.val);
}
if (min == max) {
t.next = ans.next;
ans.next = t;
} else {
while (!(he.val == max && he.next.val == min)) he = he.next;
while (!(x <= min || x >= max) && !(he.val <= x && x <= he.next.val)) he = he.next;
t.next = he.next;
he.next = t;
}
return ans;
}
}
后言
上面是力扣数据结构相关,下一篇是其他的习题,希望有所帮助,一同进步,共勉!