HDU6058 Kanade's sum(思维 链表)
Kanade's sum
Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)
Total Submission(s): 871    Accepted Submission(s): 357
Let f(l,r,k) be the k-th largest element of A[l..r].
Specially , f(l,r,k)=0 if r−l+1<k.
Give you k , you need to calculate ∑nl=1∑nr=lf(l,r,k)
There are T test cases.
1≤T≤10
k≤min(n,80)
A[1..n] is a permutation of [1..n]
∑n≤5∗105
For each test case,there are only two integers n,k on first line,and the second line consists of n integers which means the array A[1..n]
5 2
1 2 3 4 5
【题意】给你一个序列,(1~n的一个全排列),然后询问所有的子区间第k大的数,累加起来,如果区间长度不足k,则为0,求累加 和。
【分析】由于区间数量太多,我们考虑算每个数的贡献。一个数的贡献显然就是它所能存在的区间数量,在这些区间内它是第k大。首先,要保证它是这个区间里的第k大,那么我们必须在他左边和右边找到共k-1个比他大的,然后围成区间就行。我们可以再这个数左边找到最多k个比他大的,右边找到最多k个比他大的(如果存在k个的话)。比如x是当前枚举的数,k=3,a,b,c是左边比他大的数的位置,d,e,f是右边比他大的数的位置(都是位置离x最近的),现在就是这样a b c x d e f,那么现在x的贡献就是能够围成的所有合法区间的数量,比如我们可以左边取x,右边取d e,那么此时的贡献就是(x-c)*(f-e),也可以左边取c,x,右边取d,此时的贡献就是(c-b)*(e-d)...
这样我们把贡献累加起来就是了,但是现在的问题是如何找这k个数。
我们发现当我们枚举到x时,找k个数是直接跨过了比x小的数,直接跳到了比x大的,所以我们可以从小到大枚举x,用一个链表把所有比x大的数连起来,当我们处理完x的时候,我们就把x的位置从链表里删除,这样我们的链表里存的都是比x大的数,而且相对位置也没变。这样我们就是O(k)找到这k个数.总时间复杂度,O(n*k)。
#include <bits/stdc++.h>
#define inf 0x3f3f3f3f
#define met(a,b) memset(a,b,sizeof a)
#define pb push_back
#define mp make_pair
#define inf 0x3f3f3f3f
#define lson l, m, rt<<1
#define rson m+1, r, rt<<1|1
using namespace std;
typedef long long ll;
const int N = 5e5+;;
const int M = ;
const int mod = 1e9+;
const double pi= acos(-1.0);
typedef pair<int,int>pii;
int n,k;
int pos[N],pre[N],suf[N];
int l[N],r[N];
void del(int x){
int pree=pre[x];
int suff=suf[x];
if(pree)suf[pree]=suff;
if(suff<=n)pre[suff]=pree;
pre[x]=suf[x]=;
}
int main() {
int T;
scanf("%d",&T);
while(T--){
ll ans=;
scanf("%d%d",&n,&k);
for(int i=,x;i<=n;i++){
scanf("%d",&x);
pos[x]=i;
}
for(int i=;i<=n;i++){
pre[i]=i-;
suf[i]=i+;
}
for(int x=;x<=n-k+;x++){
l[]=r[]=;
int p=pos[x];
for(int i=p;i&&l[]<=k+;i=pre[i])l[++l[]]=i;
for(int i=p;i<n+&&r[]<=k+;i=suf[i])r[++r[]]=i;
l[++l[]]=;
r[++r[]]=n+;
for(int i=;i<l[];i++){
if(i+r[]>=k+&&i<=k){
ans+=1LL*(l[i]-l[i+])*1LL*(r[k-i+]-r[k-i+])*x;
}
}
del(p);
}
printf("%lld\n",ans);
}
return ;
}
HDU6058 Kanade's sum(思维 链表)的更多相关文章
- hdu6058 Kanade's sum 区间第k大
		/** 题目:Kanade's sum 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6058 题意:给定[1,n]的排列,定义f(l,r,k)表示区间[l ... 
- 【set】【链表】hdu6058 Kanade's sum
		f(l,r,K)表示区间l,r里面的K大值,问你所有连续子区间的f之和. l(i)表示i左侧第一个比它大的数的位置,r(i)表示i右侧第一个比它大的数的位置.可以用set处理出来. 把数从大到小排序, ... 
- hdu-6058 Kanade's sum
		题意:略 思路:要我们求每个区间第K大数之和,其实可以转换为求多少个区间的第K大数是X,然后我们在求和就好了. 那么我们可以从小到大枚举所有可能成为第K大的数.为什么从小到大呢? 因为从小到大我们就略 ... 
- 2017 Multi-University Training Contest - Team 3—HDU6058 Kanade's sum
		题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6058 题目意思:给你一个排列,求所有区间长度大于等于k的区间第k大的数的和…… 思路:一开始看到区间k ... 
- HDU 6058 - Kanade's sum  |  2017 Multi-University Training Contest 3
		/* HDU 6058 - Kanade's sum [ 思维,链表 ] | 2017 Multi-University Training Contest 3 题意: 给出排列 a[N],求所有区间的 ... 
- hdu 6058 Kanade's sum(模拟链表)
		Kanade's sum Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Tota ... 
- HDU 6058 Kanade's sum 二分,链表
		Kanade's sum Problem Description Give you an array A[1..n]of length n. Let f(l,r,k) be the k-th larg ... 
- 2017 Multi-University Training Contest - Team 3 Kanade's sum hd6058
		地址:http://acm.split.hdu.edu.cn/showproblem.php?pid=6058 题目: Kanade's sum Time Limit: 4000/2000 MS (J ... 
- HDU 6058 Kanade's sum —— 2017 Multi-University Training 3
		Kanade's sum Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Tota ... 
随机推荐
- Linux目录结构nginx
			alias 别名( 永久 ) 1.vim /root/.bashrc 2. alias vimens33='vim /etc/sysconfig/network-scripts/ifcfg-ens33 ... 
- 超酷算法-BK树
			前几天无意间遇到一个博客,觉得写得挺好的,自己之前的时候有个不好的习惯,那就是遇到了好资源第一反应就是收藏起来然后却很少再看!!这是坏习惯,要改!于是今天就开始通读了,读的第二篇是BK树.觉得有点意思 ... 
- SpringBoot 在CentOS7部署,注册为服务,开机启动
			1.首先在maven工程的pom文件中引入以下标签并保存 <build> <plugins> <plugin> <groupId>org.springf ... 
- 用一个时钟在FPGA中计算直方图
			直方图对数字数据的分析通常是一种有用的工具.不过,要从一个直方图获得可靠的结果,必须获得大量数据,通常是要10万到100万个点.如果需要分析一个ADC的数字输出,可以采用一片FPGA(图1). 图中显 ... 
- 汕头市队赛SRM 20 T2不净的圣杯
			不净的圣杯 SRM 20 背景 作为一张BUG级别的卡,官方打算把它修改得人畜无害一些…… 虽然名字还没想好,但是能力大概是对敌方所有单位造成d点伤害,d为自己牌组中所有卡的编号的最大公约数.这无疑是 ... 
- linux 下用 c 实现 ls -l 命令
			#include <stdio.h> #include <sys/types.h> #include <dirent.h> #include <sys/sta ... 
- bzoj 1483 链表启发式合并
			首先我们可以比较容易的在n的时间内算出来开始的答案,我们维护一些链表,分别表示不同的颜色,那么我们在计算答案的时候,只需要扫一遍所有的链表,判断链表相邻两项是否在序列中相邻,不相邻的话肯定在这其中的一 ... 
- C#   文件操作常用方法总结
			需引用 System.IO Path为绝对路径 检测指定目录是否存在 Directory.Exists(Path) 创建目录 Directory.CreateDirectory(Path) 删除目录 ... 
- Python3【模块】concurrent.futures模块,线程池进程池
			Python标准库为我们提供了threading和multiprocessing模块编写相应的多线程/多进程代码,但是当项目达到一定的规模,频繁创建/销毁进程或者线程是非常消耗资源的,这个时候我们就要 ... 
- Windows10下配置Linux下C语言开发环境
			今天为大家介绍如在Windows10下配置Linux下C语言开发环境,首先安装linux子系统:启用开发者模式 1.打开设置 2.点击更新和安全3.点击开发者选项 4.启用开发人员模式 5.更改系统功 ... 
