题面

传送门:https://www.luogu.org/problemnew/show/P1268


Solution

这是一道极其巧妙的构造题

先做一个约定[i,j]表示从i到j的距离

我们可以先从n=2,也就是最简单的情况来切入这道题

对于n=2,答案显然是[1,2]

接下来考虑n=3

如下图所示

这棵树一定是长成这样的

也就是说三这个节点一定是插在1与2两个节点之间的

我们可以发现,3节点的插入使得树的权值增加了([1,3]+[2,3]-[1,2])/2 (即紫线与蓝线的权值和减去绿线除以二)

我们可以把这个权值贡献的式子推广到一般情况

即x节点插入在[i,j]路径上

其对答案的贡献为([i,x]+[j,x]-[i,j])/2

接下来,我们继续把之前的结论推广到一般情况

四号节点接下来是不是有可能加入在[1,2],[1,3],[2,3]这三条路径中

答案要求整棵树权值和尽可能小,我们只需要在三种情况中选最小值就好

我想你已经找到了一个O(n^3)的算法

就是枚举3~n的插入点,再用两层循环枚举所插入的边

复杂度O(n^3)

事实上,我们已经可以过这道题了.

但我们的复杂度还可以更优

重新再看一下这张图

是不是可以发现我们紫色的那个其实是重复枚举

因为[1,4]+[3,4]-[1,3] 和[2,4]+[3,4]-[2,3] 其实都是一毛一样的!!!

这个结论也可以推广至一般状况中

所以说我们完全可以省去枚举中的一维

只枚举1~n就好

时间复杂度O(n^2)


Code

//Luogu P1268 树的重量
//May,30th,2018
//构造妙题
#include<iostream>
#include<cstdio>
using namespace std;
long long read()
{
long long x=0,f=1; char c=getchar();
while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
const int N=30+5;
int n,a[N][N];
int main()
{
while(1)
{
memset(a,0,sizeof a);
int ans=0;
n=read();
if(n==0) break;
for(int i=1;i<n;i++)
for(int j=i+1;j<=n;j++)
a[i][j]=a[j][i]=read(); ans+=a[1][2];
for(int i=3;i<=n;i++)
{
int t_ans=0x3f3f3f3f;
for(int j=1;j<i;j++)
for(int k=1;k<j;k++)
t_ans=min(t_ans,(a[j][i]+a[k][i]-a[j][k])/2);
ans+=t_ans;
}
printf("%d\n",ans);
}
return 0;
}

正解(三方)(C++)

//Luogu P1268 树的重量
//May,30th,2018
//构造妙题
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
long long read()
{
long long x=0,f=1; char c=getchar();
while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
const int N=30+5;
int n,a[N][N];
int main()
{
while(1)
{
memset(a,0,sizeof a);
int ans=0;
n=read();
if(n==0) break;
for(int i=1;i<n;i++)
for(int j=i+1;j<=n;j++)
a[i][j]=a[j][i]=read(); ans+=a[1][2];
for(int i=3;i<=n;i++)
{
int t_ans=0x3f3f3f3f;
for(int j=1;j<i;j++)
t_ans=min(t_ans,(a[j][i]+a[1][i]-a[j][1])/2);
ans+=t_ans;
}
printf("%d\n",ans);
}
return 0;
}

正解(平方)(C++)

[Luogu P1268] 树的重量 (巧妙的构造题)的更多相关文章

  1. luogu p1268 树的重量——构造,真正考验编程能力

    题目链接:http://www.luogu.org/problem/show?pid=1268#sub -------- 这道题费了我不少心思= =其实思路和标称毫无差别,但是由于不习惯ACM风格的题 ...

  2. 【luogu P1268 树的重量】 题解

    题目链接:https://www.luogu.org/problemnew/show/P1268 给定所有点间的最短路求原图所有路径和 形如: 我们需要计算红边+绿边 绿边 = (红边+蓝边+紫边)/ ...

  3. luogu P1268 树的重量

    一开始把这题想复杂了,,, 这里记\(di[i][j]\)表示\(i\)到\(j\)的距离 首先如果\(n=2\),答案显然为\(di[1][2]\) 如果\(n=3\) 懒得画图了盗图过来 那么3号 ...

  4. 洛谷—— P1268 树的重量

    P1268 树的重量 构造类题目,看不出个所以然来... emmm,只好看题解: 只有两个点,那一条路径就是$ans$ 考虑三个点,那么$3$这个点相对于树上的路径(已经加入树上的边的距离) 为:$( ...

  5. 洛谷P1268 树的重量

    P1268 树的重量 85通过 141提交 题目提供者该用户不存在 标签树形结构 难度提高+/省选- 提交该题 讨论 题解 记录 最新讨论 有这种情况吗!!!! 题意似乎有问题 题目描述 树可以用来表 ...

  6. 洛谷 P1268 树的重量 解题报告

    P1268 树的重量 题目描述 树可以用来表示物种之间的进化关系.一棵"进化树"是一个带边权的树,其叶节点表示一个物种,两个叶节点之间的距离表示两个物种的差异.现在,一个重要的问题 ...

  7. 洛谷P1268 树的重量 【构造 + 枚举】

    题目描述 树可以用来表示物种之间的进化关系.一棵"进化树"是一个带边权的树,其叶节点表示一个物种,两个叶节点之间的距离表示两个物种的差异.现在,一个重要的问题是,根据物种之间的距离 ...

  8. P1268 树的重量【构造】

    题目描述 树可以用来表示物种之间的进化关系.一棵“进化树”是一个带边权的树,其叶节点表示一个物种,两个叶节点之间的距离表示两个物种的差异.现在,一个重要的问题是,根据物种之间的距离,重构相应的“进化树 ...

  9. P1268 树的重量

    题目描述 树可以用来表示物种之间的进化关系.一棵“进化树”是一个带边权的树,其叶节点表示一个物种,两个叶节点之间的距离表示两个物种的差异.现在,一个重要的问题是,根据物种之间的距离,重构相应的“进化树 ...

随机推荐

  1. Python-序列常用方法 + * += extend append方法区别

    + 两边都是相同序列类型.拼接成一个新的序列 print((1, 2, 3) + (4, 6)) print("beimenchuixue" + "ximenchuife ...

  2. 借助C++探究素数的分布

    这里使用的区间是36,144,576,2304,9216,36864,147456,589824,2359296,9437184.至于这个区间是怎么得到的,感兴趣的同鞋可前往(https://www. ...

  3. 山寨一个Spring的@Component注解

    1. 前言 我们在上一篇对Mybatis如何将Mapper接口注入Spring IoC进行了分析,有同学问胖哥这个有什么用,这个作用其实挺大的,比如让你实现一个类似@Controller的注解(或者继 ...

  4. arduino 动态内存不足问题

    亲测有用, 参考:https://blog.csdn.net/weixin_33915554/article/details/86975847 如果在代码中使用到一个数组,但是数组容量超过20000个 ...

  5. devops-jenkins部署和基本使用

    1. jenkins部署和基本使用  1.1) 先关闭centos 7的自带防火墙和selinux [root@test-2 ~]# /bin/systemctl stop firewalld [ro ...

  6. 第四届58topcoder编程大赛--地图路径规划

    layout: post title: 第四届58topcoder编程大赛 subtitle: 58ACM catalog: true tags: - A* 算法 - C++ - 程序设计 问题及背景 ...

  7. 2016-12-04---tiny412平台下的iconv库的移植问题

    一.解决问题    在arm开发板上使用framebuff,在汉字显示时,因为只有gb2312的16*16的汉字字库,而ubuntu16.04默认    的编码方式时utf-8,因此需要进行转码(ut ...

  8. activiti 流程部署 保存流程图到数据库 保存二进制图片 存储失败

    activiti 流程部署 保存流程图到数据库  保存二进制图片 存储失败 具体错误如下 具体 junit测试 结果 :提示如下: 解决方法: 数据库版本不同 无法保存二进制文件到数据库表中!5.5. ...

  9. 【贪心算法】CF Emergency Evacuation

    题目大意 vjudge链接 给你一个车厢和一些人,这些人都坐在座位上,求这些人全部出去的时间最小值. 样例1输入 5 2 71 11 21 32 32 44 45 2 样例1输出 9 样例2输入 50 ...

  10. matplotlib条形图

    三个班级平均分 import matplotlib.pyplot as plt import matplotlib as mpl classes = ['class1','class2','class ...