[贪心]JZOJ 3230 【佛山市选2013】树环转换
分析
早上写的时候想的是奇怪的DP
然后搞了一阵子发现可以贪心?
只有一个儿子的节点则保留不变,值上传
两个儿子的节点可以断开与父亲的联系,将子树中所有点连成一条链的代价和断开再连接的代价加入答案
三及以上个儿子的节点可以通过断开若干个儿子的关系转变成两个儿子的状态
这里默认树根是某条链的一端,所以如果树根属于后两种状态时,无需加上断开和连接的代价
最后不要忘了加连接链两端的代价1

#include <iostream> #include <cstdio> using namespace std; const int N=1e6+10; struct Graph { int v,nx; }g[2*N]; int cnt,list[N]; int n,ans; int f[N]; int stk[N][3],top; bool b[N]; void Add(int u,int v) { g[++cnt]=(Graph){v,list[u]};list[u]=cnt; g[++cnt]=(Graph){u,list[v]};list[v]=cnt; } void DFS() { stk[++top][0]=1;stk[top][1]=0;stk[top][2]=0; while (top) { int u=stk[top][0]; if (!list[u]) { if (stk[top][2]>=2) ans+=f[u]+(stk[top][1]!=0?2:0); b[u]=stk[top][2]<2; top--; u=stk[top][0]; if (b[g[list[u]].v]) { stk[top][2]++; f[u]+=f[g[list[u]].v]; if (stk[top][2]>2) f[u]+=2; } list[u]=g[list[u]].nx; } for (int i=list[u];i;i=list[u]=g[i].nx) if (g[i].v!=stk[top][1]) { stk[++top][0]=g[i].v,stk[top][1]=u,stk[top][2]=0; break; } } } int main() { scanf("%d",&n); for (int i=1,u,v;i<n;i++) scanf("%d%d",&u,&v),Add(u,v); DFS(); printf("%d",ans+1); }
在日渐沉没的世界里,我发现了你。
分类:
技巧-贪心
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 张高兴的大模型开发实战:(一)使用 Selenium 进行网页爬虫
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构