[BZOJ3244][NOI2013]树的计数

时间:2022-04-09 23:51:27

这题大家为什么都写O(NlogN)的算法呢?……

让本蒟蒻来写一个O(N)的吧……

首先还是对BFS序和DFS序重编号,记标好的DFS序为d[1..n]。令pos[x]为x在d[]中出现的位置,即pos[d[i]]=i。

然后还是要用到一个BFS序的分段对应一棵树的结论……然后我们考察一个分段方式的合法性:首先结点1是唯一的根必须要单独一段;其次,BFS序中一层的结点出现的顺序和DFS序中的顺序一定是相同的,因此对于任何的一段[l, r],都有pos[l]<pos[l+1]<pos[l+2]<…<pos[r];最后我们还需要考虑结点深度的约束,对于DFS序中两个相邻结点,后一个结点的深度至多比前一个结点大1。综上所述,我们得到了以下三条约束:

(1)结点1单独分为一段;

(2)对于任何一段[l, r],pos[l]<pos[l+1]<pos[l+2]<…<pos[r];

(3)记depth[i]为结点i的深度,则depth[d[i+1]]<=depth[d[i]]+1。

我们尝试转化上面3条约束。建立数组s[1..n-1],s[i]=1当且仅当结点i和结点i+1分在不同的两段,否则s[i]=0。这样进行转化之后,以上三条约束变成了如下的形式:

(1)s[1]=1;

(2)若pos[i]>pos[i+1],则s[i]=1;

(3)若d[i]<d[i+1],则Σ{s[j] | j=d[i]..d[i+1]-1}<=1。

第一条转化是容易理解的。第三条转化也容易理解,因为s[d[i]]、s[d[i]+1]、…、s[d[i+1]-1]的和就等于结点d[i]和d[i+1]的高度差。当d[i]>d[i+1]时,由于d[i]的BFS序在d[i+1]后面,所以depth[d[i+1]]<=depth[d[i]]<depth[d[i]]+1。第二条的转化是把原先的描述反过来理解形成的,也就是说若pos[i]>pos[i+1],那么结点i一定不满足约束,所以必须分为两段。可以证明这样的转化与原来的约束是等效的。

显然,1+Σs[i]就是树高。

我们考虑在O(N)时间内计算出树高的期望。约束1和约束2把一些s[i]固定为1,它们对s[]造成的影响可以非常容易地O(N)处理出来。然后考虑第三类约束,首先我们处理所有Σ{s[j] | j=d[i]..d[i+1]-1}=1的第三类约束,这类约束固定了一个区间的值(某一个s[i]=1,其余s[i]均为0)。这些值可以通过前缀和的处理技巧O(1)地打好固定标记,所以处理这些约束也是O(N)的。

最后我们剩下一些Σ{s[j] | j=d[i]..d[i+1]-1}=0的约束。由于每一个s[j]都是0,这意味着对于所有的j=l..r-1,都有pos[j]<pos[j+1](否则处理第二类约束时会使得某个s[j]=1,从而不满足Σs[j]=0)。因此,我们有pos[l]<pos[l+1]<pos[l+2]<…<pos[r]。注意到这里l=d[i]而r=d[i+1],所以pos[l]=i而pos[r]=i+1。也就是说这些剩下的约束其实都是一些形如i<i+1的恒成立的不等式,直接无视掉就好了。这样就处理完了所有的约束,最后我们还剩下一些位置没有被固定,这些位置取0或1均可,对答案的贡献就是0.5。

于是我们就可以O(N)地解决这个问题。代码异常的简洁,线段树什么的根本不用写嘛……

 /**************************************************************
Problem: 3244
User: IcyF
Language: C++
Result: Accepted
Time:216 ms
Memory:6276 kb
****************************************************************/ #include <cstdlib>
#include <cstdio> using namespace std; #define rep(n) for (int i = 1; i <= n; ++i)
#define sum(l, r) (s[r] - s[l - 1]) const int MAXN = ;
int n, a[MAXN], b[MAXN], c[MAXN], pos[MAXN], x[MAXN], s[MAXN], fix[MAXN];
int main()
{
scanf("%d", &n);
rep(n) scanf("%d", a + i);
rep(n) scanf("%d", b + i);
rep(n) c[b[i]] = i;
rep(n) a[i] = c[a[i]];
rep(n) pos[a[i]] = i; x[] = , ++fix[], --fix[];
rep(n - )
if (pos[i] > pos[i + ])
x[i] = , ++fix[i], --fix[i + ];
rep(n - )
s[i] = s[i - ] + x[i];
rep(n - )
if (a[i] < a[i + ])
{
int t = sum(a[i], a[i + ] - );
if (t)
{
++fix[a[i]];
--fix[a[i + ]];
}
} double ans = ; int f = ;
rep(n - )
{
f += fix[i];
if (f)
ans += double(x[i]);
else
ans += 0.5;
}
printf("%.3lf\n", ans + 0.999);
printf("%.3lf\n", ans + 1.0);
printf("%.3lf\n", ans + 1.001);
return ;
}

[BZOJ3244][NOI2013]树的计数的更多相关文章

  1. BZOJ3244 NOI2013树的计数(概率期望)

    容易发现的一点是如果确定了每一层有哪些点,树的形态就确定了.问题变为划分bfs序. 考虑怎样划分是合法的.同一层的点在bfs序中出现顺序与dfs序中相同.对于dfs序中相邻两点依次设为x和y,y至多在 ...

  2. &lbrack;bzoj3244&rsqb;&lbrack;noi2013&rsqb;树的计数 题解

    UPD: 那位神牛的题解更新了,在这里. ------------------------------------------------------------------------------- ...

  3. BZOJ3244 &lbrack;Noi2013&rsqb;树的计数 【数学期望 &plus; 树遍历】

    题目链接 BZOJ3244 题解 不会做orz 我们要挖掘出\(bfs\)序和\(dfs\)序的性质 ①容易知道\(bfs\)序一定是一层一层的,如果我们能确定在\(bfs\)序中各层的断点,就能确定 ...

  4. &lbrack;UOJ&num;122&rsqb;&lbrack;NOI2013&rsqb;树的计数

    [UOJ#122][NOI2013]树的计数 试题描述 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的 DFS 序以及 BFS 序.两棵不同的树的 DFS 序 ...

  5. 【BZOJ3244】【UOJ&num;122】【NOI2013&rsqb;树的计数

    NOI都是酱的题怎么玩啊,哇.jpg 原题: 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的 ...

  6. BZOJ3244&sol;UOJ122 &lbrack;Noi2013&rsqb;树的计数

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...

  7. &lbrack;bzoj3244&rsqb; &lbrack;洛谷P1232&rsqb; &lbrack;Noi2013&rsqb; 树的计数

    Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同, ...

  8. 3244&colon; &lbrack;Noi2013&rsqb;树的计数 - BZOJ

    Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同, ...

  9. bzoj 3244&colon; &lbrack;Noi2013&rsqb;树的计数

    Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同, ...

随机推荐

  1. Maven工程中的右键team

    与资源库同步(S):在需要合并版本时使用 提交(C):本地代码写入源码库 更新(U):本地代码升级到服务器端版本 在点击更新时,请注意: 如果当前项目有改动(甚至是比原来多了一个空格),则此时无法更新 ...

  2. window&period;open和window&period;showdialog区别

    open打开的窗口可以点击切换到其背后的父窗口,dialog的窗口无法点击切换到其背后的父窗口, 假如用window.opener或者parent等对象时,建议用open方法,不要用dialog,否则 ...

  3. jdk源码调试功能

    JDK源码重新编译——支持eclipse调试JDK源码--转载 最近在研究jdk源码,发现debug时无法查看源码里的变量值. 因为sun提供的jdk并不能查看运行中的局部变量,需要重新编译一下rt. ...

  4. 异常学习笔记&plus;打包&plus;doc该软件包编译

    jvm调用默认的异常处理机制printStackTrace办法      欲了解更多异常处理.问题      捕获异常代码块出现继承关系 应该把被继承的异常放在子类异常块的后面 watermark/2 ...

  5. Android studio 打开别人的工程

    Android Studio正确打开项目只需要两步,或者说找到两个文件进行简单的修改就好,最好在打开之前进行修改 (1)gradle-wrapper.properities,在项目下按照如下路径可以找 ...

  6. Java:bufferedReader&period;readLine&lpar;&rpar;读取文件换行问题

    代码实现读取到的内容正常换行,并将内容复制到系统剪贴板当中去. public static void ReadAlart() { try { String encoding="utf-8&q ...

  7. 第28章 确认&lpar;Consent&rpar; - Identity Server 4 中文文档&lpar;v1&period;0&period;0&rpar;

    在授权请求期间,如果IdentityServer需要用户同意,则浏览器将被重定向到同意页面. 同意用于允许最终用户授予客户端对资源(身份或API)的访问权限.这通常仅对第三方客户端是必需的,并且可以在 ...

  8. 浅谈编码Base64、Hex、UTF-8、Unicode、GBK等

    网络上大多精彩的回答,该随笔用作自我总结: 首先计算机只认得二进制,0和1,所以我们现在看到的字都是经过二进制数据编码后的:计算机能针对0和1的组合做很多事情,这些规则都是人定义的:然后有了字节的概念 ...

  9. &lbrack;svc&rsqb;linux日志和安全日志

    last详解: http://www.cnblogs.com/kerrycode/p/4223751.html

  10. 02 - Unit05:加载笔记列表

    加载笔记列表 发送Ajax请求 绑定发送事件 获取参数: bookId 发送请求: /note/list.do 事件绑定 $(function(){ Ajax请求发送-->服务器处理--> ...