洛谷 P1440 求m区间内的最小值

本文介绍洛谷P1440题目的解题思路,使用单调队列解决区间内求最小值的问题。适用于数据规模大的情况,通过维护单调队列实现高效求解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

洛谷P1440 求m区间内的最小值

题目描述
一个含有n项的数列(n<=2000000),求出每一项前的m个数到它这个区间内的最小值。若前面的数不足m项则从第1个数开始,若前面没有数则输出0。

输入格式
第一行两个数n,m。
第二行,n个正整数,为所给定的数列。

输出格式
n行,第i行的一个数ai,为所求序列中第i个数前m个数的最小值。

输入样例
6 2
7 8 1 4 3 2
输出样例
0
7
7
1
1
3
数据规模
m ≤ n ≤ 2000000
ai ≤ 3× 1 0 7 10^7 107

题解
这题第一遍看到就知道是RMQ的题。数据又不会更改,很自然的想到ST表。这样做只能拿80分。还有什么办法———线段树,二分+递归。我要讲一下单调队列的做法。
为了求最小值,我们维护一个单调队列a,如果加进来的数小与队列末的数小,就删去,直到单调队列成升序为止。每次操作也需要删去前m个数之前的数。这样,只要每次的结果,即为单调队列的第一个位置的数值。
代码如下。

/*	2019.2.11 
 *	linwenqi
 *	luogu1440
 *	Cpp
 */
#include <iostream>
#include <cstdio>
using namespace std;
int a[2000005][2],head=0,tail=0;
//a[n][0]表示第n个数第几个入队
//a[n][1]表示第n个数的数值 head为头指针 tail为尾指针(最后元素的后一个) 
int main(){
	int n,m,i,x; 
	scanf("%d%d",&n,&m);
	printf("0\n");//第一个数前没有数 
	scanf("%d",&a[tail][1]);
	tail++;
	for(i=1;i<n;i++){
		if(i-a[head][0]>m)head++;//单调队列中的元素超过m 
		printf("%d\n",a[head][1]);//单调队列中最小元素 
		scanf("%d",&x);
		while(a[tail-1][1]>x&&head<tail)
			tail--;//单调队列入队操作 
		a[tail][0]=i;
		a[tail][1]=x;
		tail++;//注意tail后移 
	}
	return 0;
} 
### C语言中查找数组区间最小值和次小值 在C语言中,为了快速找到给定数组区间最小值和次小值,可以采用线性扫描方法。这种方法遍历指定范围内的所有元素一次,并在此过程中维护两个变量来跟踪遇到的最小值和次小值。 #### 算法描述 对于任意给定的数组`nums[]`以及起始索引`start`和结束索引`end`: - 初始化两个变量 `min_val` 和 `second_min_val` 来保存当前发现的最小值及其对应的次小值。 - 如果初始状态下存在相同的情况,则设定合理的初值以确保比较逻辑正常工作。 - 进入循环迭代过程,依次访问[start, end]之间的每一个元素: - 更新这两个变量使得`min_val`总是保持为已知范围内更小的那个数值;而当某个新元素既不是新的全局最小也不是现有最小的时候才更新`second_min_val`. 下面给出具体的代码实现方式: ```c #include <stdio.h> #include <limits.h> void findMinAndSecondMin(int nums[], int start, int end, int *minVal, int *secondMinVal) { // 设置默认的最大整型数作为初始化条件 *minVal = INT_MAX; *secondMinVal = INT_MAX; for (int i = start; i <= end; ++i) { if (nums[i] < *minVal) { *secondMinVal = *minVal; *minVal = nums[i]; } else if ((nums[i] != *minVal) && (nums[i] < *secondMinVal)) { *secondMinVal = nums[i]; } } // 处理特殊情况:如果找不到有效的次小值 if (*secondMinVal == INT_MAX) { printf("No second minimum found.\n"); } } // 测试函数 int main() { int array[] = {4, 2, 9, 7, 5, 6}; int n = sizeof(array)/sizeof(array[0]); int min, secMin; findMinAndSecondMin(array, 1, 4, &min, &secMin); printf("Minimum value is %d\n", min); printf("Second minimum value is %d\n", secMin); return 0; } ``` 此段代码实现了在一个特定区间内寻找最小值和次小值的功能[^1]。通过这种方式可以在O(n)时间复杂度下完成任务,其中n表示查询区间的大小。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值