Keen On Everything But Triangle

题目传送门

解题思路

利用主席树求区间第k小,先求区间内最大的值,再求第二大,第三大……直到找到连续的三个数可以构成一个三角形。因为对于一组数,如果不能构成三角形,就小的就是斐波那契数列,因为数的范围在10^9内,所以不会超过50个数,也就是说,我们之间这样暴力地查询,查询次数不会超过50,肯定能找到结果。

代码如下

#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
using namespace std;
typedef long long ll; inline int read(){
int res = 0, w = 0; char ch = 0;
while(!isdigit(ch)){
w |= ch == '-', ch = getchar();
}
while(isdigit(ch)){
res = (res << 3) + (res << 1) + (ch ^ 48);
ch = getchar();
}
return w ? -res : res;
} const int N = 100005; int rt[N], cnt;
struct T{
int l, r;
int lch, rch;
int sum;
}tree[N << 5];
ll a[N], b[N];
int build(int l, int r)
{
int p = ++cnt;
tree[p].l = l;
tree[p].r = r;
tree[p].lch = tree[p].rch = tree[p].sum = 0;
if(l == r)
return p;
int mid = (l + r) / 2;
tree[p].lch = build(l, mid);
tree[p].rch = build(mid + 1, r);
return p;
} int insert(int k, int x)
{
int p = ++cnt;
tree[p].l = tree[k].l;
tree[p].r = tree[k].r;
tree[p].lch = tree[k].lch;
tree[p].rch = tree[k].rch;
tree[p].sum = tree[k].sum + 1;
if(tree[k].l == tree[k].r)
return p;
int mid = (tree[k].l + tree[k].r) / 2;
if(x <= mid)
tree[p].lch = insert(tree[k].lch, x);
else
tree[p].rch = insert(tree[k].rch, x);
return p;
} int query(int k1, int k2, int x)
{
if(tree[k1].l == tree[k1].r)
return tree[k1].l;
int l1 = tree[k1].lch;
int l2 = tree[k2].lch;
if(tree[l2].sum - tree[l1].sum >= x)
return query(l1, l2, x);
else {
x -= tree[l2].sum - tree[l1].sum;
return query(tree[k1].rch, tree[k2].rch, x);
}
} int main()
{
int n, q;
while(scanf("%d%d", &n, &q) != EOF){
for(int i = 1; i <= n; i ++){
scanf("%lld", &a[i]);
b[i] = a[i];
}
sort(b + 1, b + n + 1);
int k = unique(b + 1, b + n + 1) - b - 1;
cnt = -1;
rt[0] = build(1, k);
for(int i = 1; i <= n; i ++){
int x = lower_bound(b + 1, b + k + 1, a[i]) - b;
rt[i] = insert(rt[i - 1], x);
}
for(int i = 1; i <= q; i ++){
int l, r;
l = read(), r = read();
if(r - l + 1 < 3){
printf("-1\n");
continue;
}
int x = query(rt[l - 1], rt[r], r - l + 1);
int y = query(rt[l - 1], rt[r], r - l);
ll ans = -1;
for(int i = r - l - 1; i >= 1; i --){
int z = query(rt[l - 1], rt[r], i);
if(b[y] + b[z] > b[x]){
ans = b[x] + b[y] + b[z];
break;
}
x = y;
y = z;
}
printf("%lld\n", ans);
}
}
return 0;
}

最新文章

  1. Lucene.net
  2. php.ini
  3. 电赛菜鸟营培训(四)&mdash;&mdash;STM32F103CB之ADC转换
  4. 在ASP.Net2.0中使用UrlRewritingNet实现链接重写
  5. QcheckBox
  6. ADT(android-bundler) HTML EDIT 编辑 xml HTML
  7. HTML5学习(九)----应用程序缓存
  8. 4. 绘制光谱曲线QGraphicsView类
  9. JavaScript数组知识网络
  10. 极客Web前端开发资源大荟萃
  11. [Android]Android SDk Manager中创建模拟器无法选择CPU问题解析
  12. iOS开发——Localizable.strings
  13. Centos7忘记密码
  14. Linux 安装 MongoDB
  15. 4-1 requests库的安装
  16. [转]10 Awesome Indicator Applets for Ubuntu’s Unity Desktop
  17. Django项目----快速实现增删改查组件(起步阶段!!!)
  18. linux 基础储备
  19. React.createElement: type is invalid -- expected a string (for built-in components) or a class/function (for composite components) but got: undefined.
  20. 解决coursera无法观看视频的问题

热门文章

  1. 用了WS_EX_LAYERED 后所有Twincontrl的wm_paint消息会停止(官方Layered Windows文档很多内容)good
  2. Qt之OpenSSL(有pro文件的路径格式,以及对libeay32和ssleay32的引用)
  3. winpcap在VS2012 Qt5 X64下的配置
  4. 发现 TSplitter 在嵌套时不好用, 索性写了个替代品(处理MouseDown,MouseMove,MouseUp,然后设定控件的Left值就可以了)
  5. php一个不错的分页
  6. SYN1621型 定位定向授时设备
  7. python trojan development 3rd —— use python to creative a simple shell
  8. 推荐一个Redis管理工具
  9. Java 8 并发编程
  10. 【对象属性复制】BeanUtils.copyProperties(obj1, obj2);