739. 每日温度
给定一个整数数组 temperatures ,表示每天的温度,返回一个数组 answer ,其中 answer[i] 是指对于第 i 天,下一个更高温度出现在几天后。如果气温在这之后都不会升高,请在该位置用 0 来代替。
示例 1:
输入: temperatures = [73,74,75,71,69,72,76,73]
输出: [1,1,4,2,1,1,0,0]
示例 2:
输入: temperatures = [30,40,50,60]
输出: [1,1,1,0]
示例 3:
输入: temperatures = [30,60,90]
输出: [1,1,0]
方法一:
class Solution {
// 版本 1
public int[] dailyTemperatures(int[] temperatures) {
int lens=temperatures.length;
int []res=new int[lens];
/**
如果当前遍历的元素 大于栈顶元素,表示 栈顶元素的 右边的最大的元素就是 当前遍历的元素,
所以弹出 栈顶元素,并记录
如果栈不空的话,还要考虑新的栈顶与当前元素的大小关系
否则的话,可以直接入栈。
注意,单调栈里 加入的元素是 下标。
*/
Deque<Integer> stack=new LinkedList<>();
stack.push(0);
for(int i=1;i<lens;i++){
if(temperatures[i]<=temperatures[stack.peek()]){
stack.push(i);
}else{
while(!stack.isEmpty()&&temperatures[i]>temperatures[stack.peek()]){
res[stack.peek()]=i-stack.peek();
stack.pop();
}
stack.push(i);
}
}
return res;
}
这段代码是用于解决「每日温度」问题的Java实现,这个问题的目标是给定一个整数数组 temperatures
,其中每个元素表示每天的温度,返回一个新的数组,其中每个元素表示直到未来几天才会出现更高的温度的天数。如果不存在未来几天会出现更高的温度,那么该位置的值为0。
代码解析
-
初始化:
- 创建一个与
temperatures
长度相同的数组res
,用于存放结果。 - 使用一个单调栈
stack
,数据结构为Deque
,在Java中通常使用LinkedList
实现,用于存储下标。
- 创建一个与
-
遍历并维护单调栈:
- 遍历
temperatures
数组中的每个元素。 - 对于当前遍历到的元素
temperatures[i]
:- 如果该元素小于等于栈顶元素对应的温度,即
temperatures[i] <= temperatures[stack.peek()]
,则将当前下标i
入栈。 - 否则,进入一个while循环:
- 当栈不为空且当前元素大于栈顶元素对应的温度时,弹出栈顶元素,并计算栈顶元素下标到当前下标
i
的距离,即i - stack.peek()
,并将这个距离存入结果数组res
的对应位置。 - 继续循环,直到栈为空或当前元素不大于栈顶元素对应的温度。
- 当栈不为空且当前元素大于栈顶元素对应的温度时,弹出栈顶元素,并计算栈顶元素下标到当前下标
- 最后,将当前下标
i
入栈。
- 如果该元素小于等于栈顶元素对应的温度,即
- 遍历
-
返回结果:
- 返回结果数组
res
,其中每个元素表示直到未来几天才会出现更高的温度的天数。
- 返回结果数组
时间复杂度和空间复杂度
- 时间复杂度: O(n),其中 n 是数组
temperatures
的长度。每个元素至多被放入和弹出栈一次。 - 空间复杂度: O(n),需要一个大小为
n
的结果数组res
和一个单调栈stack
。
总结
这段代码通过使用单调栈的策略,有效地解决了每日温度问题,能够快速找到每个温度元素右边第一个比它大的温度元素的位置。单调栈是一种常用的数据结构,在处理与单调性相关的数组或序列问题时非常有用,例如寻找最近的更大或更小的元素、股票价格波动分析等。在实际应用中,掌握单调栈的原理和使用方法能够帮助解决一系列经典问题,提高代码效率。
方法二:
//--------这 是一条分界线
// 版本 2
class Solution {
public int[] dailyTemperatures(int[] temperatures) {
int lens=temperatures.length;
int []res=new int[lens];
Deque<Integer> stack=new LinkedList<>();
for(int i=0;i<lens;i++){
while(!stack.isEmpty()&&temperatures[i]>temperatures[stack.peek()]){
res[stack.peek()]=i-stack.peek();
stack.pop();
}
stack.push(i);
}
return res;
}
这段代码同样是用于解决「每日温度」问题的Java实现,其目标与前一版本相同,即给定一个整数数组 temperatures
,返回一个新的数组,其中每个元素表示直到未来几天才会出现更高的温度的天数。如果不存在未来几天会出现更高的温度,那么该位置的值为0。
代码解析
-
初始化:
- 创建一个与
temperatures
长度相同的数组res
,用于存放结果。 - 使用一个单调栈
stack
,数据结构为Deque
,在Java中通常使用LinkedList
实现,用于存储下标。
- 创建一个与
-
遍历并维护单调栈:
- 遍历
temperatures
数组中的每个元素。 - 对于当前遍历到的元素
temperatures[i]
:- 进入一个while循环,只要栈不为空且当前元素大于栈顶元素对应的温度,执行以下操作:
- 计算栈顶元素下标到当前下标
i
的距离,即i - stack.peek()
,并将这个距离存入结果数组res
的对应位置。 - 弹出栈顶元素。
- 计算栈顶元素下标到当前下标
- 将当前下标
i
入栈。
- 进入一个while循环,只要栈不为空且当前元素大于栈顶元素对应的温度,执行以下操作:
- 遍历
-
返回结果:
- 返回结果数组
res
,其中每个元素表示直到未来几天才会出现更高的温度的天数。
- 返回结果数组
时间复杂度和空间复杂度
- 时间复杂度: O(n),其中 n 是数组
temperatures
的长度。每个元素至多被放入和弹出栈一次。 - 空间复杂度: O(n),需要一个大小为
n
的结果数组res
和一个单调栈stack
。
版本对比
相比于版本1,版本2的代码在遍历过程中直接从头开始,简化了代码逻辑,将入栈操作放在了while循环的外部。这种写法同样有效,且逻辑上更清晰,因为它避免了不必要的条件判断(如版本1中的 temperatures[i] <= temperatures[stack.peek()]
)。每次遍历到新元素时,代码首先处理栈中所有比当前元素小的元素,然后将当前元素入栈,这一过程确保了栈内的元素始终按温度递减的顺序排列,即形成了一个单调递减栈。
总结
通过使用单调栈的策略,版本2的代码同样有效地解决了每日温度问题。这种实现方式不仅保持了良好的时间复杂度和空间复杂度,而且在代码结构上更为精简和直观,有助于提高代码的可读性和维护性。单调栈在处理与单调性相关的数组或序列问题时是一种非常有用的工具,熟练掌握其使用方法对于解决一系列经典问题具有重要意义。
496. 下一个更大元素 I
nums1 中数字 x 的 下一个更大元素 是指 x 在 nums2 中对应位置 右侧 的 第一个 比 x 大的元素。
给你两个 没有重复元素 的数组 nums1 和 nums2 ,下标从 0 开始计数,其中nums1 是 nums2 的子集。
对于每个 0 <= i < nums1.length ,找出满足 nums1[i] == nums2[j] 的下标 j ,并且在 nums2 确定 nums2[j] 的 下一个更大元素 。如果不存在下一个更大元素,那么本次查询的答案是 -1 。
返回一个长度为 nums1.length 的数组 ans 作为答案,满足 ans[i] 是如上所述的 下一个更大元素 。
示例 1:
输入:nums1 = [4,1,2], nums2 = [1,3,4,2].
输出:[-1,3,-1]
解释:nums1 中每个值的下一个更大元素如下所述:
- 4 ,用加粗斜体标识,nums2 = [1,3,4,2]。不存在下一个更大元素,所以答案是 -1 。
- 1 ,用加粗斜体标识,nums2 = [1,3,4,2]。下一个更大元素是 3 。
- 2 ,用加粗斜体标识,nums2 = [1,3,4,2]。不存在下一个更大元素,所以答案是 -1 。
示例 2:
输入:nums1 = [2,4], nums2 = [1,2,3,4].
输出:[3,-1]
解释:nums1 中每个值的下一个更大元素如下所述:
- 2 ,用加粗斜体标识,nums2 = [1,2,3,4]。下一个更大元素是 3 。
- 4 ,用加粗斜体标识,nums2 = [1,2,3,4]。不存在下一个更大元素,所以答案是 -1 。
方法一:
class Solution {
public int[] nextGreaterElement(int[] nums1, int[] nums2) {
Stack<Integer> temp = new Stack<>();
int[] res = new int[nums1.length];
Arrays.fill(res,-1);
HashMap<Integer, Integer> hashMap = new HashMap<>();
for (int i = 0 ; i< nums1.length ; i++){
hashMap.put(nums1[i],i);
}
temp.add(0);
for (int i = 1; i < nums2.length; i++) {
if (nums2[i] <= nums2[temp.peek()]) {
temp.add(i);
} else {
while (!temp.isEmpty() && nums2[temp.peek()] < nums2[i]) {
if (hashMap.containsKey(nums2[temp.peek()])){
Integer index = hashMap.get(nums2[temp.peek()]);
res[index] = nums2[i];
}
temp.pop();
}
temp.add(i);
}
}
return res;
}
}
这段代码是用于解决「下一个更大元素 I」问题的Java实现。给定两个没有重复元素的数组 nums1
和 nums2
,其中 nums1
是 nums2
的一个子集,目标是对于 nums1
中的每一个元素,找到在 nums2
中下一个更大的元素的值。如果没有这样的元素,那么输出 -1。
代码解析
-
初始化:
- 创建一个单调栈
temp
,数据结构为Stack
,用于存储nums2
中元素的下标。 - 创建一个结果数组
res
,初始化所有元素为-1
,长度与nums1
相同。 - 创建一个哈希映射
hashMap
,用于存储nums1
中元素及其在res
数组中的下标,以便快速查找和更新结果。
- 创建一个单调栈
-
遍历并维护单调栈:
- 遍历
nums2
数组中的每个元素。 - 对于当前遍历到的元素
nums2[i]
:- 如果该元素小于等于栈顶元素对应的值,即
nums2[i] <= nums2[temp.peek()]
,则将当前下标i
入栈。 - 否则,进入一个while循环:
- 当栈不为空且当前元素大于栈顶元素对应的值时,检查栈顶元素是否在
nums1
中,如果是,则获取该元素在res
中的下标,并将当前元素的值赋给res
的对应位置。 - 继续循环,直到栈为空或当前元素不大于栈顶元素对应的值。
- 当栈不为空且当前元素大于栈顶元素对应的值时,检查栈顶元素是否在
- 最后,将当前下标
i
入栈。
- 如果该元素小于等于栈顶元素对应的值,即
- 遍历
-
返回结果:
- 返回结果数组
res
,其中每个元素表示nums1
中对应元素在nums2
中的下一个更大元素的值,若不存在则为-1
。
- 返回结果数组
时间复杂度和空间复杂度
- 时间复杂度: O(m + n),其中 m 和 n 分别是数组
nums1
和nums2
的长度。每个元素至多被放入和弹出栈一次,哈希表的查找操作平均时间复杂度为 O(1)。 - 空间复杂度: O(n),需要一个大小为
n
的单调栈temp
和一个哈希映射hashMap
,以及一个结果数组res
。
总结
这段代码通过使用单调栈和哈希映射的策略,有效地解决了下一个更大元素问题,能够快速找到 nums1
中每个元素在 nums2
中下一个更大元素的值。单调栈是一种常用的数据结构,在处理与单调性相关的数组或序列问题时非常有用,结合哈希映射能够进一步提高查找效率。在实际应用中,掌握这些数据结构和算法原理能够帮助解决一系列经典问题,提高代码效率和性能。
方法二:
// 版本2
class Solution {
public int[] nextGreaterElement(int[] nums1, int[] nums2) {
HashMap<Integer, Integer> map = new HashMap<>();
for (int i = 0; i < nums1.length; i++) {
map.put(nums1[i], i);
}
int[] res = new int[nums1.length];
Stack<Integer> stack = new Stack<>();
Arrays.fill(res, -1);
for (int i = 0; i < nums2.length; i++) {
while (!stack.isEmpty() && nums2[stack.peek()] < nums2[i]) {
int pre = nums2[stack.pop()];
if (map.containsKey(pre)) {
res[map.get(pre)] = nums2[i];
}
}
stack.push(i);
}
return res;
}
}
这段代码是用于解决「下一个更大元素 I」问题的另一种Java实现,其实现思路与之前解析的版本相似,但代码结构和变量命名有所不同,旨在解决给定两个没有重复元素的数组 nums1
和 nums2
,其中 nums1
是 nums2
的一个子集,对于 nums1
中的每一个元素,找到在 nums2
中下一个更大的元素的值。如果没有这样的元素,那么输出 -1。
代码解析
-
初始化:
- 创建一个哈希映射
map
,用于存储nums1
中元素及其在nums1
中的下标,以便快速查找和更新结果。 - 创建一个结果数组
res
,初始化所有元素为-1
,长度与nums1
相同。 - 创建一个单调栈
stack
,数据结构为Stack
,用于存储nums2
中元素的下标。
- 创建一个哈希映射
-
构建哈希映射:
- 遍历
nums1
数组,将每个元素及其在nums1
中的下标存储在map
中。
- 遍历
-
遍历并维护单调栈:
- 遍历
nums2
数组中的每个元素。 - 对于当前遍历到的元素
nums2[i]
:- 如果栈不为空并且栈顶元素对应的值小于当前元素,即
nums2[stack.peek()] < nums2[i]
,则:- 弹出栈顶元素,并获取其值。
- 如果弹出的元素在
nums1
中,即map
中存在该元素,那么在res
数组的对应位置填入当前元素的值。
- 继续这个过程,直到栈为空或者栈顶元素对应的值不再小于当前元素。
- 最后,将当前下标
i
入栈。
- 如果栈不为空并且栈顶元素对应的值小于当前元素,即
- 遍历
-
返回结果:
- 返回结果数组
res
,其中每个元素表示nums1
中对应元素在nums2
中的下一个更大元素的值,若不存在则为-1
。
- 返回结果数组
时间复杂度和空间复杂度
- 时间复杂度: O(m + n),其中 m 和 n 分别是数组
nums1
和nums2
的长度。每个元素至多被放入和弹出栈一次,哈希表的查找操作平均时间复杂度为 O(1)。 - 空间复杂度: O(n),需要一个大小为
n
的单调栈stack
和一个哈希映射map
,以及一个结果数组res
。
总结
这段代码同样通过使用单调栈和哈希映射的策略,有效地解决了下一个更大元素问题,能够快速找到 nums1
中每个元素在 nums2
中下一个更大元素的值。代码结构清晰,逻辑直观,是解决这类问题的一种典型且高效的方法。掌握这些数据结构和算法原理能够帮助解决一系列经典问题,提高代码效率和性能。在实际应用中,根据具体需求和场景选择合适的数据结构和算法是非常重要的。
503. 下一个更大元素 II
给定一个循环数组 nums ( nums[nums.length - 1] 的下一个元素是 nums[0] ),返回 nums 中每个元素的 下一个更大元素 。
数字 x 的 下一个更大的元素 是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该循环地搜索它的下一个更大的数。如果不存在,则输出 -1 。
示例 1:
输入: nums = [1,2,1]
输出: [2,-1,2]
解释: 第一个 1 的下一个更大的数是 2;
数字 2 找不到下一个更大的数;
第二个 1 的下一个最大的数需要循环搜索,结果也是 2。
示例 2:
输入: nums = [1,2,3,4,3]
输出: [2,3,4,-1,4]
class Solution {
public int[] nextGreaterElements(int[] nums) {
//边界判断
if(nums == null || nums.length <= 1) {
return new int[]{-1};
}
int size = nums.length;
int[] result = new int[size];//存放结果
Arrays.fill(result,-1);//默认全部初始化为-1
Stack<Integer> st= new Stack<>();//栈中存放的是nums中的元素下标
for(int i = 0; i < 2*size; i++) {
while(!st.empty() && nums[i % size] > nums[st.peek()]) {
result[st.peek()] = nums[i % size];//更新result
st.pop();//弹出栈顶
}
st.push(i % size);
}
return result;
}
}
这段代码是用于解决「下一个更大元素 II」问题的Java实现。给定一个循环数组 nums
(数组中元素的下一个元素是数组的第一个元素),目标是返回一个数组,其中每个元素是原数组中下一个更大元素的值,如果没有更大的元素,则对应位置的值为 -1。
代码解析
-
初始化:
- 创建一个单调栈
st
,数据结构为Stack
,用于存储nums
中元素的下标。 - 创建一个结果数组
result
,初始化所有元素为-1
,长度与nums
相同。
- 创建一个单调栈
-
遍历并维护单调栈:
- 遍历
nums
数组两次,即遍历2 * nums
的长度(考虑到循环数组的性质),使用模运算i % size
来获取实际数组下标。 - 对于当前遍历到的元素
nums[i % size]
:- 如果栈不为空并且栈顶元素对应的值小于当前元素,即
nums[st.peek()] < nums[i % size]
,则:- 更新
result
数组中对应位置的值为当前元素的值。 - 弹出栈顶元素。
- 更新
- 继续这个过程,直到栈为空或者栈顶元素对应的值不再小于当前元素。
- 最后,将当前下标
i % size
入栈。
- 如果栈不为空并且栈顶元素对应的值小于当前元素,即
- 遍历
-
返回结果:
- 返回结果数组
result
,其中每个元素表示原数组nums
中对应元素的下一个更大元素的值,若不存在则为-1
。
- 返回结果数组
时间复杂度和空间复杂度
- 时间复杂度: O(n),其中 n 是数组
nums
的长度。虽然代码中遍历了2 * nums
的长度,但是每个元素至多被放入和弹出栈一次,因此总的时间复杂度为 O(n)。 - 空间复杂度: O(n),需要一个大小为
n
的单调栈st
和一个结果数组result
。
总结
这段代码通过使用单调栈的策略,有效地解决了下一个更大元素 II 问题,能够处理循环数组并找到每个元素的下一个更大元素的值。代码逻辑清晰,通过两次遍历数组并使用模运算处理循环数组的特点,实现了高效的解题策略。单调栈在处理这类问题时表现出了强大的能力,能够快速找到特定条件下下一个更大的元素,是解决此类问题的经典方法。在实际应用中,了解和掌握单调栈的使用方法对于解决与单调性相关的数组或序列问题具有重要意义。