题目传送门

emmm..不开结构体的线段树真香!

首先我们知道“三元上升子序列”的个数就是对于序列中的每个数,它左边比他小的数*它右边比他大的数。但是如何快速求出这两个数?

我们用到权值线段树来维护。一般我们的线段树都是以下标延伸的,但是这里我们用的是权值,一般需要离散化,效果相当于一个桶。

这部分讲解请移步绝世好文

第一次我们从\(1\)$n$循环是为了找它左边的,而找比他小的值是在线段树的$1$\(seq[i]-1\)中找。第二次我们从\(n\)$1$循环是为了找它右边的,而找比他大的值是用在线段树的$seq[i]+1$\(n\)中找实现的。

不用结构体因为方便清空,\(tong\)数组记录线段树的权值,注意开\(4\)倍。

#include<cstdio>
#include<algorithm>
#include<cstring>
#define maxn 100090 using namespace std;
typedef long long ll; int n;
ll ans,sma[maxn],bigg[maxn];
int seq[maxn],tmp[maxn],tong[maxn<<2]; int ask(int p,int l,int r,int L,int R)
{
if(L<=l&&r<=R) return tong[p];
int mid=(l+r)>>1,qwq=0;
if(L<=mid) qwq+=ask(p<<1,l,mid,L,R);
if(R>mid) qwq+=ask(p<<1|1,mid+1,r,L,R);
return qwq;
} void change(int p,int l,int r,int x)
{
if(l==x&&r==x)
{
tong[p]++;
return ;
}
int mid=(l+r)>>1;
if(x<=mid) change(p<<1,l,mid,x);
else if(x>mid) change(p<<1|1,mid+1,r,x);
tong[p]=tong[p<<1]+tong[p<<1|1];
} int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
scanf("%d",&seq[i]),tmp[i]=seq[i];
sort(tmp+1,tmp+1+n);
int m=unique(tmp+1,tmp+1+n)-(tmp+1);
for(int i=1;i<=n;i++)
seq[i]=lower_bound(tmp+1,tmp+1+m,seq[i])-tmp;
for(int i=1;i<=n;i++)
{
if(seq[i]!=1) sma[i]=ask(1,1,n,1,seq[i]-1);
change(1,1,n,seq[i]);
}
memset(tong,0,sizeof(tong));
for(int i=n;i>=1;i--)
{
if(seq[i]!=n) bigg[i]=ask(1,1,n,seq[i]+1,n);
change(1,1,n,seq[i]);
}
for(int i=1;i<=n;i++)
ans+=sma[i]*bigg[i];
printf("%lld\n",ans);
return 0;
}

用线段树求逆序对是同理的,只需要求出每个序列中的位置的左边比它大的数个数就行了。

#include<cstdio>
#include<algorithm>
#include<cstring>
#define maxn 500090 using namespace std;
typedef long long ll; int n;
ll ans,lef[maxn];
int tmp[maxn],seq[maxn],tong[maxn<<2]; int ask(int p,int l,int r,int L,int R)
{
if(L<=l&&r<=R) return tong[p];
int mid=(l+r)>>1,qwq=0;
if(L<=mid) qwq+=ask(p<<1,l,mid,L,R);
if(R>mid) qwq+=ask(p<<1|1,mid+1,r,L,R);
return qwq;
} void change(int p,int l,int r,int x)
{
if(l==x&&r==x)
{
tong[p]++;
return ;
}
int mid=(l+r)>>1;
if(x<=mid) change(p<<1,l,mid,x);
else if(x>mid) change(p<<1|1,mid+1,r,x);
tong[p]=tong[p<<1]+tong[p<<1|1];
} int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",&seq[i]),tmp[i]=seq[i];
sort(tmp+1,tmp+1+n);
int m=unique(tmp+1,tmp+1+n)-(tmp+1);
for(int i=1;i<=n;i++)
seq[i]=lower_bound(tmp+1,tmp+1+m,seq[i])-tmp;
for(int i=1;i<=n;i++)
{
if(seq[i]!=n) lef[i]=ask(1,1,n,seq[i]+1,n);
change(1,1,n,seq[i]);
}
for(int i=1;i<=n;i++)
ans+=lef[i];
printf("%lld\n",ans);
return 0;
}

Luogu P1637 三元上升子序列【权值线段树】By cellur925的更多相关文章

  1. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

  2. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

  3. bzoj 4627: [BeiJing2016]回转寿司 -- 权值线段树

    4627: [BeiJing2016]回转寿司 Time Limit: 10 Sec  Memory Limit: 256 MB Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店. ...

  4. 【HDU6701】Make Rounddog Happy【权值线段树+双向单调队列】

    题意:给你一个序列,求满足要求的子序列个数,其中要求为: 1.子序列的max-子序列长度len<=k 2.子序列中不出现重复的数字 题解:首先看到子序列max,很容易想到枚举最大值然后分治,这个 ...

  5. 【树状数组套权值线段树】bzoj1901 Zju2112 Dynamic Rankings

    谁再管这玩意叫树状数组套主席树我跟谁急 明明就是树状数组的每个结点维护一棵动态开结点的权值线段树而已 好吧,其实只有一个指针,指向该结点的权值线段树的当前结点 每次查询之前,要让指针指向根结点 不同结 ...

  6. 【BZOJ-2892&1171】强袭作战&大sz的游戏 权值线段树+单调队列+标记永久化+DP

    2892: 强袭作战 Time Limit: 50 Sec  Memory Limit: 512 MBSubmit: 45  Solved: 30[Submit][Status][Discuss] D ...

  7. BZOJ 3110 ZJOI 2013 K大数查询 树套树(权值线段树套区间线段树)

    题目大意:有一些位置.这些位置上能够放若干个数字. 如今有两种操作. 1.在区间l到r上加入一个数字x 2.求出l到r上的第k大的数字是什么 思路:这样的题一看就是树套树,关键是怎么套,怎么写.(话说 ...

  8. 动态求区间K大值(权值线段树)

    我们知道我们可以通过主席树来维护静态区间第K大值.我们又知道主席树满足可加性,所以我们可以用树状数组来维护主席树,树状数组的每一个节点都可以开一颗主席树,然后一起做. 我们注意到树状数组的每一棵树都和 ...

  9. 线段树(单标记+离散化+扫描线+双标记)+zkw线段树+权值线段树+主席树及一些例题

    “队列进出图上的方向 线段树区间修改求出总量 可持久留下的迹象 我们 俯身欣赏” ----<膜你抄>     线段树很早就会写了,但一直没有总结,所以偶尔重写又会懵逼,所以还是要总结一下. ...

随机推荐

  1. Express的基本使用

    前言 列表项目Express是一个简介而灵活的node.js Web应用框架提供的一系列强大特性帮助你创建各种 Web 应用,和丰富的HTTP工具. 正文 一个简单的express框架实例 ``` / ...

  2. github 博客模板

    http://www.jianshu.com/p/d658ba3b4351 http://jekyllthemes.org/

  3. SDUT OJ类型转换函数的应用

    题目描述 处理一个复数与一个double数相加的运算,结果存放在一个double型变量d1中,输出d1的值.定义Complex(复数)类,在成员函数中包含重载类型转换运算符:operator doub ...

  4. 分享知识-快乐自己:SpringMVC 结合使用拦截器(判断是否用户是否已登陆)

    基础拦截器操作: 拦截器是一种AOP操作实现,那么在AOP之中用户一定不需要去关注拦截器的存在,用户只需要按照自己已经习惯的处理方式进行代码的编写即可. 首先我们先创建一个自定义的拦截器: packa ...

  5. 语义分割(semantic segmentation) 常用神经网络介绍对比-FCN SegNet U-net DeconvNet,语义分割,简单来说就是给定一张图片,对图片中的每一个像素点进行分类;目标检测只有两类,目标和非目标,就是在一张图片中找到并用box标注出所有的目标.

    from:https://blog.csdn.net/u012931582/article/details/70314859 2017年04月21日 14:54:10 阅读数:4369 前言 在这里, ...

  6. ES设置字段搜索权重——Query-Time Boosting

    Query-Time Boosting In Prioritizing Clauses, we explained how you could use the boost parameter at s ...

  7. (转)HLS协议,html5视频直播一站式扫盲

    本文来自于腾讯bugly开发者社区,原文地址:http://bugly.qq.com/bbs/forum.php?mod=viewthread&tid=1277 视频直播这么火,再不学就 ou ...

  8. ACM学习历程—HDU 5289 Assignment(线段树 || RMQ || 单调队列)

    Problem Description Tom owns a company and he is the boss. There are n staffs which are numbered fro ...

  9. bzoj 2850: 巧克力王国 K-D树

    题目大意 http://www.lydsy.com/JudgeOnline/problem.php?id=2850 题解 对于每个人,我们发现它能够接受的巧克力中 如果对参数分别讨论,那么一定是一个连 ...

  10. poj 1269 Intersecting Lines——叉积求直线交点坐标

    题目:http://poj.org/problem?id=1269 相关知识: 叉积求面积:https://www.cnblogs.com/xiexinxinlove/p/3708147.html什么 ...