博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
NOIp 2014 #2 联合权值 Label:图论 !!!未AC
阅读量:6186 次
发布时间:2019-06-21

本文共 3693 字,大约阅读时间需要 12 分钟。

题目描述

无向连通图G 有n 个点,n - 1 条边。点从1 到n 依次编号,编号为 i 的点的权值为W i ,每条边的长度均为1 。图上两点( u , v ) 的距离定义为u 点到v 点的最短距离。对于图G 上的点对( u, v) ,若它们的距离为2 ,则它们之间会产生Wu

×Wv 的联合权值。

请问图G 上所有可产生联合权值的有序点对中,联合权值最大的是多少?所有联合权值之和是多少?

输入输出格式

输入格式:

 

输入文件名为link .in。

第一行包含1 个整数n 。

接下来n - 1 行,每行包含 2 个用空格隔开的正整数u 、v ,表示编号为 u 和编号为v 的点之间有边相连。

最后1 行,包含 n 个正整数,每两个正整数之间用一个空格隔开,其中第 i 个整数表示图G 上编号为i 的点的权值为W i 。

 

输出格式:

 

输出文件名为link .out 。

输出共1 行,包含2 个整数,之间用一个空格隔开,依次为图G 上联合权值的最大值

和所有联合权值之和。由于所有联合权值之和可能很大,[b]输出它时要对10007 取余。 [/b]

 

输入输出样例

输入样例#1:
5  1 2  2 33 4  4 5  1 5 2 3 10
输出样例#1:
20 74

说明

 

本例输入的图如上所示,距离为2 的有序点对有( 1,3) 、( 2,4) 、( 3,1) 、( 3,5) 、( 4,2) 、( 5,3) 。

其联合权值分别为2 、15、2 、20、15、20。其中最大的是20,总和为74。

【数据说明】

对于30% 的数据,1 < n≤ 100 ;

对于60% 的数据,1 < n≤ 2000;

 对于100%的数据,1 < n≤ 200 , 000 ,0 < wi≤ 10, 000 。

代码

1 #include
2 #include
3 #include
4 #include
5 #include
6 #include
7 #define MAXN 500005 8 using namespace std; 9 10 struct cc{11 map
m;12 }nod[MAXN];13 14 int ans,sum,N;15 int vis[MAXN],w[MAXN];16 vector
G[MAXN];17 18 int cc(int x,int pre){19 if(pre!=-1){20 for(int i=0;i
30分,脑洞map
1 #include
2 #include
3 #include
4 #include
5 #include
6 #include
7 #define MAXN 500005 8 using namespace std; 9 10 int ans,sum,N;11 int vis[MAXN],w[MAXN];12 vector
G[MAXN];13 14 int cc(int x,int pre){15 if(pre!=-1){16 for(int i=0;i
60分,还是TLE,大雾
1 #include
2 #include
3 #include
4 #include
5 #include
6 #define MAXN 500005 7 using namespace std; 8 9 int ans,sum,N;10 int vis[MAXN],w[MAXN];11 vector
G[MAXN];12 13 int cc(int x){14 vis[x]=1;15 if(G[x].size()>=2){16 for(int i=0;i
60分的另外一种写法

 

 

题解://格式优化 by Radiumlrb

一、审题

首先我们一看到图就就知道图论有关,又因为有n个点和n-1条边,不难看出是一个无环图,这对题目的难度降低了很多。如果直接模拟的话肯定不行,会超时,所以我们应该换一种思维,及一个点周围的节点两两都能产生联合全值,
下面就来完善这一思想。

二、算法讨论

①、公式推导:
假如一个点x存在n个节点,分别为x1,x2…xn.
节点最大值只需存下节点中最大值,每次再用当前节点与最大节点相乘,在于max比较。
则关于x节点间的的联合全值之和为:
x(x2+x3…+xn)+x2(x1+x3…+xn)+xn(x1+x2…+xn-1) ;经整理可知就是每两个节点相乘,最后再成2即可;
化简为:
x1(x2+x3…+xn)+x2(x3+x4…+xn)+xn-2(xn-1+xn)+xn-1*xn
可理解为把每一个点和所有前面出现的的点之和相乘再相加最后乘2。

②、注意事项:

1、因为图是没有环的,所以一点到距离为二的点必定有且仅有一条,每一个点任意两个节点必将经过此点,所以我们不必讨论是否会重复。
2、因为数据范围很大,用邻接矩阵会爆内存,所以我们必须要用邻接矩阵才行(注意要存双向边)。
3、求联合权值之和时必须边做边对10007取模,不然会超出范围。

三、算法实现

用一个二维数组a存边,一维数组max1用来存该点节点中的最大值,大概就是这样吧,剩下的变量就在代码中一一解释。

四、代码展示

//Orz P党大神

1 c,qz,dd:array[0..400000]of longint;   2 n,i,tot,x,y,max:longint;   3 //max存最大值,tot存联合全值之和   4 procedure sb(x:longint);   5 var i,tot1,nn,ny,max1:longint;   6 begin   7   tot1:=0; nn:=dd[x]; max1:=0;   8   ny:=a[nn].y;  9 //循环列举该点的所有节点  tot1存前面出现的节点全值之和。10   repeat  11     if max1*qz[ny]>max then max:=max1*qz[ny];  12     if qz[ny]>max1 then max1:=qz[ny];  13     tot:=(tot+tot1*(qz[ny]mod 10007))mod 10007;  14 tot1:=(tot1+qz[ny])mod 10007;  15 //要不断对10007取模16     nn:=c[nn];  17     ny:=a[nn].y;  18   until nn=0;  19 end;  20 begin  21 read(n);  22 for i:=1 to n-1 do  23    begin  24 read(x,y);25 a[i*2-1].x:=x;a[i*2-1].y:=y;26 c[i*2-1]:=dd[x];dd[x]:=i*2-1;  27 //此处为双向边  dd数组用来把起点相同的边绑定在一起28 a[i*2].x:=y;  a[i*2].y:=x;   29 c[i*2]:=dd[y];dd[y]:=i*2;  30    end;  31 for i:=1 to n do    32     read(qz[i]);    33 for i:=1 to n do  34 sb(i);  35 tot:=tot mod 10007;  36 write(max,' ',tot*2 mod 10007);  37 //注意最后答案要乘238 end.

 

五、最后的寄语:

其实本题还有更简单的做法,就是搜边即可,没必要枚举每个点,这个留给大家自己去想,我就不多讲。

转载于:https://www.cnblogs.com/radiumlrb/p/6012303.html

你可能感兴趣的文章
java互联网相关技术
查看>>
手机单机游戏开发连载1游戏核心模块
查看>>
PKI体系结构之SCEP
查看>>
cisco和微软PING的差别
查看>>
Openstack虚机实例状态错误手工恢复vm_state:error
查看>>
synchronized详解
查看>>
CentOS安装中文输入法:ibus
查看>>
关于MAN命令
查看>>
***大会披露的Wi-Fi***没有威胁到企业WLAN
查看>>
我的友情链接
查看>>
iptables+geoip来过滤阻挡来源IP
查看>>
mysql5.7.11的多实例配置
查看>>
Excel出现链接错误,怎么解决??(原创)
查看>>
delphi中文插入数据库乱码解决
查看>>
mysql TIMESTAMP(时间戳)详解
查看>>
Linux MySQL Table is read only的具体解决方案
查看>>
Windows Server 2012从Evaluation版转成正式版
查看>>
流之过滤器流(缓冲流)
查看>>
root无法本地登录
查看>>
.net 4.0(2.0)“检测到有潜在危险的 Request.Form 值”的解决方法
查看>>