题目传送门

话说这道题不分析样例实在是太亏了...结论题啊...

但是话说回来不知道它是结论题的时候会不会想到猜结论呢...毕竟样例一、二都有些特殊。

观察样例发现选中的子图都只有一条边。

于是猜只有一条边的时候解最优。

飞快地写个暴力,然后和结论对拍,然后假装这个结论是对的,然后就$AC$了(大雾

还是证明一下这个结论吧:

用反证法。

设这样三个点的点权分别为$A$,$B$,$C$,两条边的边权为$n$,$m$

假设子图中有$A,B,C$三个点比只有两个点更优。

也就是三个点都选的答案比只选$AB$和只选$BC$都大。

三个都选的答案:$(A+B+C)/(n+m)$

只选$AB$:$(A+B)/m$

只选$BC$:$(B+C)/n$

则:

$$(A+B+C)/(n+m)>(A+B)/m$$

$$(A+B+C)/(n+m)>(B+C)/n$$

化简:(权都是正数)

$$(A+B+C)*m>(A+B)*(n+m)$$

$$(A+B+C)*n>(B+C)*(n+m)$$

$$↓$$

$$C*m>A*n+B*n$$

$$A*n>B*m+C*m$$

相加:
$$C*m+A*n>A*n+B*n+B*m+C*m$$

$0>B*n+B*m$

由于$B$和$n$,$m$都是正数,导出矛盾。

所以假设不成立。

另外,如果$AC$之间有连边的话,那三个都选肯定更不优,分子不变,分母变大了嘛。只选$AC$都不用讨论,至少在这种情况下三个都选干不过只选$AB$或$BC$。

暴力程序(拿来对拍)

有同学写的$2^n$枚举子集的暴力,我觉得略麻烦,还是更喜欢自己的(笑)

甚至还想优化一下自己的暴力,就是在以$1$以外的点为起点的时候,就不把$1$加进去,因为$1$有的状态已经在$1$为起点的算过了。

但是反正是拿来写对拍嘛,节约考试时间。

 #include<cstdio>
#include<algorithm>
#include<vector>
#include<queue>
using namespace std;
#define N 505
#define ll long long
int n,m;
int a[N];
bool vis[N];
double ans=0.0;
vector<pair<int,int> >G[N];
int rd()
{
int f=,x=;char c=getchar();
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=(x<<)+(x<<)+(c^);c=getchar();}
return f*x;
}
void dfs(int u,int d,int b)
{
for(int i=;i<G[u].size();i++)
{
int v=G[u][i].first;
if(vis[v]) continue;
vis[v]=;
int tmp=;
for(int j=;j<G[v].size();j++)
if(vis[G[v][j].first]) tmp+=G[v][j].second;
dfs(v,d+a[v],b+tmp);
vis[v]=;
}
if(b==) return ;
ans=max(ans,1.0*d/b);
}
int main()
{
n=rd(),m=rd();
for(int i=;i<=n;i++)
a[i]=rd();
for(int i=;i<=m;i++)
{
int u=rd(),v=rd(),w=rd();
G[u].push_back(make_pair(v,w));
G[v].push_back(make_pair(u,w));
}
for(int i=;i<=n;i++)
{
vis[i]=;
dfs(i,a[i],);
vis[i]=;
}
printf("%.9f\n",ans);
return ;
}
//不分析样例真的是个不好的习惯啊

Code

正解程序:(比暴力好写)

 #include<cstdio>
#include<algorithm>
#include<vector>
#include<queue>
using namespace std;
#define N 505
#define ll long long
int n,m;
int a[N];
double ans=0.0;
int rd()
{
int f=,x=;char c=getchar();
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=(x<<)+(x<<)+(c^);c=getchar();}
return f*x;
}
int main()
{
n=rd(),m=rd();
for(int i=;i<=n;i++)
a[i]=rd();
for(int i=;i<=m;i++)
{
int u=rd(),v=rd(),w=rd();
ans=max(ans,1.0*(a[u]+a[v])/w);
}
printf("%.9f\n",ans);
return ;
}

Code

最新文章

  1. docker4dotnet #5 使用VSTS/TFS搭建基于容器的持续交付管道
  2. ACM/ICPC 之 一道不太简单的DP面试题(Geeksforgeeks)
  3. Android Volley入门到精通:定制自己的Request
  4. LeetCode: Nim Game
  5. Maven之问题解决汇总
  6. PHP获取搜索引擎关键字来源(百度、谷歌、雅虎、搜狗、搜搜、必应、有道)
  7. MD5和SHA512Managed ——哈希算法
  8. PHP中 post 与get的区别 详细说明
  9. 0基础搭建Hadoop大数据处理-编程
  10. CF.802C.Heidi and Library (hard) (费用流zkw)
  11. C# Aspose.Cells导出xlsx格式Excel,打开文件报“Excel 已完成文件级验证和修复。此工作簿的某些部分可能已被修复或丢弃”
  12. vue 生命周期钩子的理解 watch computed
  13. Java - HashMap 多线程安全解析
  14. 用Jersey构建RESTful服务7--Jersey+SQLServer+Hibernate4.3+Spring3.2
  15. (最大矩阵链乘)Matrix-chain product
  16. Exception.StackTrace
  17. oracle中如何设置主键并且让其自动增长
  18. [Java][Web]利用 referer 防盗链
  19. U盘装CentOS6.4
  20. eclipse.ini配置文件

热门文章

  1. Java实现从服务器下载文件到本地的工具类
  2. LiteOS的内核——RTOS基本的特性
  3. [转]Normal Map中的值, Tangent Space, 求算 Tangent 与 Binormal 与 TBN Matrix
  4. python类的多态、多态性
  5. Python数据类型知识点
  6. java8 time计算时间差
  7. mybatis标签selectkey无法返回主键值
  8. 小米 oj 纯位数
  9. NVMe Windows 支持情况
  10. UITableView动态改变Cell高度