[BZOJ]2017省队十连测推广赛1
听学长说有比赛就随便打一打。
A.普通计算姬
题目大意:给出一棵带权树,支持一下两种操作:1.修改一个点的权值;2.给出l,r,询问以点l为根的子树和、点l+1为根的子树和、点l+2为根的子树和……点r为根的子树和的总和。(点数、操作数不超过10^5)
思路:感觉是三题中最难的。给出的[l,r]区间在树上没有实际意义,不好利用数据结构维护。考虑若不修改,可以一遍dfs算出每个点对应的dfs序,这样每棵子树都对应一个dfs序的区间,前缀和一下就能O(1)查子树和,再按点的编号顺序把子树和前缀和一下就能O(1)回答询问,若要支持修改,我们可以分块重构,每次修改我们先只记录下有这次修改而不实际修改前缀和数组,询问时先利用之前的前缀和数组计算答案,再统计之前每一次修改对这次询问的贡献;每进行K次修改,我们就O(n)重建一遍前缀和数组并删掉记录下的修改。统计修改对询问的贡献可以这么做:将每个子树对应成dfs序区间,统计询问区间内有多少个dfs序区间包含修改点,可以用主席树求出询问区间内有多少dfs序区间右端点大等于修改点,减去有多少dfs序区间左端点大于修改点就得到包含修改点的区间个数,就能计算贡献了。总复杂度O(N^2/K+KNlogN),适当调整K的大小复杂度约为O(N(NlogN)^0.5)。
#include<cstdio>
#define ll unsigned long long
inline int read()
{
int x;char c;
while((c=getchar())<''||c>'');
for(x=c-'';(c=getchar())>=''&&c<='';)x=(x<<)+(x<<)+c-'';
return x;
}
#define MN 100000
#define MK 50
#define ND 4000000
struct edge{int nx,t;}e[MN*+];
struct node{int l,r,s;}t[ND+];
int n,h[MN+],en,z[MN+],l[MN+],r[MN+],cnt,cx[MN+],cy[MN+],cn,tn,rt[MN+];
ll a[MN+],b[MN+];
inline void ins(int x,int y)
{
e[++en]=(edge){h[x],y};h[x]=en;
e[++en]=(edge){h[y],x};h[y]=en;
}
void dfs(int x,int fa)
{
l[x]=++cnt;
for(int i=h[x];i;i=e[i].nx)if(e[i].t!=fa)dfs(e[i].t,x);
r[x]=cnt;
}
void build()
{
for(int i=;i<=n;++i)a[l[i]]=z[i];
for(int i=;i<=n;++i)a[i]+=a[i-];
for(int i=;i<=n;++i)b[i]=b[i-]+a[r[i]]-a[l[i]-];
}
void ins(int pr,int pl,int x,int z)
{
for(int l=,r=n,mid,p=rt[pr]=++tn;;)
{
t[p].s=t[pl].s+z;
if(l==r)return;
int mid=l+r>>;
if(x<=mid)t[p].r=t[pl].r,p=t[p].l=++tn,pl=t[pl].l,r=mid;
else t[p].l=t[pl].l,p=t[p].r=++tn,pl=t[pl].r,l=mid+;
}
}
int query(int pl,int pr,int x)
{
int ans=;
for(int l=,r=n,mid;;)
{
if(l==x)return ans+t[pr].s-t[pl].s;
mid=l+r>>;
if(x<=mid)ans+=t[t[pr].r].s-t[t[pl].r].s,pl=t[pl].l,pr=t[pr].l,r=mid;
else pl=t[pl].r,pr=t[pr].r,l=mid+;
}
}
int main()
{
int m,i,t,x,y;ll ans;
n=read();m=read();
for(i=;i<=n;++i)z[i]=read();
for(i=;i<=n;++i)ins(read(),read());
dfs(e[h[]].t,);build();
for(i=;i<=n;++i)ins(i,rt[i-],l[i]-,-),ins(i,rt[i],r[i],);
while(m--)
{
t=read();x=read();y=read();
if(t==)
{
cx[++cn]=l[x];cy[cn]=y-z[x];z[x]=y;
if(cn==MK)cn=,build();
}
if(t==)
{
ans=b[y]-b[x-];
for(i=;i<=cn;++i)ans+=(ll)query(rt[x-],rt[y],cx[i])*cy[i];
printf("%llu\n",ans);
}
}
}
B.文艺计算姬
题目大意:求一边有n个点,另一边有m个点,共n*m条边的二分图共有多少种生成树,答案对p取模。(n,m,p<=10^18)
思路:先用矩阵树定理暴力计算一部分答案,观察容易发现答案为n^(m-1)*m^(n-1),由于模数较大乘法会爆long long,要用类似快速幂的快速乘加上快速幂,复杂度为O(logp^2)。答案公式我还没想到比较好的证明方法,想到了会在这里补上(这类题目一般打表才是比较好的做法)。
矩阵树定理暴力的代码好像被误删了,本来想贴出来的……
#include<cstdio>
#define ll long long
ll n,m,p;
inline ll mod(ll x){return x>=p?x-p:x;}
ll mul(ll a,ll b)
{
ll r=;
for(;b;b>>=,a=mod(a<<))if(b&)r=mod(r+a);
return r;
}
ll pow(ll a,ll b)
{
ll r=;
for(;b;b>>=,a=mul(a,a))if(b&)r=mul(r,a);
return r;
}
int main()
{
scanf("%lld%lld%lld",&n,&m,&p);
printf("%lld",mul(pow(n,m-),pow(m,n-)));
}
C.两双手
题目大意:一个棋子从(0,0)开始走,每次可以从(u,v)走到(u+Ax,u+Ay)或(u+Bx,u+By),有n个点不能走,问走到(Ex,Ey)有多少种方案。(0<=|Ax|,|Ay|,|Bx|,|By|,Ex,Ey,n<=500,Ax*By-Ay*Bx!=0)
思路:比较杂的数学应用吧。考虑容斥原理,走到(Ex,Ey)不经过禁止点的方案=总方案-至少经过一个禁止点的方案+至少经过两个-至少经过三个……可以用DP实现,为了让点有序方便DP,假设先把所有点旋转,使得向量(Ax,Ay)成为x轴,若(Bx,By)旋转后的纵坐标大于0,则一个点可以到另一个当且仅当该点旋转后纵坐标小于另一点旋转后纵坐标,反之同理,按照这个思路排序(实现上比较点积叉积等即可),我们就能保证只有排在前的能到达排在后的。经过奇数个点对答案贡献为负,偶数贡献为正,每次转移取负即可,令f[i]表示到第i个点的答案,则f[i]=Σ-f[j]*g[j][i] (j<i),其中g[j][i]为点j到点i的方案数,初始化(0,0)点的f值为-1即可。下面讨论计算g[j][i],设用了X条(Ax,Ay),Y条(Bx,By),暴力解方程x[j]+AxX+BxY=x[i],y[j]+AyX+ByY=y[i]就能算出X和Y,排列组合一下可以得到方案数,总算大功告成,总复杂度O(n^2)。
#include<cstdio>
#include<algorithm>
using namespace std;
#define MN 500
#define MX 500000
#define MOD 1000000007
int g[MN+][MN+],F[MX+],R[MX+],f[MN+],a,b,c,d,e;
int inv(int x)
{
int r=,y=MOD-;
for(;y;y>>=,x=1LL*x*x%MOD)if(y&)r=1LL*r*x%MOD;
return r;
}
struct node{int x,y;}p[MN+];
bool cmp(node x,node y)
{
int c1=a*x.y-b*x.x,c2=a*y.y-b*y.x;
return c1==c2?a*x.x+b*x.y<a*y.x+b*y.y:e>?c1<c2:c1>c2;
}
int main()
{
int n,i,j,k,A,B,x,y,tx,ty;
for(F[]=i=;i<=MX;++i)F[i]=1LL*F[i-]*i%MOD;
for(R[--i]=inv(F[MX]);i--;)R[i]=1LL*R[i+]*(i+)%MOD;
scanf("%d%d%d",&tx,&ty,&n);p[n+].x=tx;p[n+].y=ty;
scanf("%d%d%d%d",&a,&b,&c,&d);
for(i=;i<=n;++i)scanf("%d%d",&p[i].x,&p[i].y);
e=a*d-b*c;sort(p,p+n+,cmp);
for(i=;i<=n;++i)for(j=i;j++<=n;)
{
A=p[j].x-p[i].x;B=p[j].y-p[i].y;
if((d*A-c*B)%(a*d-c*b))continue;
x=(d*A-c*B)/(a*d-c*b);
if((b*A-a*B)%(c*b-a*d))continue;
y=(b*A-a*B)/(c*b-a*d);
if(x<||y<)continue;
g[i][j]=1LL*F[x+y]*R[x]%MOD*R[y]%MOD;
}
for(i=;i<=n+;++i)
{
if(!p[i].x&&!p[i].y)f[i]=-;
for(j=;j<i;++j)f[i]=(f[i]-1LL*f[j]*g[j][i])%MOD;
if(p[i].x==tx&&p[i].y==ty)return printf("%d",(f[i]+MOD)%MOD),;
}
}
[BZOJ]2017省队十连测推广赛1的更多相关文章
- [BZOJ]2017省队十连测推广赛1 T2.七彩树
题目大意:给你一棵n个点的树,每个点有颜色,m次询问,每次询问一个点x的子树内深度不超过depth[x]+d的节点的颜色数量,强制在线.(n,m<=100000,多组数据,保证n,m总和不超过5 ...
- [bzoj省选十连测推广赛2]T2七彩树
抄自:http://blog.csdn.net/coldef/article/details/61412577 当时看了就不会,看了别人的题解不懂怎么维护,最后抄了个代码....... 给定一棵n个点 ...
- bzoj省选十连测推广赛
A.普通计算姬 题意:给丁一棵树,每个点有一个权值,用sum(x)表示以x为根的子树的权值和,要求支持两种操作: 1 u v :修改点u的权值为v. 2 l r : 求∑sum[i] l&l ...
- bzoj 5216 [Lydsy2017省队十连测]公路建设 线段树维护 最小生成树
[Lydsy2017省队十连测]公路建设 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 93 Solved: 53[Submit][Status][ ...
- bzoj 5216: [Lydsy2017省队十连测]公路建设
5216: [Lydsy2017省队十连测]公路建设 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 66 Solved: 37[Submit][St ...
- Lydsy2017省队十连测
5215: [Lydsy2017省队十连测]商店购物 可能FFT学傻了,第一反应是前面300*300背包,后面FFT... 实际上前面背包,后面组合数即可.只是这是一道卡常题,需要注意常数.. //A ...
- bzoj 5218: [Lydsy2017省队十连测]友好城市
题意: 这题显然直接tarjan是做不了的. 这里安利另一个求SCC的算法Kosaraju,学习的话可以见这篇博客 于是结合莫队,我们有了个暴力. 发现主要瓶颈是dfs过程中找最小的未经过的点,我们用 ...
- 【BZOJ 5222】[Lydsy2017省队十连测]怪题
题目大意: 传送门 给一个长度为$n(n<=200)$的数列$h$,再给$m$个可以无限使用的操作,第$i$个操作为给长度为花费$c_i$的价值给长度为$l_i$的数列子序列+1或-1,求将数列 ...
- bzoj 5217: [Lydsy2017省队十连测]航海舰队
Description Byteasar 组建了一支舰队!他们现在正在海洋上航行着.海洋可以抽象成一张n×m 的网格图,其中有些位置是" .",表示这一格是海水,可以通过:有些位置 ...
随机推荐
- APP案例分析--扇贝单词
APP案例分析 一.调研 1.第一次上手 第一次使用时,一进APP,有一个每日一句,然后就是登录界面.有点不舒服,我都还不知道你这个APP好不好用,不让我体验一下就要注册.简单的测试了我的英语水平 ...
- Alpha冲刺Day3
Alpha冲刺Day3 一:站立式会议 今日安排: 我们把项目大体分为四个模块:数据管理员.企业人员.第三方机构.政府人员.数据管理员这一模块,数据管理员又可细分为两个模块:基础数据管理和风险信息管理 ...
- 第二次作业:APP案例分析
App案例分析 产品:三国杀-页游手游双通 选择理由 当今社会手机已经渐渐取代了电脑在人们日常生活的需求,既然要选择APP进行案例分析,首推的估计就是手机APP了.三国杀是陪伴我高中时代的主要娱乐方式 ...
- python的测试
测试 知识点 单元测试概念 使用 unittest 模块 测试用例的编写 异常测试 测试覆盖率概念 使用 coverage 模块 实验步骤 1. 应该测试什么? 如果可能的话,代码库中的所有代码都要测 ...
- BM V7000数据恢复成功案例;服务器数据恢复
IBM V7000存储是一款定位中端的存储设备,很多企业选择该服务器作为存储,最近北亚数据恢复中心接到一例V7000服务器数据恢复案例,下面将对本次数据恢复的过程和数据恢复方法进行归纳总结,希望对各位 ...
- java之多态详解
前言 什么叫多态?多态就是一种事物可以有多种表现形式 多态三要素 1.被动方必须有继承关系 2.子类一般都要重写父类方法 3.必须将主动方的功能函数的参数设置为 被动方父类的类型 举个例子司机开车 假 ...
- wyh的数列~(坑爹题目)
链接:https://www.nowcoder.com/acm/contest/93/K来源:牛客网 题目描述 wyh学长特别喜欢斐波那契数列,F(0)=0,F(1)=1,F(n)=F(n-1)+F( ...
- selenium的Python使用(一)浏览器驱动的安装及使用
一.selenium的安装 直接使用pip进行安装 pip install selenium #(安装最新版本) pip install selenium==3.6.0 #(安装指定版本) ...
- HttpClient 上传多个文件
using (System.Net.Http.HttpClient client = new System.Net.Http.HttpClient()) { client.BaseAddress = ...
- C# HttpClient设置cookies的两种办法 (转发)
一般有两种办法 第一种handler.UseCookies=true(默认为true),默认的会自己带上cookies,例如 var handler = new HttpClientHandler() ...