题目传送门

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. Java对象 的创建与构造方法

    一.创建对象的四种方法: a. new语句: b. 利用反射,调用描述类的Class对象的newInstance()实例方法: c. 调用对象的clone(): d. 反序列化: 其中new 和 ne ...

  2. 构建工具是如何用 node 操作 html/js/css/md 文件的

    构建工具是如何用 node 操作 html/js/css/md 文件的 从本质上来说,html/js/css/md ... 源代码文件都是文本文件,文本文件的内容都是字符串,对文本文件的操作其实就是对 ...

  3. JAVA- JSP中解决无法在Cookie当中保存中文字符的问题

    因为cookie的值是ASCII字符,不能直接把自定义cookie的值直接赋值为中文,但是要实现这个功能,还是有方法的. 1.java中已经给我们提供了方法,此时只需要导入该包就行 <%@ pa ...

  4. JAVA- 清除数组重复元素

    清除数组重复元素并打印新数组. import java.util.*; public class Repeat { public static void main(String[] args) { / ...

  5. legend2---开发日志15(功能需求明确,设计好类和结构的好处是)

    legend2---开发日志15(功能需求明确,设计好类和结构的好处是) 一.总结 一句话总结: 极快简化编程,节约大量时间 1.多个类型的物品,比如商店和寻宝的丹药,装备,特性书,英雄石等等 应该怎 ...

  6. Mybatis中的like模糊查询

    1.  参数中直接加入%% param.setUsername("%CD%");      param.setPassword("%11%"); <sel ...

  7. 【Lintcode】033.N-Queens

    题目: The n-queens puzzle is the problem of placing n queens on an n×n chessboard such that no two que ...

  8. 【opencv学习笔记五】一个简单程序:图像读取与显示

    今天我们来学习一个最简单的程序,即从文件读取图像并且创建窗口显示该图像. 目录 [imread]图像读取 [namedWindow]创建window窗口 [imshow]图像显示 [imwrite]图 ...

  9. WPF中数据绑定问题

    在数据库中字段不区分大小写,可以页面是区分的,一开始以为不区分,可我从数据库查出了数据在前台就是不显示想了半天才发现的. <sdk:DataGrid FrozenColumnCount =&qu ...

  10. oracle 日期问题 网上找到自己查阅时方便

    第一部分:oracle sql日期比较: oracle sql日期比较:在今天之前: select*from up_date whereupdate< to_date('2007-09-07 0 ...