题目:http://www.joyoi.cn/problem/tyvj-1940

基环树的样子,看了书上的讲解,准备写树上DP,然后挂了:

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int const MAXN=1e6+,inf=1e9;
int n,a[MAXN],f[MAXN][],head[MAXN],ct,s[MAXN],h,col[MAXN],cr,rt,ans;
int sta[MAXN],top,reg[MAXN];
bool in[MAXN],fl,vis[MAXN];
struct N{
int to,next;
N(int t=,int n=):to(t),next(n) {}
}edge[MAXN];
void add(int x,int y)
{
edge[++ct]=N(y,head[x]);head[x]=ct;
edge[++ct]=N(x,head[y]);head[y]=ct;
}
void ser(int x)
{
col[x]=cr;
for(int i=head[x],u;i;i=edge[i].next)
if(!col[u=edge[i].to])ser(u);
}
void tj(int rt,int x)
{
if(fl)return;
sta[++top]=x;vis[x]=;
for(int i=head[x],u;i;i=edge[i].next)
{
if(edge[i].to==a[x])continue;
if(!vis[u=edge[i].to])tj(rt,u);
else
{
while(sta[top]!=rt)
{
int t=sta[top];
s[++h]=t;
in[t]=;top--;
}
in[rt]=;s[++h]=rt;top--;
fl=;return;
}
}
top--;
}
void dfs(int x)
{
int mn=inf,sum=;
for(int i=head[x],u;i;i=edge[i].next)
{
u=edge[i].to;
if(u==rt)continue;
if(u==a[x]||col[u]!=col[x])continue;
dfs(u);
f[x][]+=max(f[u][],f[u][]);
sum+=f[u][];
mn=min(mn,f[u][]-f[u][]);
}
if(sum==)f[x][]=;
else f[x][]=sum-mn;
}
void dfs2(int x)
{
int mn=inf,sum=;
for(int i=head[x],u;i;i=edge[i].next)
{
u=edge[i].to;
if(u==rt)continue;
if(u==a[x]||col[u]!=col[x])continue;
dfs(u);
f[x][]+=max(f[u][],f[u][]);
sum+=f[u][];
mn=min(mn,f[u][]-f[u][]);
}
if(sum==)f[x][]=;
else if(x==a[rt])f[x][]=sum;
else f[x][]=sum-mn;
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d",&a[i]);
reg[i]++;reg[a[i]]++;
add(i,a[i]);
}
for(int i=;i<=n;i++)
{
if(!col[i])h=,cr++,ser(i);
int rt;fl=;int s=;
for(int j=;j<=n;j++)
if(reg[j]>&&col[j]==cr)
{rt=j;break;}
tj(rt,i);
dfs(rt);
s=max(f[rt][],f[rt][]);
dfs2(rt);
s=max(s,f[rt][]);
ans+=s;
}
printf("%d",ans);
return ;
}

无输出的冗长树上DP

题目挺有意思,自己本来也想过贪心的做法,但不会处理链与环交接处的问题,想不清楚一条链会对环有什么影响;

然后看了看别人的博客,才发现链对环没有影响。。。因为环上的点不论链上怎样,仍还有环上别的点限制它;

所以链与环都是隔一个选一个,贪心。

代码如下:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
queue<int>q;
int const MAXN=1e6+;
int n,a[MAXN],reg[MAXN],ans;
bool vis[MAXN],v2[MAXN];
void bfs()
{
for(int i=;i<=n;i++)
if(!reg[i])q.push(i);
while(q.size())
{
int x=q.front();q.pop();
vis[x]=;
if(!vis[a[x]])
{
ans++;//因为多起点开始,不方便直接求链的长度,所以一个一个加
vis[a[x]]=;
reg[a[a[x]]]--;
if(!reg[a[a[x]]]&&!vis[a[a[x]]])
q.push(a[a[x]]);
}
}
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d",&a[i]);
reg[a[i]]++;
}
bfs();
for(int i=;i<=n;i++)
if(!vis[i]&&!v2[i])
{
int cnt=;v2[i]=;
int j=a[i];
while(!v2[j])cnt++,j=a[j];
ans+=cnt/;
}
printf("%d",ans);
return ;
}

最新文章

  1. 修改Windows/Ubuntu/Mint多系统的默认启动顺序
  2. sql group by
  3. object-c面向对象1
  4. shell 脚本文件Windows传到Linux后编码问题
  5. 解决VS2010控制台程序运行结束不显示请按任意键继续
  6. 向左对齐的Gallery
  7. [百度空间] [原]DLL导出实例化的模板类
  8. java下socket传图片
  9. zsh : command not found pip3 的解决方案
  10. linux基础命令touch
  11. 解决virtualbox与mac文件拖拽问题
  12. mysql 增加只读用户查询指定表
  13. Elasticsearch5.3.1 IK分词,同义词/联想搜索设置
  14. hibernate数据库操作基础
  15. 000 Jquery的Hello World程序
  16. C# 中的await
  17. python实现一个栏目的分页抓取列表页抓取
  18. SQL某时间段记录查询
  19. Python学习-19.Python的Http模块
  20. javac文件系统

热门文章

  1. Qnap 中VM下的win7
  2. vue 避免渲染时闪烁
  3. MySQL高可用之——keepalived+互为主从
  4. VueJS参数绑定:v-bind:href,v-on:event
  5. grep man 有删减 百科
  6. win7-64bit下基于VMware12.5安装rhel-server-6.3-i386
  7. python3短信接口使用
  8. 模式匹配之尺度空间---scale space
  9. 网络直播流媒体协议的选择讨论,RTSP,RTMP,HTTP,私有协议?
  10. LogStash 日志搜集