博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
P1984 [SDOI2008]烧水问题(具体证明)
阅读量:5253 次
发布时间:2019-06-14

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

我见过的第二恶心的题,第一是糖果传递...

以下是一堆具体的证明,自己想的,可能考虑不周,不想看也可以直接看结论


 

首先有一个很显然的贪心,烧开的水要尽量把热量传递出去

所以有一个比较显然的方法:每杯水烧开后都与下一杯水热传递,平衡后再把剩下的温度与更后面一杯水热传递,这样一直下去...

十分显然把热量传递出去比不传递出去要更优

具体证明:设下一杯水温度为 $t_1$,此时上一杯水已经烧开,为$t_{max}$

如果它们之间进行热传递,那么下一杯水的热量就变成 $\frac{t_1+t_{max}}{2}$

显然温度会更高,所以烧开所需热量更少,答案会更优,并且可以容易推广到多杯水的情况

然后热量传出去了,按什么顺序烧开他们也是关键

方法:把温度从大到小排序,优先烧温度大的水再把热量传下去会更优

证明:设第一杯水温度为 $t_1$ ,第二杯水温度为 $t_2$,且 $t_1>t_2$

如果我们先烧第一杯水,那么需要升高的温度为 $t_{max} - t_1$ 

然后第一杯水烧开后再把热量传给第二杯水,第二杯水温度变成 $\frac{t_2+t_{max}}{2}$

把第二杯水烧开需要升高的温度为 $t_{max} - \frac{t_2+t_{max}}{2}$

因为水质量一样,所以升高的温度可以直接相加,化简得$\frac{3}{2}t_{max} - t_1 -\frac{t_2}{2}$

因为水的质量一样,所以温度的升高多少就相当于热量的消耗多少

如果我们先烧第二杯水再传温度给第一杯水再烧第一杯水

经过同样的方法可以算出总共升高的温度为$\frac{3}{2}t_{max} - t_2 -\frac{t_1}{2}$

因为 $t_1>t_2$ ,所以第一种方案更优,此结论用同样的方法也能推广到多杯水的情况

 


 

所以总结一下,就是优先烧温度大的水,然后把热量尽量传出去

知道了方案,每杯水消耗的热量自己找找规律就出来了

设 $f_n$ 表示第 n 杯水烧开消耗的热量

那么 $f_n=f_{n-1}*[1-\frac{1}{2}(n-1)]$

代码不解释

 

#include
#include
#include
#include
#include
using namespace std;typedef long long ll;inline int read(){ int x=0,f=1; char ch=getchar(); while(ch<'0'||ch>'9') { if(ch=='-') f=-1; ch=getchar(); } while(ch>='0'&&ch<='9') { x=(x<<1)+(x<<3)+(ch^48); ch=getchar(); } return x*f;}int n;double ans,now=420000.0;int main(){ n=read(); now/=n; for(int i=1;i<=n;i++) ans+=now,now*=(1-0.5/i); printf("%.2lf",ans); return 0;}

 

 

 

转载于:https://www.cnblogs.com/LLTYYC/p/9828420.html

你可能感兴趣的文章
反弹SHELL
查看>>
关闭Chrome浏览器的自动更新和升级提示
查看>>
移动、尺寸改变
查看>>
poj2255Tree Recovery【二叉树重构】
查看>>
tcpcopy 流量复制工具
查看>>
vue和react的区别
查看>>
第十一次作业
查看>>
负载均衡策略
查看>>
微信智能开放平台
查看>>
ArcGIS Engine 中的绘制与编辑
查看>>
Oracle--通配符、Escape转义字符、模糊查询语句
查看>>
子网划分讲解及练习(一)
查看>>
c# 文件笔记
查看>>
第一页 - 工具的使用(webstorm)
查看>>
Linux 进程资源用量监控和按用户设置进程限制
查看>>
IE浏览器整页截屏程序(二)
查看>>
D3.js 之 d3-shap 简介(转)
查看>>
制作满天星空
查看>>
类和结构
查看>>
CSS3选择器(二)之属性选择器
查看>>