洛谷 P1351 联合权值 题解
P1351 联合权值
题目描述
无向连通图 \(G\) 有 \(n\) 个点,\(n-1\) 条边。点从 \(1\) 到 \(n\) 依次编号,编号为 \(i\) 的点的权值为 \(W_i\),每条边的长度均为 \(1\)。图上两点 \((u, v)\) 的距离定义为 \(u\) 点到 \(v\) 点的最短距离。对于图 \(G\) 上的点对 \((u, v)\),若它们的距离为 \(2\),则它们之间会产生\(W_v \times W_u\) 的联合权值。
请问图 \(G\) 上所有可产生联合权值的有序点对中,联合权值最大的是多少?所有联合权值之和是多少?
输入格式
第一行包含 \(1\) 个整数 \(n\)。
接下来 \(n-1\) 行,每行包含 \(2\) 个用空格隔开的正整数 \(u,v\),表示编号为 \(u\) 和编号为 \(v\) 的点之间有边相连。
最后 \(1\) 行,包含 \(n\) 个正整数,每两个正整数之间用一个空格隔开,其中第 \(i\) 个整数表示图 \(G\) 上编号为 \(i\) 的点的权值为 \(W_i\)。
输出格式
输出共 \(1\) 行,包含 \(2\) 个整数,之间用一个空格隔开,依次为图 \(G\) 上联合权值的最大值和所有联合权值之和。由于所有联合权值之和可能很大,输出它时要对\(10007\)取余。
输入输出样例
输入 #1
5
1 2
2 3
3 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 \leq 100\);
对于60%的数据,\(1 < n \leq 2000\);
对于100%的数据,\(1 < n \leq 200000, 0 < W_i \leq 10000\)。
保证一定存在可产生联合权值的有序点对。
【思路】
树上问题 + dfs
一开始没往树上操作这个地方想就想打个暴力搜索那点部分分
然后发现暴搜是很麻烦的,好像是没法处理类似菊花图的图形
所以只能上树
可以分为两个部分第一个部分先预处理出每一个点对应的长度为1的菊花图里面的最大值和最大值的位置,同时也求出
这个点对应的菊花图里面所有子节点的和
因为后面求总和的时候可以用一个点的值乘以这个菊花图的总和减去那个点的值
然后再来一个dfs
先处理处不是菊花图也就是直上直下不拐弯的那种可能出现的情况也暴力比较找最大值
然后再枚举菊花图,比较里面哪一个除了最大点的点和最大的的乘积最大
也就是找出次大点记录,同时求和,就用上面的方法就可以了
【完整代码】
#include<iostream>
#include<cstdio>
#include<vector>
using namespace std;
const int Max = 2e5 + 10;
const int mo = 1e4 + 7;
int a[Max];
int father[Max];
int maxw[Max],maxv[Max];//某个点对应的子节点里面的最大值和最大值的位置
int sum[Max];//某个点对应的子节点之和
vector<int>e[Max];
int n;
int ans1,ans2;
void dfs1(int x,int fa)
{
for(int i = 0;i < e[x].size();i ++)
{
if(e[x][i] != fa)
{
if(a[e[x][i]] > maxw[x])
maxw[x] = a[e[x][i]],maxv[x] = e[x][i];
sum[x] = (sum[x] + a[e[x][i]]) % mo;
dfs1(e[x][i],x);
}
}
}
void dfs2(int x,int fa)
{
father[x] = fa;
if(father[father[x]] != 0)
{
ans1 = max(ans1,a[father[father[x]]] * a[x]);
ans2 = (ans2 + 2 * a[x] * a[father[father[x]]]) % mo;
}
for(int i = 0;i < e[x].size();i ++)
{
if(e[x][i] != fa)
{
if(e[x][i] != maxv[x])ans1 = max(ans1,a[e[x][i]] * maxw[x]);
ans2 = (ans2 + a[e[x][i]] * (sum[x] - a[e[x][i]])) % mo;
dfs2(e[x][i],x);
}
}
}
int main()
{
// freopen("2","r",stdin);
int x,y;
scanf("%d",&n);
// for(int i = 1;i <= n;++ i)
// father[i] = i;
for(int i = 1;i < n;++ i)
{
scanf("%d%d",&x,&y);
e[x].push_back(y);
e[y].push_back(x);
}
for(int i = 1;i <= n;++ i)
scanf("%d",&a[i]);
dfs1(1,0);
dfs2(1,0);
cout<<ans1<<" "<<ans2 % mo<<endl;
return 0;
}
洛谷 P1351 联合权值 题解的更多相关文章
- 『题解』洛谷P1351 联合权值
更好的阅读体验 Portal Portal1: Luogu Portal2: LibreOJ Description 无向连通图\(\mathrm G\)有\(n\)个点,\(n - 1\)条边.点从 ...
- [NOIP2014] 提高组 洛谷P1351 联合权值
题目描述 无向连通图G 有n 个点,n - 1 条边.点从1 到n 依次编号,编号为 i 的点的权值为W i ,每条边的长度均为1 .图上两点( u , v ) 的距离定义为u 点到v 点的最短距离. ...
- 洛谷 P1351 联合权值
题目描述 无向连通图G 有n 个点,n - 1 条边.点从1 到n 依次编号,编号为 i 的点的权值为W i ,每条边的长度均为1 .图上两点( u , v ) 的距离定义为u 点到v 点的最短距离. ...
- 洛谷——P1351 联合权值
https://www.luogu.org/problem/show?pid=1351 题目描述 无向连通图G 有n 个点,n - 1 条边.点从1 到n 依次编号,编号为 i 的点的权值为W i , ...
- 洛谷P1351 联合权值(树形dp)
题意 题目链接 Sol 一道很简单的树形dp,然而被我写的这么长 分别记录下距离为\(1/2\)的点数,权值和,最大值.以及相邻儿子之间的贡献. 树形dp一波.. #include<bits/s ...
- 洛谷 P1351 联合权值 —— 树形DP
题目:https://www.luogu.org/problemnew/show/P1351 树形DP,别忘了子树之间的情况(拐一下距离为2). 代码如下: #include<iostream& ...
- 洛谷P1351 联合权值
\(\Large\textbf{Description:}\) \(\large一棵树,父子之间距离为1,求距离为2的两点点权之积的最大值与和.\) \(\Large\textbf{Solution: ...
- Luogu P1351 联合权值 题解
这是一个不错的树形结构的题,由于本蒟蒻不会推什么神奇的公式其实是懒得推...,所以很愉快的发现其实只需要两个点之间的关系为祖父和儿子.或者是兄弟即可. 然后问题就变得很简单了,只需要做一个正常的DFS ...
- 洛谷 1351 联合权值——树形dp
题目:https://www.luogu.org/problemnew/show/P1351 对拍了一下,才发现自己漏掉了那种拐弯的情况. #include<iostream> #incl ...
随机推荐
- ELK学习笔记之logstash将配置写在多个文件
0x00 概述 我们用Logsatsh写配置文件的时候,如果读取的文件太多,匹配的正则过多,会使配置文件动辄成百上千行代码,可能会造成阅读和修改困难.这时候,我们可以将配置文件的输入.过滤.输出分别放 ...
- C# vb .net实现马赛克焦距像素化特效滤镜
在.net中,如何简单快捷地实现Photoshop滤镜组中的马赛克焦距像素化效果呢?答案是调用SharpImage!专业图像特效滤镜和合成类库.下面开始演示关键代码,您也可以在文末下载全部源码: 设置 ...
- 2019年Amazon AWS-Solutions-Architect-Professional考试最新题库(AWS SAP题库)带考试模拟器
大家好,由于最近自己备考Amazon AWS-Solutions-Architect-Professional考试,购买了以下链接的题库,并通过了考试 https://www.kaoguti.gq/A ...
- win10设置锁屏密码
1.点击右下角窗口键 2.选择点击设置 3.点击账户 4.点击登录选项 5.点击密码,添加密码 6.设置密码 7.使用快捷键“窗口键+l”锁屏,就会提示你输入密码
- Abp vNext抽茧剥丝01 使用using临时更改当前租户
在Abp vNext中,如果开启了多租户功能,在业务代码中默认使用当前租户的数据,如果我们需要更改当前租户,可以使用下面的方法 /* 此时当前租户 */ using (CurrentTenant.Ch ...
- 爬取网易云音乐歌手和id
pip install lxml csv requests from lxml import etree from time import sleep import csv import reques ...
- Vue+Django项目部署
本地项目配置 1 复制 luffy/settings/dev.py为prop.py 修改luffy/settings/prop.py中以下几项 (1) allow_hosts ALLOWED_HOST ...
- 19,flask消息闪现-flash
Flash消息 请求完成后给用户的提醒消息,flask的核心特性, flash函数实现效果 视图函数中调用flash()方法 html中要使用get_flashed_messages() 后端代码: ...
- Linux命令——mknode
参考:What is the mknod command used for? 前言 Linux下面,一切皆文件,当然也包括设备.Linux通过major.minor号来区分不同设备,如下图
- 微信小程序校验文件在浏览器无法打开
txt文件放在网站根目录后浏览器无法打开,同一目录的其他txt文件却可以打开.我试过了修改文件权限.修改所有者.修改文件编码等各种办法都没有用. 最后找到了.htaccess文件,原来在这个文件里修改 ...