题面:给出长度为n的数列,然后算出其区间和乘区间最小数所能得到的最大值,并且输出区间

样例输入:

6

3 1 6 4 5 2

样例输出:

60

3 5

原题链接:https://vjudge.net/problem/UVA-1619


分析:

这里有两种算法,一种是O(nlogn)的,用st表+递归,另一种是O(n)的,用单调栈。

容易知道对于数列中的每一个数作为相应区间最小值时,虽然这个相应区间不一定唯一的,但是这个最大区间和一定是唯一的。

举个栗子:

对于数列{0, 0, 0, 0, 0}来说,我们不管是选择哪一个元素作为区间最小值,其相应的答案区间都有多种可能,但最大区间和肯定都是0(你选取第一个0作为区间最小值, 那区间可以是[1, 1],[1, 2],[1, 3]等等,但是区间和最大值肯定都是0)。

我们看原题的话就会发现原题说明是关于这个区间的输出,我们输出其中合法的任何一个就行了。

但是!但是!其实这是在坑你!表面上是spj,实际上并不是spj!对于每一个给定的数据,其输出的区间也必须也是要跟他给的一毛一样才行orz,具体的潜规则我也不是很清楚,反正用单调栈才能AC,不过本着真理至上的原则,还是两个算法都讲一遍。

================

st表+递归解法

这种做法是枚举区间,然后通过st表O(1)找出区间的最小值,配合前辍和能直接计算出一个可能答案。

那怎么枚举区间呢?

先说下,st表保存的应该是区间最小值的下标 。

比较直观的一点就是当我们选取这个数列的最小值作为区间的最小值的时候,这个区间[1, n]肯定是合法的,设这个最小值的位置为k,那么我们可以知道[1, k-1], [k+1, n]这两个区间是另外两个该被枚举的区间,然后找出这两个区间最小值的位置,又能将区间再次分割。。。每次所划分的区间所对应的区间最小值所在的位置都是不一样的,通过这个递归过程,如此一来,我们就得到了一个O(n)枚举区间的方法,之所以这个算法是O(nlogn)只是因为建st表要O(nlogn)罢了,其他操作要么O(n)要么O(1)的,只可惜实际上这道题不是spj,这种方法跟单调栈的做法虽然都是对的,但是所得区间不一定一样,不然数据水点的话说不定也能过(x)。

 #include<iostream>
#include<cstdio>
#include<cstring>
using namespace std; const int N = 1e5 + ;
int Logn[N];
long long a[N];
long long sum[N]; //前辍和
int st[N][]; //记录区间[l, r]中最小元素下标的st表
long long ans; //最终答案
int L, R; //最终答案区间 void pre() {
Logn[] = ;
for (int i = ; i < N; i++)
Logn[i] = Logn[i / ] + ;
} void Build_st(int n) //建立st表
{
int logn = Logn[n];
for(int j=; j<=logn ;j++)
for (int i = ; i + ( << j) - <= n; i++) {
if (a[st[i][j - ]] < a[st[i + ( << (j - ))][j - ]])
st[i][j] = st[i][j - ];
else st[i][j] = st[i + ( << (j - ))][j - ];
}
} int Query(int l, int r)
{
int s = Logn[r - l + ]; if (a[st[l][s]] < a[st[r + - ( << s)][s]]) return st[l][s];
else return st[r + - ( << s)][s];
} void solve(int l, int r) //递归处理
{
if (l > r) return; int m = Query(l, r);
long long res = (sum[r] - sum[l-]) * a[m];
if (res > ans) {
ans = res;
L = l;
R = r;
} solve(l, m - );
solve(m + , r);
} int main()
{
//freopen("data.txt", "r", stdin);
//freopen("WA.txt", "w", stdout);
ios::sync_with_stdio(false);
pre();
int n;
while (cin >> n) {
ans = -; for (int i = ; i <= n; i++) {
cin >> a[i];
sum[i] = a[i] + sum[i - ];
st[i][] = i;
} Build_st(n);
solve(, n);
cout << ans << endl;
cout << L << " " << R << endl;
cout << endl;
} return ;
}

================

单调栈解法

单调栈的话做法就很单纯了,用数组l[i], r[i]分别保存元素a[i]作为区间最小值时所能延伸到的最左端跟最右端,用单调栈扫描两次这两个数组的值就能全部出来了,实际写代码的时候博主写惯了不小心写成了单调队列,不过操作都一样的。。。

 #include<iostream>
#include<algorithm>
#include<queue>
using namespace std; typedef long long ll;
const int Maxn = 1e5 + ;
ll a[Maxn];
ll l[Maxn];
ll r[Maxn];
ll sum[Maxn]; int main()
{
int n;
bool first = true;
while (cin >> n) {
if (first) first = false;
else cout << endl; for (int i = ; i <= n; i++) {
cin >> a[i];
sum[i] = sum[i - ] + a[i];
} deque <ll > rq, lq;
for (int i = ; i <= n; i++) {
while (!rq.empty() && a[rq.back()] > a[i]) {
r[rq.back()] = i - ;
rq.pop_back();
} rq.push_back(i);
}
while (!rq.empty()) {
r[rq.front()] = n;
rq.pop_front();
} for (int i = n; i >= ; i--) {
while (!lq.empty() && a[lq.back()] > a[i]) {
l[lq.back()] = i + ;
lq.pop_back();
} lq.push_back(i);
}
while (!lq.empty()) {
l[lq.front()] = ;
lq.pop_front();
} ll ans = ;
ll ans_l = , ans_r = ; for(int i=; i<=n ;i++)
if ((sum[r[i]] - sum[l[i] - ]) * a[i] > ans) {
ans = (sum[r[i]] - sum[l[i] - ]) * a[i];
ans_l = l[i];
ans_r = r[i];
} cout << ans << endl;
cout << ans_l << " " << ans_r << endl;
} return ;
}

Uva 1609 Feel Good的更多相关文章

  1. UVa 1609 - Foul Play

    链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  2. UVa 1609 (博弈) Foul Play

    姑且把它归类为一道博弈吧,毕竟这也是在找必胜方案. 十分有意思的一道题目,设计一种方案让你支持的1队获胜. 题目给出了两个很重要的条件: 1队能打败至少一半的队伍 对于1队不能打败的黑队,一定存在一个 ...

  3. UVA 1609 Foul Play 不公平竞赛 (构(luan)造(gao)+递归)

    题意:有n支队伍(n是2的整数幂,2<=n<=4),打淘汰赛,胜者进入下一轮,其中1号队伍能打败至少一半的队伍,对于它不能打败的队伍l,一定存在一支它能够打败的队伍w,使得w能直接打败l, ...

  4. 紫书 例题8-17 UVa 1609 (构造法)(详细注释)

    这道题用构造法, 就是自己依据题目想出一种可以得到解的方法, 没有什么规律可言, 只能根据题目本身来思考. 这道题的构造法比较复杂, 不知道刘汝佳是怎么想出来的, 我想的话肯定想不到. 具体思路紫书上 ...

  5. uva 1354 Mobile Computing ——yhx

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAABGcAAANuCAYAAAC7f2QuAAAgAElEQVR4nOy9XUhjWbo3vu72RRgkF5

  6. UVA 10564 Paths through the Hourglass[DP 打印]

    UVA - 10564 Paths through the Hourglass 题意: 要求从第一层走到最下面一层,只能往左下或右下走 问有多少条路径之和刚好等于S? 如果有的话,输出字典序最小的路径 ...

  7. UVA 11404 Palindromic Subsequence[DP LCS 打印]

    UVA - 11404 Palindromic Subsequence 题意:一个字符串,删去0个或多个字符,输出字典序最小且最长的回文字符串 不要求路径区间DP都可以做 然而要字典序最小 倒过来求L ...

  8. UVA&&POJ离散概率与数学期望入门练习[4]

    POJ3869 Headshot 题意:给出左轮手枪的子弹序列,打了一枪没子弹,要使下一枪也没子弹概率最大应该rotate还是shoot 条件概率,|00|/(|00|+|01|)和|0|/n谁大的问 ...

  9. UVA计数方法练习[3]

    UVA - 11538 Chess Queen 题意:n*m放置两个互相攻击的后的方案数 分开讨论行 列 两条对角线 一个求和式 可以化简后计算 // // main.cpp // uva11538 ...

随机推荐

  1. NPM install -save 和 -save-dev 区别

    最近在写Node程序的时候,突然对 npm install 的-save和-save-dev 这两个参数的使用比较混乱.其实博主在这之前对这两个参数的理解也是模糊的,各种查资料和实践后对它们之间的异同 ...

  2. Shiro -- (四) 数据库支持

    主要就是JdbcRealm这个类 先看一下部分源码: 先建表:users(用户名 / 密码).user_roles(用户 / 角色).roles_permissions(角色 / 权限),并且往use ...

  3. linux中的链接命令

    ln 解释 命令名称:ln 命令英文原意:link 命令所在路径:/bin/ln 执行权限:所有用户 功能描述:生成链接文件 语法 ln -s [源文件] [目标文件] -s 创建软链接 示例 # 创 ...

  4. 9maven依赖传递性、依赖原则

    maven的依赖传递: A.jar->B.jar->C.jar 要使 A.jar ->C.jar:当且仅当 B.jar 依赖于C.jar的范围是compile,如果B依赖于C的范围不 ...

  5. 将Markdown编辑器搬进您的博客-让我们更优雅的书写文章

    各位小伙伴们,冷月今天给大家发一波福利.我们都知道markdown编辑器非常的好用,是我们写作的好帮手.这样的一款好用的文章编辑器,我们怎么才能让自己的博客也支持呢,冷月今天来教大家如何将Markdo ...

  6. Session注销后,浏览器后退仍显示Session信息

    初学JavaWeb,在一次测试登录功能的时候,发现了登进去后,点击退出按钮注销session,浏览器按后退能够显示session信息,并且点击登录还能够不输入密码登录(前端页面没写JS限制输入后登录) ...

  7. CF926B Add Points

    一道尚未评定的水题 更好的阅读体验 思路 来分析分析样例: 3 -5 10 5 我们把它升序排列,会得到这个东西↑ 不仔细地观察后可以发现:加一个(0,0)的点显然是最优的 再用脚趾头想想为什么,我们 ...

  8. external IP 和 local IP 的区别

    外部(external)和本地(local) IP地址都具有相同的用途,不同之处在于范围.整个Internet使用外部或公共IP地址来定位计算机系统和设备.专用网络内部使用本地或内部IP地址来定位与其 ...

  9. modbus 协议说明及常用格式

    --- 说明: modbus协议一般适用于一个主设备访问多个从设备的硬件开发环境,类似于zigbee网络中的一个路由器多个协调器的一对多模型. modbus常用的寄存器类型为 3X 保持寄存器和 4X ...

  10. 安全师(网络安全类pdf电子书籍)

    2020-02-17  天气晴,西安. 今天找到一个可以下载网络安全(渗透,kali,web)电子书籍网站. https://www.secshi.com/