bzoj 1063: [Noi2008]道路设计 (树形dp)

1063: [Noi2008]道路设计

Time Limit: 20 Sec   Memory Limit: 162 MB
Submit: 990   Solved: 547
[ Submit][ Status][ Discuss]

Description

  Z国坐落于遥远而又神奇的东方半岛上,在小Z的统治时代公路成为这里主要的交通手段。Z国共有n座城市,一
些城市之间由双向的公路所连接。非常神奇的是Z国的每个城市所处的经度都不相同,并且最多只和一个位于它东
边的城市直接通过公路相连。Z国的首都是Z国政治经济文化旅游的中心,每天都有成千上万的人从Z国的其他城市
涌向首都。为了使Z国的交通更加便利顺畅,小Z决定在Z国的公路系统中确定若干条规划路线,将其中的公路全部
改建为铁路。我们定义每条规划路线为一个长度大于1的城市序列,每个城市在该序列中最多出现一次,序列中相
邻的城市之间由公路直接相连(待改建为铁路)。并且,每个城市最多只能出现在一条规划路线中,也就是说,任意
两条规划路线不能有公共部分。当然在一般情况下是不可能将所有的公路修建为铁路的,因此从有些城市出发去往
首都依然需要通过乘坐长途汽车,而长途汽车只往返于公路连接的相邻的城市之间,因此从某个城市出发可能需要
不断地换乘长途汽车和火车才能到达首都。我们定义一个城市的“不便利值”为从它出发到首都需要乘坐的长途汽
车的次数,而Z国的交通系统的“不便利值”为所有城市的不便利值的最大值,很明显首都的“不便利值”为0。小
Z想知道如何确定规划路线修建铁路使得Z国的交通系统的“不便利值”最小,以及有多少种不同的规划路线的选择
方案使得“不便利值”达到最小。当然方案总数可能非常大,小Z只关心这个天文数字modQ后的值。注意:规划路
线1-2-3和规划路线3-2-1是等价的,即将一条规划路线翻转依然认为是等价的。两个方案不同当且仅当其中一个方
案中存在一条规划路线不属于另一个方案。

Input

  第一行包含三个正整数N、M、Q,其中N表示城市个数,M表示公路总数,N个城市从1~N编号,其中编号为1的是首都
。Q表示上文提到的设计路线的方法总数的模数。接下来M行,每行两个不同的正数ai、bi(1≤ai,bi≤N)表示有一条
公路连接城市ai和城市bi。输入数据保证一条公路只出现一次。

Output

  包含两行。第一行为一个整数,表示最小的“不便利值”。第二行为一个整数,表示使“不便利值”达到最小时
不同的设计路线的方法总数modQ的值。如果某个城市无法到达首都,则输出两行-1。

Sample Input

5 4 100
1 2
4 5
1 3
4 1

Sample Output

1
10

HINT

  以下样例中是10种设计路线的方法:

(1)4-5

(2)1-4-5

(3)4-5,1-2

(4)4-5,1-3

(5)4-5,2-1-3

(6)2-1-4-5

(7)3-1-4-5

(8)1-4

(9)2-1-4

(10)3-1-4

【数据规模和约定】

对于100%的数据,满足1≤N,M≤100000,1≤Q≤120000000。

Source

[ Submit][ Status][ Discuss]


题解:树形dp

我们可以发现这种修建的方式最终形成的关系很想树链剖分中的轻重链,每个点到首都的不便利值其实就是走过的轻边的数量。对于这个题来说显然我们需要轻边的数量尽可能小,树链剖分中这个数目不超过logn个,所以对于这个题来说第一问的答案一定不超过logn,而这个题与树链剖分还存在不同,因为还可以V型覆盖,就是路径的起终点在同一个点的子树中,对于一个点来说,最多有一个V型覆盖,也最多只有一个简单路径覆盖。

设f[n][m][i]表示以n为根的子树中,答案不超过m,与儿子有i条重边相连的方案数。

其中i<=2

定义g[v][0]表示从n出发走轻边到达父亲节点

g[v][0]=f[v][m-1][0]+f[v][m-1][1]+f[v][m-1][2]

g[v][1]表示从n出发走重边到达父亲节点

g[v][1]=f[v][m-1][0]+f[v][m-1][1] (没有f[v][m-1][2]是因为如果是走重边上来,那么v这个点下面的儿子中只可能有一个儿子是重边)

然后考虑如何计算f[n][m][i]


因为模数的原因,有可能最终的结果是0,为了区分最终结果是0还是不存在合法解。我们在x%p时保证若x!=0&&x%p=0,那么x%p=p

最后找到答案不为0的最小的m即可

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define N 100003
#define LL long long
using namespace std;
int n,m;
LL f[N][23][4],g[N][3],p;
int point[N],nxt[N*2],v[N*2],tot;
void add(int x,int y)
{
	tot++; nxt[tot]=point[x]; point[x]=tot; v[tot]=y;
	tot++; nxt[tot]=point[y]; point[y]=tot; v[tot]=x; 
}
LL solve(LL x)
{
	if (x&&x%p==0) return p;
	else return x%p;
}
void dfs(int x,int fa)
{
	bool pd=false;
	for (int i=0;i<=20;i++) f[x][i][0]=1;
	for (int i=point[x];i;i=nxt[i])
	if (v[i]!=fa) {
		dfs(v[i],x);
	    for (int j=0;j<=20;j++)
	      {
		  	if (j) 
			  g[v[i]][0]=solve(f[v[i]][j-1][0]+f[v[i]][j-1][1]+f[v[i]][j-1][2]);
		  	g[v[i]][1]=solve(f[v[i]][j][0]+f[v[i]][j][1]);
		  	f[x][j][2]=solve(f[x][j][1]*g[v[i]][1]+f[x][j][2]*g[v[i]][0]);
		  	f[x][j][1]=solve(f[x][j][0]*g[v[i]][1]+f[x][j][1]*g[v[i]][0]);
			f[x][j][0]=solve(f[x][j][0]*g[v[i]][0]);
		 	pd=true; 
		  }
	}
}
int main()
{
	freopen("design.in","r",stdin);
	freopen("design.out","w",stdout);
	scanf("%d%d%I64d",&n,&m,&p);
	if (m<n-1) {
		printf("-1\n-1\n");
		return 0;
	}
	for (int i=1;i<=m;i++) {
		int x,y; scanf("%d%d",&x,&y);
		add(x,y);
	}
	memset(f,0,sizeof(f));
	dfs(1,0);
	for (int i=0;i<=20;i++)
	 if (f[1][i][0]+f[1][i][1]+f[1][i][2]>0)
	  {
	  	 printf("%d\n",i);
	  	 printf("%I64d\n",(f[1][i][0]+f[1][i][1]+f[1][i][2])%p);
	  	 return 0;
	  }
	printf("-1\n-1\n");
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值