hdu 4605 Magic Ball Game
http://acm.hdu.edu.cn/showproblem.php?pid=4605
可以离线求解
把所以可能出现的 magic ball 放在一个数组里(去重),从小到大排列
先不考虑特殊情况,对二叉树进行dfs 搜索的过程中需要维护各个magic ball到当前节点的概率
维护:根据当前节点大小 和要去左子树还是右子树的情况,可以得到magic数组中哪个段的x和y需要同时加上多少
可以用线段树维护
特殊情况:
1,根节点一定可以到达
2,到达不了的情况需要标记
代码:
#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<cmath>
#include<set>
#include<map>
#include<stack>
#include<vector>
#include<algorithm>
#include<queue>
#include<stdexcept>
#include<bitset>
#include<cassert>
#include<deque>
#include<numeric> #pragma comment(linker, "/STACK:1024000000,1024000000") using namespace std; typedef long long ll;
typedef unsigned int uint;
typedef pair<int,int> pp;
const double eps=1e-12;
const int INF=0x3f3f3f3f;
const ll MOD=1000000007;
const int N=100010;
struct node
{
int l,r;
int fm,fn;
}tree[N*4];
int f[N];
int a[N];
int stop[N];
int qv[N],qx[N],qm[N],qn[N];
int bl[N],br[N];
int weight[N];
vector<int>vt[N];
void build(int x,int l,int r)
{
tree[x].l=l;
tree[x].r=r;
tree[x].fm=0;
tree[x].fn=0;
if(l==r)
return ;
int mid=(l+r)>>1;
build((x<<1),l,mid);
build((x<<1)|1,mid+1,r);
}
void add(int x,int l,int r,int km,int kn)
{
if(l>r) return ;
if(tree[x].l==l&&tree[x].r==r)
{
tree[x].fm+=km;
tree[x].fn+=kn;
return ;
}
int mid=(tree[x].l+tree[x].r)>>1;
if(r<=mid)
add((x<<1),l,r,km,kn);
else if(l>mid)
add((x<<1)|1,l,r,km,kn);
else
{
add((x<<1),l,mid,km,kn);
add((x<<1)|1,mid+1,r,km,kn);
}
}
void get(int x,int k,int &m,int &n)
{
m+=tree[x].fm;n+=tree[x].fn;
if(tree[x].l==tree[x].r)
{return ;}
int mid=(tree[x].l+tree[x].r)>>1;
if(k<=mid)
get((x<<1),k,m,n);
else
get((x<<1)|1,k,m,n);
}
int bse(int l,int r,int k)
{
while(l<=r)
{
int m=(l+r)>>1;
if(a[m]<=k)
l=m+1;
else
r=m-1;
}
return r;
}
void dfs(int x,int ln)
{
for(unsigned int i=0;i<vt[x].size();++i)
{
int t=vt[x][i];
int w=bse(1,ln,qx[t]);
int m=0,n=0;
if(!stop[w])
{get(1,w,m,n);}
qm[t]=m;
qn[t]=n;
}
if(bl[x]!=0&&br[x]!=0)
{
int l=bse(1,ln,weight[x]);
int ll=l,rr=l+1;
if(a[l]==weight[x])
stop[l]++;
add(1,1,ll,0,1);
add(1,rr,ln,0,3);
dfs(bl[x],ln);
add(1,1,ll,0,-1);
add(1,rr,ln,0,-3); add(1,1,ll,0,1);
add(1,rr,ln,1,3);
dfs(br[x],ln);
add(1,1,ll,0,-1);
add(1,rr,ln,-1,-3);
if(a[l]==weight[x])
stop[l]--;
}
}
int main()
{
//freopen("data.in","r",stdin);
//freopen("1006.in","r",stdin);
//freopen("my.out","w",stdout);
int T;
scanf("%d",&T);
while(T--)
{
for(int i=0;i<N;++i)
vt[i].clear();
int n;
scanf("%d",&n);
for(int i=1;i<=n;++i)
scanf("%d",&weight[i]);
int m;
scanf("%d",&m);
memset(bl,0,sizeof(bl));
memset(br,0,sizeof(br));
while(m--)
{
int v,l,r;
scanf("%d %d %d",&v,&l,&r);
f[l]=v;
f[r]=v;
bl[v]=l;
br[v]=r;
}
int ln=0;
a[++ln]=0;
a[++ln]=1e9+10;
int Q;
scanf("%d",&Q);
for(int i=0;i<Q;++i)
{
scanf("%d %d",&qv[i],&qx[i]);
a[++ln]=qx[i];
if(qv[i]!=1)
{
vt[qv[i]].push_back(i);
}
}
sort(a+1,a+ln+1);
int l=0;
for(int i=1;i<=ln;++i)
if(i==1||a[i]!=a[i-1])
a[++l]=a[i];
ln=l; build(1,1,ln);
memset(stop,0,sizeof(stop));
dfs(1,ln); for(int i=0;i<Q;++i)
{
if(qv[i]==1)
{printf("0 0\n");continue;} if(qm[i]==0&&qn[i]==0)
printf("0\n");
else
printf("%d %d\n",qm[i],qn[i]);
}
}
return 0;
}
hdu 4605 Magic Ball Game的更多相关文章
- HDU 4605 Magic Ball Game(可持续化线段树,树状数组,离散化)
Magic Ball Game Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) ...
- hdu 4605 Magic Ball Game (在线主席树/离线树状数组)
版权声明:本文为博主原创文章,未经博主允许不得转载. hdu 4605 题意: 有一颗树,根节点为1,每一个节点要么有两个子节点,要么没有,每个节点都有一个权值wi .然后,有一个球,附带值x . 球 ...
- HDU 4605 Magic Ball Game (在线主席树|| 离线 线段树)
转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents by---cxlove 题意:给出一棵二叉树,每个结点孩子数目为0或者2. ...
- HDU 4605 Magic Ball Game (dfs+离线树状数组)
题意:给你一颗有根树,它的孩子要么只有两个,要么没有,且每个点都有一个权值w. 接着给你一个权值为x的球,它从更节点开始向下掉,有三种情况 x=w[now]:停在此点 x<w[now]:当有孩子 ...
- HDU 4605 Magic Ball Game(离线算法)
题目链接 思路就很难想+代码实现也很麻烦,知道算法后,已经写的很繁琐而且花了很长时间,200+,好久没写过这么长的代码了. #pragma comment(linker, "/STACK:1 ...
- HDU 4605 Magic Ball Game 树状数组
题目大意很简单. 有一颗树(10^5结点),所有结点要么没有子结点,要么有两个子结点.然后每个结点都有一个重量值,根结点是1 然后有一个球,从结点1开始往子孙结点走. 每碰到一个结点,有三种情况 如果 ...
- HDU 4605 Magic Ball Game 主席树
题意: 给一棵\(n(1 \leq n \leq 10^5)\)个节点的二叉树,除叶子节点外,每个点都有左儿子和右儿子. 每个点上都有一个权值. 游戏规则是这样的:在根节点放一个权值为\(X\)的小球 ...
- HDU 4602 Magic Ball Game(离线处理,树状数组,dfs)
Magic Ball Game Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) ...
- 【HDOJ】4605 Magic Ball Game
思路1:树状数组+离线处理,对所有的w离散化处理,边dfs边使用树状数组更新左右w的情况.思路2:主席树,边bfs边建树.结点信息存储cnt,然后在线查询.树状数组. /* 4605 */ #incl ...
随机推荐
- poj3304Segments(直线与多条线段相交)
链接 枚举两点(端点),循环遍历与直线相交的线段. #include <iostream> #include<cstdio> #include<cstring> # ...
- js中event.keyCode用法及keyCode对照表
HTML 用户名:<input type="text" id="UserAccount" onKeyPress="JumpByEnter(Use ...
- iOS开发之Xcode 6更新默认不支持armv7s架构
最近一次的Xcode 6更新默认不再支持arm7s架构,究竟是要废除不用呢还是仅仅只是一个疏忽? 目前的Xcode 6配置里定义${ARCHS_STANDARD}为armv7, arm64,当然这个定 ...
- Javascript模块化编程(一):模块的写法 (转载 学习中。。。。)
转载地址:http://www.ruanyifeng.com/blog/2012/10/javascript_module.html 阮一峰 大神:http://www.ruanyifeng.com/ ...
- double 逆序
请设计一个函数,不许用到字符串函数,用数学运算,将double类型数据转换,例如123.456转换成654.321 int _tmain(int argc, _TCHAR* argv[]) { con ...
- C#_DataTable导出Execl为自定义标题
public bool ExportExcel(DataTable tb, string path, string tbName) { //excel 2003格式 string connString ...
- hdu_3555 bomb
数位动态规划 数位动态规划是求解一个大区间[L, R]中间满足条件Q的所有数字的个数(或者和,或其他)的一种方法.它通过分析每一位上的数字,一般用 dp[len][digit][...] 来表 ...
- 用NAN简化Google V8 JS引擎的扩展
通过C++扩展Google V8 JS引擎的文章很多,Google V8 JS带的例子也容易明白.但是大部分文章都是Hello World型的,真正使用时发现处处是坑.扩展V8最经典的例子就是node ...
- 一个最简html5文档来说明html5的新特性和写法
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="utf-8&quo ...
- Web网页性能管理详解
你遇到过性能很差的网页吗? 这种网页响应非常缓慢,占用大量的 CPU 和内存,浏览起来常常有卡顿,页面的动画效果也不流畅. 你会有什么反应?我猜想,大多数用户会关闭这个页面,改为访问其他网站.作为一个 ...