P4827「国家集训队」 Crash 的文明世界
「国家集训队」 Crash 的文明世界
提供一种不需要脑子的方法。
其实是看洛谷讨论版看出来的(
(但是全网也就这一篇这个方法的题解了)
首先这是一个关于树上路径的问题,我们可以无脑上点分治。
考虑当以 \(root\) 为根时,如何计算经过 \(root\) 的路径对某一个点的贡献。
若现在我们要找经过 \(root\) 的路径中长度为 \(d\) 且路径的一端为 \(u\)。
则这一部分的贡献为 \(v_{d}cnt_{d-h_u}\),其中 \(v_d=d^k\),\(h_u\) 表示点 \(u\) 的深度,\(cnt_i\) 表示深度为 \(i\) 的节点个数。
当然这里会有一种不合法的情况,就是找到的路径两端点在 \(root\) 的同一棵子树中。这可以用点分治惯用的容斥解决。
以 \(root\) 为根时,路径对点 \(u\) 的贡献为(事实上对深度为 \(h_u\) 的节点贡献是相同的)
\]
为了处理起来更加方便,我们增加一些无用的部分
于是有
\]
令 \(n=2\times maxdeep\)。
\]
按照套路,将 \(cnt\) 数组翻转一下
\]
令
\]
这是一个卷积的形式,直接 \(\texttt{FFT/NTT}\) 即可。
所以总时间复杂度为 \(O(n\log_2^2n)\)。
(所以为啥不把这题的 k 开到和 n 同级呢)
下面讲讲常数优化:
- 预处理原根、单位根必不可少。
- 能不取模尽量别取模。
- 由于这也是在分治的过程中进行 \(\texttt{FFT}\) 的计算,所以当规模较小时暴力会更快。
另外值得注意的是,由于本题的模数不是一个 \(\texttt{NTT}\) 模数,而中间过程中的结果最大可能为 \(10006^2\) ,所以我们可能得选择恰当的 \(\texttt{NTT}\) 模数。
这样的话结果就一定不会有问题。
这个题就这样非常套路地被我们解决了。
贴一个很丑的代码:
#include<bits/stdc++.h>
using namespace std;
const int maxn=2e5+5;
const int p=1e4+7;
const int P=1004535809;
int n,k;
struct edge{
int to,nex;
}e[maxn<<1];
int head[maxn],tot;
int siz[maxn],dp[maxn],vis[maxn],rt;
int w[maxn],cnt[maxn],ans[maxn];
int f[maxn],g[maxn],rev[maxn],len=1;
void add(int a,int b){
e[++tot]=(edge){b,head[a]};
head[a]=tot;
}
int ksm(int a,int b,int p){
int ans=1;
while(b){
if(b&1) ans=1ll*ans*a%p;
b>>=1,a=1ll*a*a%p;
}
return ans;
}
vector<int> W[20];
void INIT(){
for(int i=1,num=0;num<=17;++num,i<<=1){
int w=ksm(3,(P-1)/(i<<1),P),tmp=1;
for(int k=0;k<i;++k)
W[num].emplace_back(tmp),tmp=1ll*tmp*w%P;
}
}
void NTT(int *f){
for(int i=0;i<len;++i)
if(i<rev[i]) swap(f[i],f[rev[i]]);
for(int i=1,num=0;i<len;i<<=1,++num){
for(int j=0;j<len;j+=(i<<1)){
for(int k=0;k<i;++k){
int x=f[j|k],y=1ll*W[num][k]*f[i|j|k]%P;
f[j|k]=x+y>P?x+y-P:x+y;
f[i|j|k]=x-y<0?x-y+P:x-y;
}
}
}
}
void init(int x){
len=1;
while(len<=x) len<<=1;
f[0]=g[0]=0;
for(int i=1;i<len;++i)
rev[i]=rev[i>>1]>>1|((i&1)?len>>1:0);
memset(f,0,sizeof (int)*len);
memset(g,0,sizeof (int)*len);
}
void getroot(int u,int f,int sum){
siz[u]=1,dp[u]=0;
for(int i=head[u];i;i=e[i].nex){
int v=e[i].to;
if(v==f||vis[v]) continue;
getroot(v,u,sum);
siz[u]+=siz[v];
dp[u]=max(siz[v],dp[u]);
}
dp[u]=max(dp[u],sum-siz[u]);
if(dp[u]<dp[rt]) rt=u;
}
void clear(int u,int f,int dis,int &mx){
mx=max(mx,dis);
for(int i=head[u];i;i=e[i].nex){
int v=e[i].to;
if(v==f||vis[v]) continue;
clear(v,u,dis+1,mx);
}
}
void getdis(int u,int f,int dis){
++cnt[dis];
if(cnt[dis]>=p) cnt[dis]-=p;
for(int i=head[u];i;i=e[i].nex){
int v=e[i].to;
if(v==f||vis[v]) continue;
getdis(v,u,dis+1);
}
}
int owo[251];
void mul(int *a,int *b,int n){
if(n<=100){
memset(owo,0,sizeof (int)*(2*n+1));
for(int i=0;i<=n;++i)
for(int j=0;j<=n;++j)
owo[i+j]=owo[i+j]+1ll*a[i]*b[j]%P>P?owo[i+j]+1ll*a[i]*b[j]%P-P:owo[i+j]+1ll*a[i]*b[j]%P;
for(int i=0;i<=2*n;++i) a[i]=owo[i];
return ;
}
memcpy(f,a,sizeof (int)*(n+1));
memcpy(g,b,sizeof (int)*(n+1));
NTT(f),NTT(g);
for(int i=0;i<len;++i) f[i]=1ll*f[i]*g[i]%P;
NTT(f);
reverse(f+1,f+len);
int inv=ksm(len,P-2,P);
for(int i=0;i<=2*n;++i) a[i]=1ll*f[i]*inv%P;
}
void dfs(int u,int f,int dis,int opt){
ans[u]+=opt*cnt[dis];
for(int i=head[u];i;i=e[i].nex){
int v=e[i].to;
if(v==f||vis[v]) continue;
dfs(v,u,dis+1,opt);
}
}
void calc(int u,int dis,int opt){
int n=0;
clear(u,0,dis,n);n*=2;
memset(cnt,0,sizeof (int)*(n+1));
getdis(u,0,dis);
reverse(cnt,cnt+n+1);
init(2*n);
mul(cnt,w,n);
for(int i=0;i<=n;++i) cnt[i]=cnt[i+n];
dfs(u,0,dis,opt);
}
void solve(int u){
vis[u]=1;
calc(u,0,1);
for(int i=head[u];i;i=e[i].nex){
int v=e[i].to;
if(vis[v]) continue;
calc(v,1,-1);
rt=0;
getroot(v,0,siz[v]);
solve(rt);
}
}
int main(){
// freopen("1.in","r",stdin);
// freopen("1.out","w",stdout);
ios::sync_with_stdio(0);
cin.tie(0),cout.tie(0);
cin>>n>>k;
INIT();
for(int i=1;i<n;++i){
int a,b;cin>>a>>b;
add(a,b),add(b,a);
}
for(int i=0;i<n;++i) w[i]=ksm(i,k,p);
dp[0]=(1<<30);
getroot(1,0,n);
solve(rt);
for(int i=1;i<=n;++i) cout<<ans[i]%p<<'\n';
}
P4827「国家集训队」 Crash 的文明世界的更多相关文章
- 「国家集训队」Crash的数字表格
题目描述 求(对 \(20101009\) 取模,\(n,m\le10^7\) ) \[\sum_{i=1}^n\sum_{j=1}^m\operatorname{lcm}(i,j)\] 大体思路 推 ...
- 「国家集训队」middle
「国家集训队」middle 传送门 按照中位数题的套路,二分答案 \(mid\),序列中 \(\ge mid\) 记为 \(1\),\(< mid\) 的记为 \(-1\) 然后只要存在一个区间 ...
- 「国家集训队」小Z的袜子
「国家集训队」小Z的袜子 传送门 莫队板子题. 注意计算答案的时候,由于分子分母都要除以2,所以可以直接约掉,这样在开桶算的时候也方便一些. 参考代码: #include <algorithm& ...
- 「洛谷1903」「BZOJ2120」「国家集训队」数颜色【带修莫队,树套树】
题目链接 [BZOJ传送门] [洛谷传送门] 题目大意 单点修改,区间查询有多少种数字. 解法1--树套树 可以直接暴力树套树,我比较懒,不想写. 稍微口胡一下,可以直接来一个树状数组套主席树,也就是 ...
- Solution -「国家集训队」「洛谷 P2619」Tree I
\(\mathcal{Description}\) Link. 给一个 \(n\) 个点 \(m\) 条边的带权无向图,边有权值和黑白颜色,求恰选出 \(K\) 条白边构成的最小生成树. ...
- Solution -「国家集训队」「洛谷 P2839」Middle
\(\mathcal{Description}\) Link. 给定序列 \(\{a_n\}\),\(q\) 组询问,给定 \(a<b<c<d\),求 \(l\le[a,b] ...
- Solution -「国家集训队」「洛谷 P4451」整数的 lqp 拆分
\(\mathcal{Description}\) Link. 求 \[\sum_{m>0\\a_{1..m}>0\\a_1+\cdots+a_m=n}\prod_{i=1}^mf ...
- [国家集训队] Crash 的文明世界(第二类斯特林数)
题目 [国家集训队] Crash 的文明世界 前置 斯特林数\(\Longrightarrow\)斯特林数及反演总结 做法 \[\begin{aligned} ans_x&=\sum\limi ...
- 【BZOJ2159】Crash的文明世界
[2011集训贾志鹏]Crash的文明世界 Description Crash小朋友最近迷上了一款游戏--文明5(Civilization V).在这个游戏中,玩家可以建立和发展自己的国家,通过外交和 ...
随机推荐
- Psexec和wmiexec的原理和区别
PSEXEC 针对远程建立连接的方式有两种,一种先建立IPC通道连接,然后直接使用,操作如下: net use \\192.168.0.1\ipc$ "password" /use ...
- Go基础结构与类型03---标准输入与输出
package main import ( "fmt" "strconv" ) //每次接收一个用户输入 func main031() { //定义a, b两个 ...
- Go基础结构与类型02---使用iota定义常量组
package main import "fmt" /*const ( USA = 0 China = 1 Russia = 2 Britain = 3 France = 4 )* ...
- 改造 Firefox 浏览器——GitHub 热点速览
作者:HelloGitHub-小鱼干 上周推荐了一个可以在浏览器上用 VS Code 的项目,这次 Firefox-UI-Fix 带你给 Firefox 来个大变身,在它现有 Proton UI 下进 ...
- Tensor Core技术解析(下)
Tensor Core技术解析(下) 让FP16适用于深度学习 Volta的深度学习能力是建立在利用半精度浮点(IEEE-754 FP16)而非单精度浮点(FP32)进行深度学习训练的基础之上. 该能 ...
- MEMS传感器作为变革的驱动力
MEMS sensors as drivers for change 物联网(IoT)正在改变与周围世界互动的方式.每个人,每件事,都是相互联系的,很快就会相互联系.微机电系统(MEMS)设备和传感器 ...
- 谈谈stream的运行原理
害,别误会,我这里说的stream不是流式编程,不是大数据处理框架.我这里说的是stream指的是jdk中的一个开发工具包stream. 该工具包在jdk8中出现,可以说已经是冷饭了,为何还要你说?只 ...
- 618技术特辑(三)直播带货王,“OMG买它”的背后,为什么是一连串技术挑战?
[本期推荐]为什么一到大促,我们的钱包总是被掏空?是大家自制力不够,还是电商平台太会读懂人心,从技术维度,抽丝剥茧一探究竟. 摘要:动辄几十上百万人同时在线的直播间,让所有人能同时公平的去抢购,并且还 ...
- NOIP模拟测试13「矩阵游戏·跳房子·优美序列」
矩阵游戏 考试时思路一度和正解一样,考试到最后还是打了80分思路,结果80分打炸了只得了40分暴力分 题解 算出来第一列的总值,每次通过加每两列之间的差值得出下一列的总值 算第一列我们只需要让当前点* ...
- ORA-00937: not a single-group group function
有时候查询会遇到如下错误 SCOTT@PROD> select deptno,sum(sal) from emp; select deptno,sum(sal) from emp ...