题目传送门

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. Unity中几种简单的相机跟随

    #unity中相机追随 固定相机跟随,这种相机有一个参考对象,它会保持与该参考对象固定的位置,跟随改参考对象发生移动 using UnityEngine; using System.Collectio ...

  2. Android Wifi简单的梳理【转】

    本文转载自:http://blog.csdn.net/gabbzang/article/details/10005411 代表一个已经配置过的网络.包含网络ID(networkId).该网络ID代表的 ...

  3. laravel登录后台500错误!

    登录页面正常显示,填写完用户名密码 点登录后 页面一片空白,没有任何输出.debug可以看到一个500错误,preview和response都是空的.追踪了一下 发现在public/index.php ...

  4. poj 2689Prime Distance(区间素数)埃氏筛法

    这道题的L和R都很大,所以如果直接开一个1~R的数组明显会超时.但是R-L并不大,所以我们考虑把这个区间(L--R)移动到(1--(R-L+1))这个区间再开数组(就是把每个数减L再加1).接下来先用 ...

  5. ACM学习历程—HDU4717 The Moving Points(模拟退火 || 三分法)

    Description There are N points in total. Every point moves in certain direction and certain speed. W ...

  6. 洛谷P2530 [SHOI2001]化工厂装箱员

    题目描述 118号工厂是世界唯一秘密提炼锎的化工厂,由于提炼锎的难度非常高,技术不是十分完善,所以工厂生产的锎成品可能会有3种不同的纯度,A:100%,B:1%,C:0.01%,为了出售方便,必须把不 ...

  7. BZOJ2467五角形生成树——数学

    题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2467 可以得出只需在每个五角形中去掉任意一条边,在某个五角形中去掉包括内边的两条边即可. 代码 ...

  8. 重学JAVA基础(一):PATH和CLASSPATH

    我想大多数Java初学者都会遇到的问题,那就是怎么配置环境,执行java -jar xxx.jar  都会报NoClassDefFindError,我在最开始学习的时候,也遇到了这些问题. 1.PAT ...

  9. WPF中Visible设为Collapse时,VisualTreeHelper.GetChildrenCount为0

    今天遇到一个奇怪的问题, 在给一个控件内的子元素绑定事件时,失败. 发现原因是,这个控件初始化时Visible="Collapse",这时控件内的可视树就没有生成.导致绑定事件失败 ...

  10. Ubuntu ssh免密登录

    ssh免密登录工作原理 server A免登录到server B: 1.在A上生成公钥私钥. 2.将公钥拷贝给server B,要重命名成authorized_keys(从英文名就知道含义了) 3.S ...