hdu3486Interviewe(二分是错的)(ST算法RMQ + 判定上下界枚举)
题目大意是找最小的m使得前m段中每一段的最大值相加严格大于k,每一段长度为[n/m](n/m向下取整,多余的后半部分部分n-m*[n/m]不要)
先给一段我一开始的思路,和网上许多题解思路一样,但其实是有错误的。二分答案+RMQ (RMQ部分也可以用线段树)
#include <bits/stdc++.h>
using namespace std; const int maxn = 2e5 + ;
const int inf = 0x3f3f3f3f;
int f[maxn][];
int n, k, m, v[maxn];
int mx, mi, tot;
inline int max( int a,int b ){
return a>b ? a:b;
} inline int min( int a, int b ){
return a<b ? a:b;
} inline void ST_make(){
for( int i=; i<=n; i++ )
f[i][] = v[i];
for( int j=; (<<j)<=n; j++ )
for( int i=; i<=n; i++ )
f[i][j] = max( f[i][j-], f[i+(<<(j-))][j-] );
} inline int ST_query( int l, int r ){
int k = log(r-l+)/log();
return max( f[l][k], f[r-(<<k)+][k]);
} int main(){
while( ~scanf("%d%d", &n, &k) ){
if( n< && k< ) break;
memset( f, , sizeof(f) );
mx = -inf; mi = inf;
tot = ;
for( int i=; i<=n; i++ ){
scanf("%d", &v[i]);
tot += v[i];
mx = max( v[i], mx );
mi = min( v[i], mi );
}
if( mx>k ) {puts("1\n"); continue;}
else if( tot<=k ) {puts("-1\n"); continue;}
ST_make();
mx = mx ? mx:;
mi = mi ? mi:;
int l = k/mx, r = min( n, k/mi+ ), sum; //判断上下界,节省时间
while( l<r ){
sum = ;
int mid = l+r>>; //这个地方不能直接用m = l+r>>1
int t = n/mid;
int tmp = ;
int d = t*mid;
while( tmp<d){
sum += ST_query( tmp, tmp+t- );
tmp += t;
}
if( sum>k ){ //符合条件再令m = mid;
r = mid;
m = mid;
}
else l = mid+;
}
printf("%d\n", m);
} return ;
}
思路都写好了,然鹅是错的。。。
给一份Discuss里dalao给的卡二分的数据
#include <bits/stdc++.h>
using namespace std; const int maxn = 2e5 + ;
const int inf = 0x3f3f3f3f;
int f[maxn][];
int n, k, m, v[maxn];
int mx, mi, tot;
inline int max( int a,int b ){
return a>b ? a:b;
} inline int min( int a, int b ){
return a<b ? a:b;
} inline void ST_make(){
for( int i=; i<=n; i++ )
f[i][] = v[i];
int t = log(n)/log()+;
for( int j=; j<t; j++ )
for( int i=; i+(<<j)-<=n; i++ )
f[i][j] = max( f[i][j-], f[i+(<<(j-))][j-] );
} inline int ST_query( int l, int r ){
int k = log(r-l+)/log();
return max( f[l][k], f[r-(<<k)+][k]);
} int main(){
while( ~scanf("%d%d", &n, &k) ){
if( n< && k< ) break;
mx = -inf; mi = inf;
tot = ;
for( int i=; i<=n; i++ ){
scanf("%d", &v[i]);
tot += v[i];
mx = max( v[i], mx );
mi = min( v[i], mi );
}
if( tot<=k ){
printf("-1\n");
continue ;
}
ST_make();
mx = mx ? mx:;
mi = mi ? mi:;
int l = k/mx, r = (k/mi+, n); //判定上下界,节省时间(好像判了上界r还慢了。。)
if(l==) l = ;
bool ok = ;
for( m=l; m<=r; m++ ){
int sum = ;
int t = n/m;
int d = t*m;
for( int i=; i<=d; i+=t ) //这个地方要等于号,可举例n = m时 若i<d为判定条件则不能进行,卡了一年
sum += ST_query( i, i+t- );
if( sum>k ){
ok = ;
break;
}
}
if(ok) printf("%d\n", m);
else printf("-1\n");
}
return ;
}
hdu3486Interviewe(二分是错的)(ST算法RMQ + 判定上下界枚举)的更多相关文章
- 【37.48%】【hdu 2587】How far away ?(3篇文章,3种做法,LCA之ST算法(RMQ))
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s) ...
- 【原创】RMQ - ST算法详解
ST算法: ID数组下标: 1 2 3 4 5 6 7 8 9 ID数组元素: 5 7 3 1 4 8 2 9 8 1.ST算法作 ...
- RMQ问题之ST算法
RMQ问题之ST算法 RMQ(Range Minimum/Maximum Query)问题,即区间最值问题.给你n个数,a1 , a2 , a3 , ... ,an,求出区间 [ l , r ]的最大 ...
- HDU 5289 Assignment (ST算法区间最值+二分)
题目链接:pid=5289">http://acm.hdu.edu.cn/showproblem.php?pid=5289 题面: Assignment Time Limit: 400 ...
- HDU 5443 The Water Problem (ST算法)
题目链接:HDU 5443 Problem Description In Land waterless, water is a very limited resource. People always ...
- CodeForces 359D (数论+二分+ST算法)
题目链接: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=47319 题目大意:给定一个序列,要求确定一个子序列,①使得该子序 ...
- 求解区间最值 - RMQ - ST 算法介绍
解析 ST 算法是 RMQ(Range Minimum/Maximum Query)中一个很经典的算法,它天生用来求得一个区间的最值,但却不能维护最值,也就是说,过程中不能改变区间中的某个元素的值.O ...
- RMQ之ST算法模板
#include<stdio.h> #include<string.h> #include<iostream> using namespace std; ; ],M ...
- RMQ问题(线段树+ST算法)
转载自:http://kmplayer.iteye.com/blog/575725 RMQ (Range Minimum/Maximum Query)问题是指:对于长度为n的数列A,回答若干询问RMQ ...
随机推荐
- IBM X3650 M4 M5 设置服务器用UEFI模式启动支持磁盘GPT分区
1 系统启动 2 按 F1 3 进入BIOS 4 进入 System Configuration 5 找到 Boot Manager 6 找到Boot Modes 7 进入Boot Modes, 找到 ...
- Blazor 机制初探以及什么是前后端分离,还不赶紧上车?
标签: Blazor .Net 上一篇文章发了一个 BlazAdmin 的尝鲜版,这一次主要聊聊 Blazor 是如何做到用 C# 来写前端的,传送门:https://www.cnblogs.com/ ...
- DOM事件: DOM事件级别、DOM事件流、DOM事件模型、DOM事件捕获过程、自定义事件
前端面试中只要问到事件,就肯定会有DOM事件:如果回答出来了,就会一直向下延申,其实这些东西都很简单,但我第一次被问到的时候,也是懵的: DOM事件级别: DOM0 element.onclick = ...
- netcore 步骤
1.创建工程目录 d:\project 2.进入目录,创建解决方案 dotnet new sln 3.确定开发版本 dotnet --list-sdks //列出sdk版本 dotnet new gl ...
- spring整合mybatis报.UnsatisfiedDependencyException错误
tomcat启动报org.springframework.beans.factory.UnsatisfiedDependencyException:错误 org.springframework.bea ...
- Python基础(七)——文件和异常
1.1 读取整个文件 我们可以创建一个 test.txt 并写入一些内容,使用 Python 读文件操作,读出文本内容. with open(r'E:\test.txt') as file_objec ...
- Django框架(十)--ORM多对多关联关系三种创建方式、form组件
多对多的三种创建方式 1.全自动(就是平常我们创建表多对多关系的方式) class Book(models.Model): title = models.CharField(max_length=32 ...
- 配置安全web服务
为站点 http://system1.group8.example.com 配置TLS加密: 1.一个已签名证书从 http://server.group8.example.com/pub/tls/c ...
- 引用和自包含令牌(Reference Tokens and Introspection)
访问令牌可以有两种形式:自包含的和引用的. 自包含令牌(Self-contained tokens): 使用受保护的.有时间限制的数据结构,该结构包含元数据,并声明通过网络传递用户或客户机的身份.一种 ...
- C# 练习题 将一个正整数分解质因数
题目:将一个正整数分解质因数.例如:输入90,打印出90=2*3*3*5.程序分析:对n进行分解质因数,应先找到一个最小的质数k,然后按下述步骤完成:(1)如果这个质数恰等于n,则说明分解质因数的过程 ...