题目描述

小$w$在天堂看到了一棵世界树。
世界树上有$n$个节点,其中$1$节点为根,每个节点有一个正整数权值$c_i$。
现在小$w$想要对每个节点$u$求出它的祖先$v$中$\frac{c_v-c_u}{dis(u,v)}$的最小值。


输入格式

第一行一个整数$n$表示节点个数。
第二行$n$个整数,表示每个点的$c_i$。
第三行$n-1$个整数,表示$2$到$n$每个节点的父亲。


输出格式

$n-1$行,每行一个小数,表示$2$到$n$每个节点的答案,绝对值误差不超过${10}^{-6}$。


样例

样例输入:

8
31516 11930 18726 12481 79550 63015 64275 7608
1 1 2 4 2 4 5

样例输出:

19586.0000000000
12790.0000000000
-551.0000000000
-67069.0000000000
-51085.0000000000
-51794.0000000000
1440.6666666667


数据范围与提示

对于$20\%$的数据,$n\leqslant 500$。
对于$40\%$的数据,$n\leqslant 5\times {10}^4$。
对于另外$20\%$的数据,保证数据随机。
对于$100\%$的数据,$n\leqslant 5\times {10}^5,c_i\leqslant {10}^9$。


题解

$20\%$算法:

暴力枚举所有父亲,记得$double$就好啦,但是你会意外的发现你得了$50$分,为什么呢?

注意有另外$20\%$的数据为随机数据,也就意味着差不多是$\log n$层的,那么时间复杂度其实趋近于$\Theta(n\log n)$,至于另外的$10$分,那我也不知道啦。

最劣时间复杂度:$\Theta(n^2)$。

最优时间复杂度:$\Theta(n\log n)$。

期望得分:$20$分。

实际得分:$50$分。

$100\%$算法:

来化一下那个式子:$\frac{c_v-c_u}{dis(u,v)}=\frac{c_v-c_u}{dep_u-dep_v}=-\frac{c_u-c_v}{dep_u-dep_v}$。

惊喜的发现这是一个凸包,而且只用维护下凸包就好啦,但是你只能获得$80$分,为什么呢?

因为暴力弹栈最劣情况下是能把程序卡成$\Theta(n^2)$的,类似菊花图,那么怎么办呢?

考虑使用可持久化栈即可。

时间复杂度:$\Theta(n\log n)$。

期望得分:$100$分。

实际得分:$100$分。


代码时刻

$20\%$算法:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. int n;
  4. double c[500001];
  5. int fa[500001];
  6. double ans[500001];
  7. int main()
  8. {
  9. memset(ans,127,sizeof(ans));
  10. scanf("%d",&n);
  11. for(int i=1;i<=n;i++)
  12. scanf("%lf",&c[i]);
  13. for(int i=2;i<=n;i++)
  14. {
  15. int x;
  16. scanf("%d",&x);
  17. fa[i]=x;
  18. }
  19. for(int i=2;i<=n;i++)
  20. {
  21. int flag=i;
  22. double dis=0.0;
  23. while(flag!=1)
  24. {
  25. flag=fa[flag];
  26. dis++;
  27. ans[i]=min(ans[i],(c[flag]-c[i])/dis);
  28. }
  29. }
  30. for(int i=2;i<=n;i++)
  31. printf("%.8lf\n",ans[i]);
  32. return 0;
  33. }

$100\%$算法:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. struct rec
  4. {
  5. int nxt;
  6. int to;
  7. }e[500000];
  8. int head[500001],cnt;
  9. int n;
  10. int c[500001];
  11. int fa[500001][20];
  12. int depth[500001];
  13. int ans[500001];
  14. void add(int x,int y)
  15. {
  16. e[++cnt].nxt=head[x];
  17. e[cnt].to=y;
  18. head[x]=cnt;
  19. }
  20. double calc(int x,int y){return 1.0*(c[y]-c[x])/(depth[x]-depth[y]);}
  21. void dfs(int x)
  22. {
  23. int father=fa[x][0];
  24. for(int i=19;~i;i--)
  25. {
  26. if(fa[father][i]<=1)continue;
  27. if(calc(x,fa[father][i])>=calc(x,fa[fa[father][i]][0]))father=fa[fa[father][i]][0];
  28. }
  29. if(father>1&&calc(x,father)>=calc(x,fa[father][0]))father=fa[father][0];
  30. ans[x]=fa[x][0]=father;
  31. for(int i=1;i<=19;i++)fa[x][i]=fa[fa[x][i-1]][i-1];
  32. for(int i=head[x];i;i=e[i].nxt)
  33. {
  34. depth[e[i].to]=depth[x]+1;
  35. dfs(e[i].to);
  36. }
  37. }
  38. int main()
  39. {
  40. scanf("%d",&n);
  41. for(int i=1;i<=n;i++)
  42. scanf("%d",&c[i]);
  43. for(int i=2;i<=n;i++)
  44. {
  45. int x;
  46. scanf("%d",&x);
  47. fa[i][0]=x;
  48. add(x,i);
  49. }
  50. dfs(1);
  51. for(int i=2;i<=n;i++)printf("%.8lf\n",calc(i,ans[i]));
  52. return 0;
  53. }

rp++

[CSP-S模拟测试]:Lost My Music(凸包)的更多相关文章

  1. Android单元测试与模拟测试详解

    测试与基本规范 为什么需要测试? 为了稳定性,能够明确的了解是否正确的完成开发. 更加易于维护,能够在修改代码后保证功能不被破坏. 集成一些工具,规范开发规范,使得代码更加稳定( 如通过 phabri ...

  2. [开源]微信在线信息模拟测试工具(基于Senparc.Weixin.MP开发)

    目前为止似乎还没有看到过Web版的普通消息测试工具(除了官方针对高级接口的),现有的一些桌面版的几个测试工具也都是使用XML直接请求,非常不友好,我们来尝试做一个“面向对象”操作的测试工具. 测试工具 ...

  3. 安装nginx python uwsgi环境 以及模拟测试

    uwsgi帮助文档: http://uwsgi-docs-cn.readthedocs.io/zh_CN/latest/WSGIquickstart.html http://uwsgi-docs.re ...

  4. 利用Python中的mock库对Python代码进行模拟测试

    这篇文章主要介绍了利用Python中的mock库对Python代码进行模拟测试,mock库自从Python3.3依赖成为了Python的内置库,本文也等于介绍了该库的用法,需要的朋友可以参考下     ...

  5. 转 C#实现PID控制的模拟测试和曲线绘图

    C#实现PID控制的模拟测试和曲线绘图   本文分两部分,一部分是讲PID算法的实现,另一部分是讲如何用动态的曲线绘制出PID运算的结果. 首先,PID算法的理论模型请参考自动控制理论,最早出现的是模 ...

  6. Mockito:一个强大的用于Java开发的模拟测试框架

    https://blog.csdn.net/zhoudaxia/article/details/33056093 介绍 本文将介绍模拟测试框架Mockito的一些基础概念, 介绍该框架的优点,讲解应用 ...

  7. NOIP模拟测试1(2017081501)

    好,今天是cgg第一次举行模拟测试,希望各位支持. 时间限制:2小时 题目链接: 题目一:水得都没名字了 题目二:车站 题目三:选数 不要觉得2小时太少,我的题目很良心,都很简单. 答案可以在模拟测试 ...

  8. Mock 模拟测试简介及 Mockito 使用入门

    Mock 是什么mock 测试就是在测试过程中,对于某些不容易构造或者不容易获取的对象,用一个虚拟的对象来创建以便测试的测试方法.这个虚拟的对象就是mock对象.mock对象就是真实对象在调试期间的代 ...

  9. Mac下Jmeter快速安装与入门-模拟测试Post请求及设置Http头

    [1]去Apache官网下载 Binaries系列的最新Jmeter.gz包 [2]下载到本地之后解压缩,进入到解压之后的目录然后,找到apache-jmeter-4.0/bin/jmeter.sh ...

  10. Python 的mock模拟测试介绍

    如何不靠耐心测试 可能我们正在写一个社交软件并且想测试一下"发布到Facebook的功能",但是我们不希望每次运行测试集的时候都发布到Facebook上. Python的unitt ...

随机推荐

  1. Learn Python the hard way, ex39 列表的操作

    #!/usr/bin/python #coding:utf-8 ten_things = "apples oranges crows telephone light sugar" ...

  2. poj3253Fence Repair (Huffman)

    Huffman树:具有n个外部节点(叶子节点)的二叉树 每个外部节点都有一个对应的权值Wi 叶节点带权外部路径长度总和WPL=Wi*Li(i从1到n)最小(权越大的节点里根越进) 构造Huffman树 ...

  3. 关于Python的10大实用编程技巧

      Python 是一种通用的脚本开发语言,比其他编程语言更加简单.易学,其面向对象特性甚至比Java.C#..NET更加彻底,因此非常适合快速开发. Python 已经成为最受欢迎的程序设计语言之一 ...

  4. Python入门习题1.温度转换

    这一节的课堂例题为: 例1.编写一个Python程序,完成摄氏度到华氏度,华氏度到摄氏度的温度转换. 解: (1)分析问题:利用程序实现温度转换,由用户输入温度值,程序给出输出结果. (2)划分边界: ...

  5. Warning: session_start(): open(/var/lib/php/session/)

    Warning: session_start(): open(/var/lib/php/session/) 今天放置一个新的站点www.96net.com.cn在里面,登陆后台出现这种错,之后再lin ...

  6. 32.Group Anagrams(相同元素的不同组合)

    Level:   Medium 题目描述: Given an array of strings, group anagrams together. Example: Input: ["eat ...

  7. windows与linux安装Python虚拟环境

    我这里觉得还是一步到位用virtualenvwrapper  工具,不再讲述virtualenv了,有了工具很好用 windows : 首先安装工具 pip install virtualenvwra ...

  8. 【学习总结】Python-3-转义字符

    参考: 本教程的评论区:菜鸟教程-Python3-Python数字 转义字符: 在需要在字符中使用特殊字符时,python用反斜杠()转义字符 END

  9. Java笔试题-List l = new List()

    前言: 最近遇到的一道很基础的题,有时候大家可能离开了编译器就不行了. import java.util.List; /** * * @author catchegg * create date: 2 ...

  10. tomcat常用功能

    修改端口号 1024-655365 之间取端口号 Tomcat有3个重要端口: 默认访问端口:8080 默认监听关闭tomcat的端口:8005 默认AJP访问端口:8009 vim tomcat/c ...