NOI2013 树的计数
85%做法:
动态规划。
首先重编号,BFS序变成1...n,然后DFS序相应重编号。
记pos[i]为i号点在DFS中的位置,即pos[d[i]]=i。
记F[l][r]表示BFS序中l...r号点相同高度时的高度和,G[l][r]表示BFS序中l...r号点相同高度的个数。
我们要寻找合法的k,使得[r+1][k]作为下一层,然后转移:
F[r+1][k]+=F[l][r]+G[l][r]
G[r+1][k]+=G[l][r]
我们要找到合法的k,k要满足以下条件:
(1)[l,r]在DFS序中的下一个一定在[1,k]中。
分析:
假设A∈[l,r],那么A在DFS序中的下一个B只有下面3种情况:

这3种情况中,B的高度最多比A多1。
[k+1,N]的点应该在[r+1,k]那一层下面,如果[l,r]在DFS序中的下一个在[k+1,N]中,就会在[r+1,k]那一层乱入一个[k+1,N]的点,这样是违法的。
所以这个条件限制了[k+1,N]进入[r+1,k]那一层。
(2)r+1在DFS序中的上一个在[l,r]中。
如下面这种情况:

如果r+1在DFS序中的上一个不在[l,r]那么,r+1就不能另开一层了。
(3)[r+1,k]在DFS序中的位置是递增的。
是为了防止这种情况发生:

时间复杂度O(N^3),然而实际上满足条件的k并不多,可以过85%的数据。
#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<fstream>
#include<algorithm>
#include<cstring>
#include<string>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<utility>
#include<set>
#include<bitset>
#include<vector>
#include<functional>
#include<deque>
#include<cctype>
#include<climits>
#include<complex>
//#include<bits/stdc++.h>适用于CF,UOJ,但不适用于poj using namespace std; typedef long long LL;
typedef double DB;
typedef pair<int,int> PII;
typedef complex<DB> CP; #define mmst(a,v) memset(a,v,sizeof(a))
#define mmcy(a,b) memcpy(a,b,sizeof(a))
#define re(i,a,b) for(i=a;i<=b;i++)
#define red(i,a,b) for(i=a;i>=b;i--)
#define fi first
#define se second
#define m_p(a,b) make_pair(a,b)
#define SF scanf
#define PF printf
#define two(k) (1<<(k)) template<class T>inline T sqr(T x){return x*x;}
template<class T>inline void upmin(T &t,T tmp){if(t>tmp)t=tmp;}
template<class T>inline void upmax(T &t,T tmp){if(t<tmp)t=tmp;} const DB EPS=1e-;
inline int sgn(DB x){if(abs(x)<EPS)return ;return(x>)?:-;}
const DB Pi=acos(-1.0); inline void clear(vector<int> *A,int a,int b){int i,j;A->clear();re(i,,a)re(j,,b)A[i].push_back();} inline int gint()
{
int res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
}
inline LL gll()
{
LL res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
} const int maxN=; int N;
int d[maxN+],b[maxN+];
int bak[maxN+],pos[maxN+];
DB F[maxN+][maxN+],G[maxN+][maxN+];
DB sumF,sumG,ans; int main()
{
freopen("count.in","r",stdin);
freopen("count.out","w",stdout);
int i,j,k;
N=gint();
re(i,,N)d[i]=gint();
re(i,,N)b[i]=gint();
re(i,,N)bak[b[i]]=i;
re(i,,N)d[i]=bak[d[i]],b[i]=bak[b[i]];
re(i,,N)pos[d[i]]=i;
F[][]=1.0,G[][]=1.0;
re(i,,N)re(j,i,N)
{
if(G[i][j]==0.0)continue;
int t=;
re(k,i,j)upmax(t,d[pos[k]+]);
for(k=j+;k<=N;k++)
{
if(k<t)continue;
if(!(i<=d[pos[j+]-] && d[pos[j+]-]<=j))break;
if(k->=j+ && pos[k]<pos[k-])break;
F[j+][k]+=F[i][j]+G[i][j];
G[j+][k]+=G[i][j];
}
}
sumF=sumG=0.0;
re(i,,N)sumF+=F[i][N],sumG+=G[i][N];
ans=(sgn(sumG)==)?0.0:sumF/sumG;
PF("%0.3lf\n",ans);
}
100%做法:
动态规划。
首先重编号,BFS序变成1...n,然后DFS序相应重编号。
记pos[i]为i号点在DFS中的位置,即pos[d[i]]=i。
记h[i]为第i号点的期望高度。
容易知道h[1]为1,h[2]为2
我们按照BFS序从小到大枚举1到N,我们知道i号点和i-1号点高度相差0或1,且只能出现在下面蓝色框(i-1号点右边的兄弟),红色框(i-1号点左边兄弟的儿子,此时蓝色框为空)和绿色框中(i-1号点的儿子,此时蓝色框和红色框为空)。

1.pos[i-1]>pos[i]
i号点一定在红色框中,所以h[i]=h[i-1]+1
2.pos[i-1]<pos[i]
2.1 pos[i-1]+1!=pos[i]
i号点一定在蓝色框中。
如果i号点在绿色框中,那么i一定是i-1的第一个儿子,这样的话pos[i-1]+1=pos[i],矛盾,所以i号点一定在蓝色框中。
所以h[i]=h[i-1]
2.2 pos[i-1]+1=pos[i]
考虑绿色框,我们想要i号点在绿色框中就必须保证蓝色框和红色框为空。怎么保证呢?就是[1..i-1]号点在DFS序中的下一个都在[1..i]中。
2.2.1 [1..i-1]号点在DFS序中的下一个都在[1..i]中。
这时候i号点既可以在绿色框中也也可以在蓝色框中,所以h[i]=h[i-1]或h[i]=h[i-1]+1
因为这两种情况等概率,所以h[i]=h[i-1]+0.5
2.2.2 [1..i-1]号点在DFS序中的下一个并不都在[1..i]中。
这时候i号点只能在蓝色框,所以h[i]=h[i-1]
#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<fstream>
#include<algorithm>
#include<cstring>
#include<string>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<utility>
#include<set>
#include<bitset>
#include<vector>
#include<functional>
#include<deque>
#include<cctype>
#include<climits>
#include<complex>
//#include<bits/stdc++.h>适用于CF,UOJ,但不适用于poj using namespace std; typedef long long LL;
typedef double DB;
typedef pair<int,int> PII;
typedef complex<DB> CP; #define mmst(a,v) memset(a,v,sizeof(a))
#define mmcy(a,b) memcpy(a,b,sizeof(a))
#define re(i,a,b) for(i=a;i<=b;i++)
#define red(i,a,b) for(i=a;i>=b;i--)
#define fi first
#define se second
#define m_p(a,b) make_pair(a,b)
#define SF scanf
#define PF printf
#define two(k) (1<<(k)) template<class T>inline T sqr(T x){return x*x;}
template<class T>inline void upmin(T &t,T tmp){if(t>tmp)t=tmp;}
template<class T>inline void upmax(T &t,T tmp){if(t<tmp)t=tmp;} const DB EPS=1e-;
inline int sgn(DB x){if(abs(x)<EPS)return ;return(x>)?:-;}
const DB Pi=acos(-1.0); inline void clear(vector<int> *A,int a,int b){int i,j;A->clear();re(i,,a)re(j,,b)A[i].push_back();} inline int gint()
{
int res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
}
inline LL gll()
{
LL res=;bool neg=;char z;
for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
if(z==EOF)return ;
if(z=='-'){neg=;z=getchar();}
for(;z!=EOF && isdigit(z);res=res*+z-'',z=getchar());
return (neg)?-res:res;
} const int maxN=; int N;
int d[maxN+],b[maxN+];
int bak[maxN+],pos[maxN+];
DB h[maxN+]; int main()
{
freopen("count.in","r",stdin);
freopen("count.out","w",stdout);
int i;
N=gint();
re(i,,N)d[i]=gint();
re(i,,N)b[i]=gint();
re(i,,N)bak[b[i]]=i;
re(i,,N)d[i]=bak[d[i]],b[i]=bak[b[i]];
re(i,,N)pos[d[i]]=i;
int t=;
h[]=1.0;
upmax(t,d[pos[]+]);
h[]=2.0;
upmax(t,d[pos[]+]);
re(i,,N)
{
if(pos[i-]>pos[i])
h[i]=h[i-]+1.0;
else
if(pos[i-]+!=pos[i])
h[i]=h[i-];
else
if(t<=i)
h[i]=h[i-]+0.5;
else
h[i]=h[i-];
upmax(t,d[pos[i]+]);
}
PF("%0.3lf\n",h[N]);
return ;
}
NOI2013 树的计数的更多相关文章
- [UOJ#122][NOI2013]树的计数
[UOJ#122][NOI2013]树的计数 试题描述 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的 DFS 序以及 BFS 序.两棵不同的树的 DFS 序 ...
- 3244: [Noi2013]树的计数 - BZOJ
Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同, ...
- bzoj 3244: [Noi2013]树的计数
Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同, ...
- 【BZOJ3244】【UOJ#122】【NOI2013]树的计数
NOI都是酱的题怎么玩啊,哇.jpg 原题: 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的 ...
- BZOJ3244 NOI2013树的计数(概率期望)
容易发现的一点是如果确定了每一层有哪些点,树的形态就确定了.问题变为划分bfs序. 考虑怎样划分是合法的.同一层的点在bfs序中出现顺序与dfs序中相同.对于dfs序中相邻两点依次设为x和y,y至多在 ...
- 【uoj122】 NOI2013—树的计数
http://uoj.ac/problem/122 (题目链接) 题意 给出一棵树的dfs序和bfs序,保证一定可以构成一棵树.问构成的树的期望深度. Solution 这是一个悲伤的故事,我YY的东 ...
- [BZOJ3244][NOI2013]树的计数
这题大家为什么都写O(NlogN)的算法呢?…… 让本蒟蒻来写一个O(N)的吧…… 首先还是对BFS序和DFS序重编号,记标好的DFS序为d[1..n].令pos[x]为x在d[]中出现的位置,即po ...
- [bzoj3244][noi2013]树的计数 题解
UPD: 那位神牛的题解更新了,在这里. ------------------------------------------------------------------------------- ...
- BZOJ3244/UOJ122 [Noi2013]树的计数
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- luogu P1232 [NOI2013]树的计数
传送门 这题妙蛙 首先考虑构造出一个合法的树.先重新编号,把bfs序整成\(1,2,3...n\),然后bfs序就是按照从上到下从左往右的遍历顺序,所以可以考虑对bfs序分层,可以知道分层方式只会对应 ...
随机推荐
- js学习之道:js防止表单重复提交
第一种:用flag标识,下面的代码设置checkSubmitFlg标志: <script language="”javascript”"> var checkSubmi ...
- 介绍一款超实用的演讲必备工具 ZoomIt
最近在整理以前的文件时,发现躺在某个角落的一个超好的实用的小工具——ZoomIt.此工具是 SysinternalsSuite(已被MS收购)众多实用工具的一个. 相信大家在演讲时,经常需要调整屏幕文 ...
- 数据库中的schema概念
原文地址:http://blog.sina.com.cn/s/blog_7952e89001010jlj.html 数据库的初学者往往会对关系型数据库模式(schema).数据库(database). ...
- 为什么p标签不能嵌套div??
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta http ...
- node.js + express(ejs) + mongodb(mongoose) 增删改实例
MongoDB 安装步骤总结: 1.解压目录到d盘 mongodb 2.安装目录的下新建文件mongo.config文件 ##store data here dbpath=D:\mongodb\dat ...
- HybridApp开发准备工作——WebView
如大家所见,手机真是越来越离不开我们的日常生活了,像我,现在出门必带的是手机.移动电源.公交卡:钱包什么的再也没出过门.两年前,我还在Android的应用开发中当了一次过客.嗯,当时JAVA学得太糟糕 ...
- @JoinTable和@JoinColumn
默认情况下,JPA 持续性提供程序在映射多对多关联(或在单向的一对多关联中)的拥有方上的实体关联时使用一个连接表.连接表名称及其列名均在默认情况下指定,且 JPA 持续性提供程序假设:在关系的拥有方上 ...
- 获取WebView加载HTML时网页中的内容
main.xml如下: [html] view plaincopy <RelativeLayout xmlns:android="http://schemas.android.com/ ...
- [转]Android Volley完全解析(四),带你从源码的角度理解Volley
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/17656437 经过前三篇文章的学习,Volley的用法我们已经掌握的差不多了,但是 ...
- MySQL 可以用localhost 连接,但不能用IP连接的问题,局域网192.168.*.* 无法连接mysql
Mysql 默认是没有开启这个权限的(只允许使用 host:localhost,或者 host:127.0.0.1),如果想用 host:192.168.1.* ,来访问mysql ,需要手动开启这个 ...