[vijos1892]树上的最大匹配(树形DP)
分析:(100分其实用到各种c++优化,没什么实际意义,所以弄70就可以了)
题目很简单,很容易想出用树形DP,但是求方案数的时候,满满都是细节……,本渣考试时候就跪了……只能膜拜神犇代码……
#include <cstdio>
#include <cstring>
//#include <algorithm> using namespace std; typedef long long LL; const int MaxN = ; struct Node{
int v;
Node *nxt;
}pool[MaxN << ],*tail=pool,*g[MaxN]; int n;
LL m;
LL h[MaxN][];
int fa[MaxN],f[MaxN][];
LL pre[MaxN],suf[MaxN]; inline void make_edge(int u,int v){
tail->v=v;tail->nxt=g[u];g[u]=tail++;
tail->v=u;tail->nxt=g[v];g[v]=tail++;
} inline int max(int a,int b){return a>b ? a : b;}
void dp(){
static int q[MaxN],l,r;
memset(fa,0xff,sizeof(fa));
for(fa[q[l=r=]=]=;l<=r;l++)
for(Node *p=g[q[l]];p;p=p->nxt) if(!~fa[p->v])
fa[q[++r]=p->v]=q[l];
for(int i=r;i>=;i--){
int u=q[i];
int maxt=0xc0c0c0c0;
int cnt=,j;
f[u][]=,f[u][]=;
h[u][]=,h[u][]=;
for(Node *p=g[u];p;p=p->nxt) if(p->v!=fa[u]){
LL dt=;
f[u][]+=max(f[p->v][],f[p->v][]);
f[u][]+=max(f[p->v][],f[p->v][]);
maxt=max(maxt,f[p->v][]+-max(f[p->v][],f[p->v][])); if(f[p->v][]>f[p->v][]) dt=h[p->v][];
else if(f[p->v][]<f[p->v][]) dt=h[p->v][];
else dt=(h[p->v][]+h[p->v][])%m;
pre[++cnt]=dt;suf[cnt]=dt;
(h[u][]*=dt)%=m;
}
pre[]=suf[cnt+]=;
for(int i=;i<=cnt;i++) (pre[i]*=pre[i-])%=m;
for(int i=cnt;i;i--) (suf[i]*=suf[i+])%=m;
f[u][]+=maxt;
j=;
for(Node *p=g[u];p;p=p->nxt) if(p->v!=fa[u]){
if(f[p->v][]+-max(f[p->v][],f[p->v][])==maxt)
(h[u][]+=pre[j-]*suf[j+]%m*h[p->v][]%m)%=m;
j++;
}
}
if(f[][]==f[][]) printf("%d\n%lld\n",f[][],(h[][]+h[][])%m);
else if(f[][]>f[][]) printf("%d\n%lld\n",f[][],h[][]);
else printf("%d\n%lld\n",f[][],h[][]);
}
int main()
{
scanf("%d",&n);
for(int i=;i<n;i++){
int u,v;scanf("%d%d",&u,&v);
make_edge(u,v);
}
scanf("%lld",&m);
dp();
return ;
}
细节反思:
1、求f和求g的过程可以一块写,思路比较清晰一点
2、求g[u][1]的时候的技巧:
本渣只能想到先求所有的乘积,然后再枚举每一个位置的,除掉,因为取模只能求逆
但此神犇的做法很厉害:
先在求f的过程中把u的每个子节点的最优值记下来保存在数组中,并记下来u往叶子节点连边能得到的最大增值maxt
然后把记最优值的数组从前往后累乘得到pre,从后往前乘得到suf
然后对于每次枚举的连边的子节点i,首先判断连i所能得到的增值是否为maxt,如果是那么增加的方案数也就确定了:pre[i-1]*suf[i+1]*g[i][0]
细节方面真的很重要……
[vijos1892]树上的最大匹配(树形DP)的更多相关文章
- Vijos p1892 树上的最大匹配 树形DP+计数 被卡常我有特殊技巧heheda
https://vijos.org/p/1892 此题需要手动开栈: <<; //256MB char *p=(char*)malloc(size)+size; __asm__(" ...
- BZOJ_4033_[HAOI2015]树上染色_树形DP
BZOJ_4033_[HAOI2015]树上染色_树形DP Description 有一棵点数为N的树,树边有边权.给你一个在0~N之内的正整数K,你要在这棵树中选择K个点,将其染成黑色,并 将其他的 ...
- 2021.07.17 P3177 树上染色(树形DP)
2021.07.17 P3177 树上染色(树形DP) [P3177 HAOI2015]树上染色 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 重点: 1.dp思想是需要什么,维护 ...
- 树上的等差数列 [树形dp]
树上的等差数列 题目描述 给定一棵包含 \(N\) 个节点的无根树,节点编号 \(1\to N\) .其中每个节点都具有一个权值,第 \(i\) 个节点的权值是 \(A_i\) . 小 \(Hi\) ...
- 【BZOJ4033】[HAOI2015] 树上染色(树形DP)
点此看题面 大致题意: 给你一棵点数为N的带权树,要你在这棵树中选择K个点染成黑色,并将其他的N-K个点染成白色.要求你求出黑点两两之间的距离加上白点两两之间距离的和的最大值. 树形\(DP\) 这道 ...
- 洛谷P3177 [HAOI2015]树上染色(树形dp)
题目描述 有一棵点数为 N 的树,树边有边权.给你一个在 0~ N 之内的正整数 K ,你要在这棵树中选择 K个点,将其染成黑色,并将其他 的N-K个点染成白色 . 将所有点染色后,你会获得黑点两两之 ...
- BZOJ4033: [HAOI2015]树上染色(树形DP)
4033: [HAOI2015]树上染色 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 3461 Solved: 1473[Submit][Stat ...
- bzoj 4033: [HAOI2015]树上染色【树形dp】
准确的说应该叫树上分组背包?并不知道我写的这个叫啥 设计状态f[u][j]为在以点u为根的子树中有j个黑点,转移的时候另开一个数组,不能在原数组更新(因为会用到没更新时候的状态),方程式为g[j+k] ...
- BZOJ 4033[HAOI2015] 树上染色(树形DP)
4033: [HAOI2015]树上染色 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 3188 Solved: 1366[Submit][Stat ...
随机推荐
- 用SecureCRT在windows和CentOS间上传下载文件
安装lrzsz: # yum -y install lrzsz 现在就可以正常使用rz.sz命令上传.下载数据了 配置SecureCRT的session选项的SFTP标签页和X/Y/Zmodem中的目 ...
- C++STL之迭代器
迭代器 迭代器提供对一个容器中的对象的访问方法,并且定义了容器中对象的范围.迭代器就如同一个指针.事实上,C++的指针也是一种迭代器.但是,迭代器不仅仅是指针,因此你不能认为他们一定具有地址值.例如, ...
- SQL SERVER 2012 使用订阅发布同步数据库
软件做大了,客户就多了,一个数据库服务器是远远不够的,当有一台数据服务器卦掉,那整个系统就会崩溃,所以必须考虑到数据库的自动同步与备份,当一台数据库服务 器宕机,自然就有用一台数据服务器启动起来保证整 ...
- struts2 基本用法
Struts2必需库: commons-fileupload.jar.commons-io-1.3.2.jar.freemarker-2.3.16.jar.javassist-3.7.ga.jar.o ...
- 【ASP.NET 进阶】定时执行任务
原理:利用全局应用程序类 Global.asax 和 System.Timers.Timer 类定时处理任务. 示例效果图: 其 Global.asax 类代码如下: using System; u ...
- 不会UML的程序员不是好构架师?
情况描述 我已经工作两年半, 参加过一个网页游戏项目和一个IOS应用项目, 自以为参与度非常高, 也经常涉及到底层引擎和主逻辑业务. 目前想更快的向构架师方向发展. 最近在看\<Learning ...
- javascript判断回文数
"回文"是指正读反读都能读通的句子,它是古今中外都有的一种修辞方式和文字游戏,如"我为人人,人人为我"等.在数学中也有这样一类数字有这样的特征,成为回文数(pa ...
- jquery中attr和prop的区别(转)
在网络上看到这样一篇关于jquery中attr和prop的区别文章,觉得不错,所以转载了. 在jQuery 1.6中,.attr()方法查询那些没有设置的属性,则会返回一个undefined.如果你要 ...
- python中str()和repr()的区别
>>> s = 'Hello, world.' >>> str(s) 'Hello, world.' >>> repr(s) "'Hel ...
- c查漏补缺
restrict 要理解什么是restrict,首先要知道Pointer aliasing:指两个或以上的指针指向同一数据,例如: ; int *a = &i; int *b = &i ...