题目 树的最长路径【树形DP】
https://www.acwing.com/problem/content/description/1074/
解析
https://www.acwing.com/file_system/file/content/whole/index/content/2826773/
- #include
- using namespace std;
- const int N=1e4+10,M=2*N;
- int e[M],ne[M],h[N],idx,w[M];
- int f1[N],f2[N]; //用来记录各点路径最大值和次大值
- int res;
- void add(int a,int b,int c)//邻接表的方式存树
- {
- e[idx]=b,w[idx]=c,ne[idx]=h[a],h[a]=idx++;
- }
- void dfs(int u,int f)//该节点和该节点父节点
- {
- f1[u]=0,f2[u]=0; //用来记录最大值和次大值
- for(int i=h[u];~i;i=ne[i])//枚举一遍子树
- {
- int j=e[i];
- if(j==f)continue;//防止循环遍历
- dfs(j,u);//该子树到u
- int m=f1[j]+w[i];//该子树到u的距离
- if(m>=f1[u])f2[u]=f1[u],f1[u]=m;//假如大于最大值,更新一遍最大值和次大值
- else if(m>f2[u]) f2[u]=m;//假如大于次大值,则更新次大值
- }
- res=max(res,f1[u]+f2[u]);//答案求一遍经过该点的最长路径
- }
- int main()
- {
- int n;
- cin>>n;
- memset(h,-1,sizeof h);
- for(int i=1;i
- {
- int a,b,c;
- cin>>a>>b>>c;
- add(a,b,c),add(b,a,c);
- }
- dfs(1,-1);
- cout<
- }
树的中心
题目 树的中心
https://www.acwing.com/problem/content/description/1075/
题目 树的最长路径【树形DP】
https://www.acwing.com/problem/content/description/1074/
- #include
- using namespace std;
- const int N=1e4+10,M=2*N,INF=0x3f3f3f3f;
- int e[M],ne[M],h[N],idx,w[M];
- int d1[N],d2[N],p1[N],p2[N],up[N];
-
- void add(int a,int b,int c)
- {
- e[idx]=b,w[idx]=c,ne[idx]=h[a],h[a]=idx++;
- }
- int dfs_down(int u,int f)//返回u的最长向下路径
- {
- d1[u]=-INF,d2[u]=-INF;
- for(int i=h[u];~i;i=ne[i])
- {
- int j=e[i];
- if(j==f)continue; //避免向上查找,进入死循环
- dfs_down(j,u);
- int m=d1[j]+w[i];
- if(m>=d1[u])//更新一下最长和第二长的路径,并记录下从该路径是从哪一个点下去的
- {
- d2[u]=d1[u],p2[u]=p1[u];
- d1[u]=m,p1[u]=j;
- }
- else if(m>d2[u]) d2[u]=m,p2[u]=j;
- }
- if(d1[u]==-INF) d1[u]=d2[u]=0; //如果没有改变过该点的距离,就证明这个点是叶节点
- return d1[u];
- }
- void dfs_up(int u,int f) //用父节点更新一下子节点向上的最长路径
- {
- for(int i=h[u];~i;i=ne[i])
- {
- int j=e[i];
- if(j==f)continue;
- if(p1[u]==j) up[j]=max(up[u],d2[u])+w[i]; //如果从父节点向下的最长路径进过了要更新的子节点,那么就用第二长的路径更新
- else up[j]=max(up[u],d1[u])+w[i]; //假如不经过自己,则用向上走的最大值更新自己
- dfs_up(j,u);
- }
- }
- int main()
- {
- int n;
- cin>>n;
- memset(h,-1,sizeof h);
- for(int i=1;i
- {
- int a,b,c;
- cin>>a>>b>>c;
- add(a,b,c),add(b,a,c);
- }
- dfs_down(1,-1);//向下走
- dfs_up(1,-1);//向上走
- int ans=INF;
- for(int i=1;i<=n;i++)
- ans=min(ans,max(up[i],d1[i]));
- cout<
- }
数字转换【树形DP】
题目 数字转换【树形DP】
http://ybt.ssoier.cn:8088/problem_show.php?pid=1577
- #include
- using namespace std;
- const int N=5e4+10;
- int e[N],ne[N],h[N],idx,w[N];
- int sum[N];
- bool st[N];//用来标记该点的有父节点
- int ans;
- void add(int a,int b)
- {
- e[idx]=b,ne[idx]=h[a],h[a]=idx++;
- }
- int dfs(int u)
- {
- int d1=0,d2=0;
- for(int i=h[u];~i;i=ne[i])
- {
- int j=e[i];
- int m=dfs(j)+1;
- if(m>=d1)d2=d1,d1=m;
- else if(m>d2)d2=m;
- }
- ans=max(ans,d1+d2);
- return d1;//返回u为根的最大距离
- }
- int main()
- {
- int n;
- cin>>n;
- memset(h,-1,sizeof h);
- for(int i=1;i<=n;i++)//求一个数的约数和
- for(int j=2;j<=n/i;j++)//用类似质数筛的方法
- {
- sum[i*j]+=i;//该数加上这个约数
- }
- for(int i=2;i<=n;i++)//因为1没有在约了,所以从2开始
- {
- if(sum[i]
- {
- add(sum[i],i);//把i的父节点令为sum[i]
- st[i]=true;//标记这个点有父节点
- }
- }
- for(int i=1;i<=n;i++)
- {
- if(!st[i])
- dfs(i);
- }
- cout<
- }
二叉苹果树【有依赖背包DP】
题目
http://ybt.ssoier.cn:8088/problem_show.php?pid=1575
- #include
- using namespace std;
- const int N=110,M=2*N;
- int n,m;
- int h[N],e[M],idx,ne[M],w[M];
- int f[N][M];
- void add(int a,int b,int c)
- {
- e[idx]=b,w[idx]=c,ne[idx]=h[a],h[a]=idx++;
- }
- void dfs(int u,int father)
- {
- for(int i=h[u];~i;i=ne[i])//枚举物品组
- {
- if(e[i]==father) continue;
- dfs(e[i],u);
- for(int j=m;j>=0;j--)//体积
- for(int k=0;k
//枚举决策 - f[u][j]=max(f[u][j],f[u][j-k-1]+f[e[i]][k]+w[i]);
- }
- }
- int main()
- {
- cin>>n>>m;
- memset(h,-1,sizeof h);
- for(int i=1;i
- {
- int a,b,c;
- cin>>a>>b>>c;
- add(a,b,c),add(b,a,c);
- }
- dfs(1,-1);
- cout<
1][m]<//输出第一个根的m个树枝 - return 0;
- }
战略游戏【树形DP+状态机模型】
题目 战略游戏
https://www.acwing.com/problem/content/description/325/
解析
https://www.acwing.com/solution/content/66365/

- #include
- using namespace std;
- const int N=1510,M=2*N;
- int n;
- int h[N],e[M],idx,ne[M];
- bool st[N];
- int f[N][M];
- void add(int a,int b)
- {
- e[idx]=b,ne[idx]=h[a],h[a]=idx++;
- }
- void dfs(int u)
- {
- f[u][0]=0;//表示不放哨兵
- f[u][1]=1;//表示放一个哨兵
- for(int i=h[u];~i;i=ne[i])//枚举子树
- {
- int j=e[i];
- dfs(j);
- f[u][0]+=f[j][1];//当前没哨兵加上子树一定得有哨兵
- f[u][1]+=min(f[j][0],f[j][1]);//当前有哨兵,子树可有可无
- }
- }
- int main()
- {
- while(scanf("%d",&n)!=EOF)
- {
- memset(h,-1,sizeof h);//清空上一次的状态
- idx=0;//清空上一次的状态
- memset(st,0,sizeof st);//清空上一次的状态
- for(int i=0;i
- {
- int a,cnt;
- scanf("%d:(%d)",&a,&cnt);
- while(cnt--)
- {
- int b;
- scanf("%d",&b);
- add(a,b);
- st[b]=true;
- }
- }
- int root=0;
- while(st[root]) root++;//找根节点
- dfs(root);
- printf("%d\n",min(f[root][0],f[root][1]));//输出放或者不放的最小值
- }
- return 0;
- }
皇宫看守【树形DP+状态机模型】
题目 皇宫看守
http://ybt.ssoier.cn:8088/problem_show.php?pid=1579
解析
https://www.acwing.com/solution/content/66594/
- #include
- using namespace std;
- const int N=1510,M=2*N;
- int n;
- int h[N],e[M],idx,ne[M],w[M];
- bool st[N];
- int f[N][M];
- void add(int a,int b)
- {
- e[idx]=b,ne[idx]=h[a],h[a]=idx++;
- }
- void dfs(int u)
- {
- f[u][2]=w[u];//表示放一个哨兵,所花费的费用
- for(int i=h[u];~i;i=ne[i])//枚举子树
- {
- int j=e[i];
- dfs(j);
- f[u][0]+=min(f[j][1],f[j][2]);//当前被父节点看到,则子节点只能自己放或者被子节点的子节点看到,两边取最小
- f[u][2]+=min(f[j][0],min(f[j][1],f[j][2]));//当前有哨兵,则子树三种情况都有可能,取最小
- }
- f[u][1]=0x3f3f3f3f;//因为要求最小值,所以初始化为正无穷
- for(int i=h[u];~i;i=ne[i])//枚举子树
- {
- int k=e[i];
- //这里f[u][0]就是以u根的子树所有f[j][1]与f[j][2]的最小值的和,因为前面更新状态时已经求过
- //所以下面就得减去除第k的f[k][1],f[k][2]的最小值的和,剩下就是状态计算的所求
- f[u][1]=min(f[u][1],f[k][2]+f[u][0]-min(f[k][1],f[k][2]));//表示第k个子树有哨兵,可以看到父节点,其他的只能是要么被儿子看到要么自己也放
- }
- }
- int main()
- {
- cin>>n;
- memset(h,-1,sizeof h);//清空上一次的状态
- for(int i=1;i<=n;i++)
- {
- int a,c,cnt;
- cin>>a>>c>>cnt;
- w[a]=c;
- while(cnt--)
- {
- int b;
- cin>>b;
- add(a,b);
- st[b]=true;
- }
- }
- int root=1;
- while(st[root]) root++;//找根节点
- dfs(root);
- printf("%d\n",min(f[root][1],f[root][2]));//输出被子节点看到或者放的最小值
- return 0;
- }
-
相关阅读:
计算机毕业设计(附源码)python智慧小区团购系统
腾讯云轻量2核4G5M可容纳多少人访问?
git的基本使用(一)
SpringSecurity系列 - 10 传统Web项目表单认证: UsernamePasswordAuthenticationFilter 过滤器
java毕业设计某市教育局综合信息管理平台mybatis+源码+调试部署+系统+数据库+lw
MetaAI的融合怪:BlenderBot
js基础笔记学习52-练习2计算水仙花数1
测试开发——项目
使用C语言进行问题分析
Revit中创建基于线的砌体墙及【快速砌体排砖】
-
原文地址:https://blog.csdn.net/m0_64378422/article/details/127843813