test20190830 NOIP 模拟赛
100+70+0=170。这套题早就被上传到BZOJ上了,可惜我一到都没做过。
BZOJ4765 普通计算姬
小G的计算姬可以解决这么个问题:给定一棵n个节点的带权树,节点编号为1到n,以root为根,设sum[p]表示以点p为根的这棵子树中所有节点的权值和。计算姬支持下列两种操作:
- 给定两个整数u,v,修改点u的权值为v。
- 给定两个整数l,r,计算sum[l]+sum[l+1]+....+sum[r-1]+sum[r]
尽管计算姬可以很快完成这个问题,可是小G并不知道它的答案是否正确,你能帮助他吗?
N<=105,M<=105
题解
分块。
查询就整块答案+边角余料,对于边角余料我直接树状数组。
我对每个元素维护他到根的链上每个块中的元素出现了多少次,这样修改就扫一遍所有块就行了。
这样空间会很卡,还必须开 unsigned long long
,这出题人真的毒。
#include<bits/stdc++.h>
using namespace std;
template<class T> T read(){
T x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
for(;isdigit(c);c=getchar()) x=x*10+c-'0';
return x;
}
template<class T> T read(T&x){
return x=read<T>();
}
#define co const
#define il inline
typedef unsigned long long LL;
co int N=100000+1,M=599+1;
int n,m,val[N];
vector<int> to[N];
int pos[N],dfn,lst[N];
namespace T{
LL sum[N];
il void change(int p,int v){
for(int i=p;i<=n;i+=i&-i) sum[i]+=v;
}
il LL query(int p){
LL ans=0;
for(int i=p;i;i-=i&-i) ans+=sum[i];
return ans;
}
}
int blo,num,bel[N];
int L[M],R[M],cnt[N][M];
LL sum[M];
void dfs(int x,int fa){
pos[x]=++dfn,T::change(pos[x],val[x]);
copy(cnt[fa]+1,cnt[fa]+num+1,cnt[x]+1),++cnt[x][bel[x]];
for(int i=1;i<=num;++i) sum[i]+=(LL)val[x]*cnt[x][i]; // edit 1:long long
for(int i=0;i<(int)to[x].size();++i){
int y=to[x][i];
if(y==fa) continue;
dfs(y,x);
}
lst[x]=dfn;
}
int main(){
freopen("common.in","r",stdin),freopen("common.out","w",stdout);
read(n),read(m);
blo=ceil(sqrt(n)/1.9),num=(n+blo-1)/blo;
for(int i=1;i<=num;++i) L[i]=R[i-1]+1,R[i]=min(i*blo,n);
for(int i=1;i<=n;++i){
read(val[i]);
bel[i]=(i+blo-1)/blo;
}
int root;
for(int i=1;i<=n;++i){
int x=read<int>(),y=read<int>();
if(!x) {root=y;continue;}
to[x].push_back(y),to[y].push_back(x);
}
dfs(root,0);
while(m--){
if(read<int>()==1){
int x=read<int>(),v=read<int>();
T::change(pos[x],v-val[x]);
for(int i=1;i<=num;++i) sum[i]+=(LL)(v-val[x])*cnt[x][i]; // edit 1
val[x]=v;
}
else{
int l=read<int>(),r=read<int>();
LL ans=0;
if(bel[l]==bel[r]){
for(int i=l;i<=r;++i) ans+=T::query(lst[i])-T::query(pos[i]-1);
}
else{
for(int i=bel[l]+1;i<=bel[r]-1;++i) ans+=sum[i];
for(int i=l;i<=R[bel[l]];++i) ans+=T::query(lst[i])-T::query(pos[i]-1);
for(int i=L[bel[r]];i<=r;++i) ans+=T::query(lst[i])-T::query(pos[i]-1);
}
printf("%llu\n",ans);
}
}
return 0;
}
实际上可以不用树状数组,对 DFS 序再次分块,维护前缀和即可 O(1)。
BZOJ4766 文艺计算姬
文艺计算姬能计算一个带标号完全二分图的生成树个数。更具体地,给定一个一边点数为n,另一边点数为m,共有n*m条边的带标号完全二分图K_{n,m},计算姬能快速算出其生成树个数。小W不知道计算姬算的对不对,你能帮助他吗?
1 <= n,m,p <= 10^18
题解
考试的时候手玩70分巨爽。
回想起来我学习矩阵树还是初中的时候,那时我还不会在BZOJ上面找题做。不然我绝对会把这道经典题做了。
先把上面 n-1 行每一行都加到第 n 行去。 然后第 n 行变成 n-1 个 m-1 然后一个 1 再来 m-1 个 1-n。
把下面 m-1 行每一行都加到第 n 行去。 然后第 n 行变成只有后 m 个位置是 1。
把第 n 行加到前 n-1 行去,就把后面那堆-1消掉了。
然后变成下三角矩阵,行列式就是主对角线的乘积。 \(ans=n^{m-1}m^{n-1}\)。
#include<bits/stdc++.h>
using namespace std;
template<class T> T read(){
T x=0,w=1;char c=getchar();
for(;!isdigit(c);c=getchar())if(c=='-') w=-w;
for(;isdigit(c);c=getchar()) x=x*10+c-'0';
return x*w;
}
template<class T> T read(T&x){
return x=read<T>();
}
#define co const
#define il inline
#define int long long
int mod;
il int mul(int a,int b){
int ans=0;
for(;b;b>>=1,a=(a+a)%mod)
if(b&1) ans=(ans+a)%mod;
return ans;
}
il int fpow(int a,int b){
int ans=1;
for(;b;b>>=1,a=mul(a,a))
if(b&1) ans=mul(ans,a);
return ans;
}
signed main(){
int n=read<int>(),m=read<int>();read(mod);
printf("%lld\n",mul(fpow(n,m-1),fpow(m,n-1)));
return 0;
}
BZOJ4767 两双手
老W是个棋艺高超的棋手,他最喜欢的棋子是马,更具体地,他更加喜欢马所行走的方式。老W下棋时觉得无聊,便决定加强马所行走的方式,更具体地,他有两双手,其中一双手能让马从(u,v)移动到(u+Ax,v+Ay)而另一双手能让马从(u,v)移动到(u+Bx,v+By)。
小W看见老W的下棋方式,觉得非常有趣,他开始思考一个问题:假设棋盘是个无限大的二维平面,一开始马在原点(0,0)上,若用老W的两种方式进行移动,他有多少种不同的移动方法到达点(Ex,Ey)呢?两种移动方法不同当且仅当移动步数不同或某一步所到达的点不同。
老W听了这个问题,觉得还不够有趣,他在平面上又设立了n个禁止点,表示马不能走到这些点上,现在他们想知道,这种情况下马有多少种不同的移动方法呢?
答案数可能很大,你只要告诉他们答案模(10^9+7)的值就行。
保证Ax*By-Ay*Bx≠0
|Ax|,|Ay|,|Bx|,|By| <= 500, 0 <= n,Ex,Ey <= 500
题解
题面保证了给出的两个向量叉积为 0,就是说它们不平行。不平行的两个向量可以作为一组基底,根据向量唯一分解定理,原先平面上的所有点就获得了一个新坐标。于是问题变成了:从 (0,0) 走到 (n,m) ,中间不能经过指定的 k 个点,求方案数。
这是个简单的容斥 DP。记 \(f[i]\) 表示从 (0,0) 走到 i 号障碍点,中间不经过其它障碍点的方案数,这可以用组合数减去不合法的方案数求得。而不合法的方案必然存在第一个遇到的非 i 号障碍点 j ,于是有
\]
其中 \(g(i,j)\) 表示第 \(i\) 个点到第 \(j\) 个点的方案数。这个就是个简单的组合数。
看到这个 DP 式,我觉得很熟悉。我大概做过这道题的简化版。
#include<bits/stdc++.h>
using namespace std;
template<class T> T read(){
T x=0,w=1;char c=getchar();
for(;!isdigit(c);c=getchar())if(c=='-') w=-w;
for(;isdigit(c);c=getchar()) x=x*10+c-'0';
return x*w;
}
template<class T> T read(T&x){
return x=read<T>();
}
#define co const
#define il inline
typedef long long LL;
co int mod=1000000000+7;
il int add(int a,int b){
return (a+=b)>=mod?a-mod:a;
}
il int mul(int a,int b){
return (LL)a*b%mod;
}
il int fpow(int a,int b){
int ans=1;
for(;b;b>>=1,a=mul(a,a))
if(b&1) ans=mul(ans,a);
return ans;
}
struct Vector {int x,y;};
il int cross(co Vector&a,co Vector&b){
return a.x*b.y-a.y*b.x;
}
bool trans(co Vector&A,co Vector&B,Vector&E){
int s=cross(A,B),x=cross(E,B);
if(x%s) return 0;
int t=cross(B,A),y=cross(E,A);
if(y%t) return 0;
E=(Vector){x/s,y/t};
return 1;
}
il bool operator<(co Vector&a,co Vector&b){
return a.x!=b.x?a.x<b.x:a.y<b.y;
}
co int N=501,M=1000000+1;
Vector E,A,B,S[N];
int fac[M],ifac[M],f[N];
il int binom(int n,int m){
return mul(fac[n],mul(ifac[m],ifac[n-m]));
}
int main(){
read(E.x),read(E.y);
int n=read<int>();
read(A.x),read(A.y),read(B.x),read(B.y);
if(!trans(A,B,E)) {puts("0");return 0;}
for(int i=1;i<=n;++i){
read(S[i].x),read(S[i].y);
if(!trans(A,B,S[i])||S[i].x<0||S[i].x>E.x||S[i].y<0||S[i].y>E.y) --n,--i;
}
sort(S+1,S+n+1),S[++n]=E;
// for(int i=1;i<=n;++i) cerr<<i<<" = "<<S[i].x<<" "<<S[i].y<<endl;
fac[0]=1;
for(int i=1;i<M;++i) fac[i]=mul(fac[i-1],i);
ifac[M-1]=fpow(fac[M-1],mod-2);
for(int i=M-2;i>=0;--i) ifac[i]=mul(ifac[i+1],i+1);
for(int i=1;i<=n;++i){
f[i]=binom(S[i].x+S[i].y,S[i].x);
for(int j=1;j<i;++j){
int dx=S[i].x-S[j].x,dy=S[i].y-S[j].y;
if(dx<0||dy<0) continue;
f[i]=add(f[i],mod-mul(f[j],binom(dx+dy,dx)));
}
}
printf("%d\n",f[n]);
return 0;
}
说一下如何向量分解。假设两个基向量是 \(i,j\),新向量分解的结果是 \(xi+yj\),考虑以下式子
\frac{(xi+yj)\times j}{i\times j}=x
\]
这样就不用解方程了,虽然效果都一样。
test20190830 NOIP 模拟赛的更多相关文章
- NOIP模拟赛20161022
NOIP模拟赛2016-10-22 题目名 东风谷早苗 西行寺幽幽子 琪露诺 上白泽慧音 源文件 robot.cpp/c/pas spring.cpp/c/pas iceroad.cpp/c/pas ...
- contesthunter暑假NOIP模拟赛第一场题解
contesthunter暑假NOIP模拟赛#1题解: 第一题:杯具大派送 水题.枚举A,B的公约数即可. #include <algorithm> #include <cmath& ...
- NOIP模拟赛 by hzwer
2015年10月04日NOIP模拟赛 by hzwer (这是小奇=> 小奇挖矿2(mining) [题目背景] 小奇飞船的钻头开启了无限耐久+精准采集模式!这次它要将原矿运到泛光之源的矿 ...
- 大家AK杯 灰天飞雁NOIP模拟赛题解/数据/标程
数据 http://files.cnblogs.com/htfy/data.zip 简要题解 桌球碰撞 纯模拟,注意一开始就在袋口和v=0的情况.v和坐标可以是小数.为保险起见最好用extended/ ...
- 队爷的讲学计划 CH Round #59 - OrzCC杯NOIP模拟赛day1
题目:http://ch.ezoj.tk/contest/CH%20Round%20%2359%20-%20OrzCC杯NOIP模拟赛day1/队爷的讲学计划 题解:刚开始理解题意理解了好半天,然后发 ...
- 队爷的Au Plan CH Round #59 - OrzCC杯NOIP模拟赛day1
题目:http://ch.ezoj.tk/contest/CH%20Round%20%2359%20-%20OrzCC杯NOIP模拟赛day1/队爷的Au%20Plan 题解:看了题之后觉得肯定是DP ...
- 队爷的新书 CH Round #59 - OrzCC杯NOIP模拟赛day1
题目:http://ch.ezoj.tk/contest/CH%20Round%20%2359%20-%20OrzCC杯NOIP模拟赛day1/队爷的新书 题解:看到这题就想到了 poetize 的封 ...
- CH Round #58 - OrzCC杯noip模拟赛day2
A:颜色问题 题目:http://ch.ezoj.tk/contest/CH%20Round%20%2358%20-%20OrzCC杯noip模拟赛day2/颜色问题 题解:算一下每个仆人到它的目的地 ...
- CH Round #52 - Thinking Bear #1 (NOIP模拟赛)
A.拆地毯 题目:http://www.contesthunter.org/contest/CH%20Round%20%2352%20-%20Thinking%20Bear%20%231%20(NOI ...
随机推荐
- maven设置阿里云仓库
到maven安装目录的conf下setting.xml文件 找到mirrors标签中添加 <mirror> <id>nexus-aliyun</id> <mi ...
- FineUI window弹层设置
需在页面先设置 <f:Window ID="Window1" runat="server" Height="600px" Widt ...
- [转帖]keepalived工作原理和配置、使用
keepalived工作原理和配置.使用 https://www.iteye.com/blog/aoyouzi-2288124 keepalived是什么 keepalived是集群管理中保证集群高可 ...
- Django框架4——form表单
HTML表单一直是交互性网站的支柱,使用form组件对用户通过表单提交的数据进行访问.有效性检查以及其他处理 从Request对象中获取数据 URL相关信息 属性/方法 说明 举例 request.p ...
- 函数的练习1——python编程从入门到实践
8-1 消息: 编写一个名为display_message()的函数,它打印一个句子指出你在本章学的是什么.调用这个函数,确认显示的消息正确无误. def display_message(): pri ...
- 16.Python略有小成(内置模块Ⅱ)
Python(内置模块Ⅱ) 一.规范化开发 一个py文件中 文件加载问题 代码可读性,查询差 要将一个py文件分开,合理的分成多个py文件 配置文件放在一起 : 一些轻易不改变的变量,一般命名 set ...
- 最详细的Android SDK下载安装及配置教程
文章转载与:https://blog.csdn.net/dr_neo/article/details/49870587 最近Neo突发神经,想要将学过的一些计算机视觉.机器学习中的算法都放到移动设备上 ...
- JAVA线程Disruptor核心链路应用(八)
import java.util.concurrent.atomic.AtomicInteger; /** * Disruptor中的 Event * @author Alienware * */ p ...
- c#的一些书写技巧
从非创建线程访问线程资源 Invoke(new Action<int>((o)=> { textBox1.Text = (Convert.ToInt32(textBox1.Text ...
- 北航OO课程完结总结
什么是OO? 面向对象,是一种编程的思想方法,但是在这门课程中,我们实际学习到的是将理论运用到具体实践上,将自己的想法付诸实践,不断去探索和优化的这一体验. 后两次作业架构总结 本单元两次作业,我们面 ...