WC2019 全国模拟赛第一场 T1 题解
由于只会T1,没法写游记,只好来写题解了...
题目链接
题目大意
给你一个数列,每次可以任取两个不相交的区间,取一次的贡献是这两个区间里所有数的最小值,求所有取法的贡献和,对 \(10^9+7\) 取模。
数列长度 \(2\times 10^5\) ,值域 \(1\) ~ \(10^9\) 。
\(O(n^4)\) 做法
预处理区间最小值,枚举选的两个区间。
#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std;
const int M=1000000007;
int n,a[60][60],ans;
int main()
{
int i,j,k,l;
cin>>n;
for (i=1;i<=n;++i)
{
cin>>a[i][i];
}
for (i=1;i<n;++i)
{
for (j=i+1;j<=n;++j)
{
a[i][j]=min(a[i][j-1],a[j][j]);
}
}
for (i=1;i<n;++i)
{
for (j=i;j<n;++j)
{
for (k=j+1;k<=n;++k)
{
for (l=k;l<=n;++l)
{
ans=(ans+min(a[i][j],a[k][l]))%M;
}
}
}
}
cout<<ans;
return 0;
}
\(O(nlogn)\) 做法
warning:接下来的文章里“的”字嵌套情况非常严重,文字叙述比较繁杂,看不懂十分正常,建议看懂一小部分然后自己推。
考虑每个元素作为贡献的区间是哪些,为了把每个区间分给唯一的元素,规定一个区间的贡献是最小值里最靠左的( e.g. 4 3 2 4 2 2 的贡献是 \(3\) 号元素,即最左边的 \(2\) )。所以,可以利用栈在 \(O(n)\) 的时间内预处理出每个元素作为贡献的区间的左端点和右端点的范围:
for (i=1;i<=n;++i)
{
while (top&&a[sta[top]].w>a[i].w)
{
a[sta[top--]].r=i-1;
}
sta[++top]=i;
}
while (top)
{
a[sta[top--]].r=n;
}
for (i=n;i>=1;--i)
{
while (top&&a[sta[top]].w>=a[i].w)
{
a[sta[top--]].l=i+1;
}
sta[++top]=i;
}
while (top)
{
a[sta[top--]].l=1;
}
每个元素作为贡献的区间就是 \([x,y](l_i\le x\le i\le y\le r_i)\),每个元素作为贡献的区间数就是 \(t_i=(i-l_i+1)\times(r_i-i+1)\) 。
然后,将元素按值从大到小排序,就能计算出区间数的后缀和 \(suf[i]\),但一个元素的总贡献并不是 \(t_i\times suf[i+1]\),因为这些区间可能与当前元素作为贡献的区间相交。
注意到,要想和当前元素作为贡献的区间相交,必须 \([x,y](l_i\le x\le y\le r_i)\) ,而这样的区间除了当前元素作为贡献的区间,贡献都排在当前元素之后(值比当前元素大或值相等但位置靠后),所以这样的区间除了当前元素作为贡献的区间,都是我们要找的与当前元素作为贡献的区间相交的贡献更靠后的区间。
注:下面这段话中“相交的区间对”指(与当前元素作为贡献的区间相交的贡献更靠后的区间,当前元素作为贡献的区间)这样的一对区间;“相交的区间”指与当前元素作为贡献的区间相交的贡献更靠后的区间。
接下来就要计算相交的区间有多少对。首先,相交的区间不可能跨过当前元素,否则就是当前元素作为贡献的区间;所以,相交的区间要么是 \([x,y](l_i\le x\le y<i)\) ,要么是 \([x,y](i<x\le y\le r_i)\)。先计算 \([x,y](l_i\le x\le y<i)\) 与当前元素作为贡献的区间相交的对数,先考虑 \(y\) 固定时,个数为 \((r_i-i+1)\times(y-l_i+1)^2\) ,其中:\(y-l_i+1\) 既是相交的区间左端点的个数,也是与相交的区间相交的当前元素作为贡献的区间的左端点的个数;\(r_i-i+1\) 是与相交的区间相交的当前元素作为贡献的区间的右端点的个数。所以,总数是 \((r_i-i+1)\times\sum\limits_{y=l_i}^{i-1}(y-l_i+1)^2\) ,乘号右边是自然数平方和,可以用公式计算,所以就是 \((r_i-i+1)\times\frac{(i-l_i)\times(i-l_i+1)\times(2i-2l_i+1)}6\) 。\([x,y](i<x\le y\le r_i)\) 同理,总数为 \((i-l_i+1)\times\frac{(r_i-i)\times(r_i-i+1)\times(2r_i-2i+1)}6\) 。
所以,把相交的总对数减掉就可以了。
参考代码:
#include <iostream>
#include <cstdio>
#include <cctype>
#include <algorithm>
using namespace std;
int read()
{
int out=0;
char c;
while (!isdigit(c=getchar()));
for (;isdigit(c);c=getchar())
{
out=out*10+c-'0';
}
return out;
}
const int N=200010;
const int M=1000000007;
const int SIX=166666668; //6模1e9+7的逆元
struct Node
{
long long id,w,l,r,t;
bool operator<(const Node& b) const
{
return w<b.w;
}
} a[N];
long long n,suf[N],sta[N],top,ans;
int main()
{
int i;
n=read();
for (i=1;i<=n;++i)
{
a[i].w=read();
a[i].id=i;
}
for (i=1;i<=n;++i)
{
while (top&&a[sta[top]].w>a[i].w)
{
a[sta[top--]].r=i-1;
}
sta[++top]=i;
}
while (top)
{
a[sta[top--]].r=n;
}
for (i=n;i>=1;--i)
{
while (top&&a[sta[top]].w>=a[i].w)
{
a[sta[top--]].l=i+1;
}
sta[++top]=i;
}
while (top)
{
a[sta[top--]].l=1;
}
for (i=1;i<=n;++i)
{
a[i].t=(i-a[i].l+1)*(a[i].r-i+1)%M;
}
sort(a+1,a+n+1);
for (i=n;i>=1;--i)
{
suf[i]=(suf[i+1]+a[i].t)%M;
}
for (i=1;i<=n;++i)
{
ans=(ans+(a[i].w*suf[i+1]%M)*a[i].t)%M;
ans=(ans-(a[i].id-a[i].l)*(a[i].id-a[i].l+1)%M*(2*a[i].id-2*a[i].l+1)%M*SIX%M*(a[i].r-a[i].id+1)%M*a[i].w%M+M)%M; //重复区间在左
ans=(ans-(a[i].r-a[i].id)*(a[i].r-a[i].id+1)%M*(2*a[i].r-2*a[i].id+1)%M*SIX%M*(a[i].id-a[i].l+1)%M*a[i].w%M+M)%M; //重复区间在右
}
cout<<ans;
return 0;
}
WC2019 全国模拟赛第一场 T1 题解的更多相关文章
- NOI.AC省选模拟赛第一场 T1 (树上高斯消元)
link 很容易对于每个点列出式子 \(f_{x,y}=(f_{x,y-1}+f_{x,y}+f_{x,y+1}+f_{x+1,y})/4\)(边角转移类似,略) 这个转移是相互依赖的就gg了 不过你 ...
- NOI.AC NOIP模拟赛 第一场 补记
NOI.AC NOIP模拟赛 第一场 补记 candy 题目大意: 有两个超市,每个超市有\(n(n\le10^5)\)个糖,每个糖\(W\)元.每颗糖有一个愉悦度,其中,第一家商店中的第\(i\)颗 ...
- contesthunter暑假NOIP模拟赛第一场题解
contesthunter暑假NOIP模拟赛#1题解: 第一题:杯具大派送 水题.枚举A,B的公约数即可. #include <algorithm> #include <cmath& ...
- nowcoder(牛客网)提高组模拟赛第一场 解题报告
T1 中位数(二分) 这个题是一个二分(听说是上周atcoder beginner contest的D题???) 我们可以开一个数组b存a,sort然后二分b进行check(从后往前直接遍历check ...
- nowcoder(牛客网)普及组模拟赛第一场 解题报告
蒟蒻我可能考了一场假试 T1 绩点 这题没什么好说的,应该是只要会语言的就会做. T2 巨大的棋盘 一个模拟题吧qwq,但是要注意取模的时候先加上n或者m再取模,要不然会错的. #include< ...
- 【CQ18阶梯赛第一场】题解
[A-风格不统一如何写程序] 输入字符串,得到长度,对于每个字符:如果是大写,则改为:‘_’+小写:如果是‘_’则忽略‘_’,并且把后面的小写改为大写. #include<cstdio> ...
- 2021ICPC网络赛第一场部分题解-The 2021 ICPC Asia Regionals Online Contest (I)
写在前面 本来应该6题的,结果不知道哪个铸币发了H的clar,当即把我们的思路转向三维几何上.当时我们还在想这三维计算几何的正确率有点太高了还在感叹ICPC选手的含金量,直到赛后我才知道这H题的铸币出 ...
- CSP-S全国模拟赛第二场 【nan】
A.count 本场比赛最难的题... 隔板法组合数容斥 xjb 搞搞就好了 //by Judge #include<cstdio> #include<iostream> #d ...
- Contest1585 - 2018-2019赛季多校联合新生训练赛第一场(部分题解)
Contest1585 - 2018-2019赛季多校联合新生训练赛第一场 C 10187 查找特定的合数 D 10188 传话游戏 H 10192 扫雷游戏 C 传送门 题干: 题目描述 自然数中除 ...
随机推荐
- 【转】简述TCP的三次握手过程
TCP握手协议 在TCP/IP协议中,TCP协议提供可靠的连接服务,采用三次握手建立一个连接.第一次握手:建立连接时,客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状态,等待服务器确 ...
- java普通类如何调用Spring的Service层?
首先在Service层上面添加 @Service("myService") 然后,在main方法中调用,String[]中为配置文件,如下所示: ApplicationContex ...
- uva10288 Coupons 【概率 分数】
题目: 题意: 一共n种不同的礼券,每次得到每种礼券的概率相同.求期望多少次可以得到所有n种礼券.结果以带分数形式输出.1<= n <=33. 思路: 假设当前已经得到k种,获得新的一种的 ...
- 链表求和12 · Add Two Numbers
反向存储,从左往右加 [抄题]: 你有两个用链表代表的整数,其中每个节点包含一个数字.数字存储按照在原来整数中相反的顺序,使得第一个数字位于链表的开头.写出一个函数将两个整数相加,用链表形式返回和.给 ...
- mysql查询赋值、修改拼接字符串
sql中修改字符串类型的字段可以这么拼接:update tbName set UserName='abc'+UserName; 但mysql中就不行了,需要这样:update tbName set U ...
- max文件属性设置,
之前一直都没找到 用到的时候就是用net 弄了.哎.还在开发东西都是在9上面, 这次脚本必须在 max8 上面 逼的我找到了他 getFileAttribute <filename_string ...
- Redis数据结构(五)
存储set (1)和List类型不同的是,Set集合中不允许出现重复的元素 (2)set可包含的最大元素数量是4294967295 存储set常用命令: (1)添加/删除元素 添加 sadd myli ...
- Linux设备驱动模型底层架构及组织方式
1.什么是设备驱动模型? 设备驱动模型,说实话这个概念真的不好解释,他是一个比较抽象的概念,我在网上也是没有找到关于设备驱动模型的一个定义,那么今天就我所学.所了解 到的,我对设备驱动模型的一个理解: ...
- Shell编程-01-Shell脚本初步入门
目录 什么是Shell 什么是Shell脚本 Shell脚本语言的种类 常用操作系统默认Shell Shell 脚本的建立和执行 脚本规范 什么是Shell 简单来说Shell其实就是一个命令 ...
- HttpClient Timeout
1. Overview This tutorial will show how to configure a timeout with the Apache HttpClient 4. If you ...