题意:

给一棵\(n(1 \leq n \leq 10^5)\)个节点的二叉树,除叶子节点外,每个点都有左儿子和右儿子。

每个点上都有一个权值。

游戏规则是这样的:在根节点放一个权值为\(X\)的小球,假设当前节点的权值是\(w_i\)

  • 如果\(X=w_i\),小球就停在这个节点。
  • 如果\(X<w_i\),小球等概率地往左右两个儿子走下去。
  • 如果\(X>w_i\),小球以\(\frac{1}{8}\)的概率走到左儿子,以\(\frac{7}{8}\)的概率走到右儿子。

下面有若干次询问\(v \, X\),问从根节点放一个权值为\(X\)的小球走到节点\(v\)的概率是多少。

分析:

构造一棵主席树,维护父亲权值在区间\([L,R]\)中左儿子和右儿子的个数。

首先判断一下概率为\(0\)的情况,如果找到父亲权值等于小球权值\(X\)的点,那么概率为\(0\)。

否则就统计一下父亲权值小于\(X\)的左儿子个数\(lcnt\),右儿子个数\(rcnt\),以及所有的儿子个数\(sons\)。

所求的概率为:\(p=\frac{1^{lcnt} \cdot 7^{rcnt} \cdot 4^{sons-lcnt-rcnt}}{8}\)

解得\(x=rcnt, \, y=sons+lcnt+rcnt\)

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int maxn = 100000 + 10;
const int maxnode = maxn << 5; struct Node
{
int lcnt, rcnt;
Node(int l = 0, int r = 0): lcnt(l), rcnt(r) {}
Node operator + (const Node& t) const {
return Node(lcnt + t.lcnt, rcnt + t.rcnt);
}
}; int sz;
Node T[maxnode];
int lch[maxnode], rch[maxnode];
int root[maxn]; int update(int pre, int L, int R, int pos, Node t) {
int rt = ++sz;
T[rt] = T[pre] + t;
lch[rt] = lch[pre];
rch[rt] = rch[pre];
if(L < R) {
int M = (L + R) / 2;
if(pos <= M) lch[rt] = update(lch[pre], L, M, pos, t);
else rch[rt] = update(rch[pre], M+1, R, pos, t);
}
return rt;
} int n, m, Q;
int a[maxn], b[maxn * 2], tot;
int v[maxn], x[maxn];
int ch[maxn][2];
int dep[maxn]; void dfs(int u) {
if(!ch[u][0]) return;
root[ch[u][0]] = update(root[u], 1, tot, a[u], Node(1, 0));
root[ch[u][1]] = update(root[u], 1, tot, a[u], Node(0, 1));
dep[ch[u][0]] = dep[ch[u][1]] = dep[u] + 1;
dfs(ch[u][0]);
dfs(ch[u][1]);
} bool queryequal(int rt, int L, int R, int pos) {
if(L == R) { return T[rt].lcnt + T[rt].rcnt != 0; }
int M = (L + R) / 2;
if(pos <= M) return queryequal(lch[rt], L, M, pos);
else return queryequal(rch[rt], M+1, R, pos);
} Node queryless(int rt, int L, int R, int pos) {
if(R <= pos) return T[rt];
int M = (L + R) / 2;
if(pos <= M) return queryless(lch[rt], L, M, pos);
else return T[lch[rt]] + queryless(rch[rt], M+1, R, pos);
} int main()
{
int _; scanf("%d", &_);
while(_--) {
int n; scanf("%d", &n);
for(int i = 1; i <= n; i++) {
scanf("%d", a + i);
b[i - 1] = a[i];
}
scanf("%d", &m);
memset(ch, 0, sizeof(ch));
while(m--) {
int u; scanf("%d", &u);
scanf("%d%d", &ch[u][0], &ch[u][1]);
}
scanf("%d", &Q);
tot = n;
for(int i = 1; i <= Q; i++) {
scanf("%d%d", v + i, x + i);
b[tot++] = x[i];
} sort(b, b + tot);
tot = unique(b, b + tot) - b;
for(int i = 1; i <= n; i++)
a[i] = lower_bound(b, b + tot, a[i]) - b + 1;
for(int i = 1; i <= Q; i++)
x[i] = lower_bound(b, b + tot, x[i]) - b + 1; sz = 0;
dfs(1); for(int i = 1; i <= Q; i++) {
if(queryequal(root[v[i]], 1, tot, x[i])) {
printf("0\n"); continue;
}
Node ans;
if(x[i] > 1) ans = queryless(root[v[i]], 1, tot, x[i] - 1);
int sons = dep[v[i]];
int ans2 = sons * 3;
ans2 -= (sons - ans.lcnt - ans.rcnt) * 2;
int ans7 = ans.rcnt;
printf("%d %d\n", ans7, ans2);
}
} return 0;
}

HDU 4605 Magic Ball Game 主席树的更多相关文章

  1. HDU 4605 Magic Ball Game (在线主席树|| 离线 线段树)

    转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlove 题意:给出一棵二叉树,每个结点孩子数目为0或者2. ...

  2. hdu 4605 Magic Ball Game (在线主席树/离线树状数组)

    版权声明:本文为博主原创文章,未经博主允许不得转载. hdu 4605 题意: 有一颗树,根节点为1,每一个节点要么有两个子节点,要么没有,每个节点都有一个权值wi .然后,有一个球,附带值x . 球 ...

  3. HDU 4605 Magic Ball Game(可持续化线段树,树状数组,离散化)

    Magic Ball Game Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

  4. hdu 4605 Magic Ball Game

    http://acm.hdu.edu.cn/showproblem.php?pid=4605 可以离线求解 把所以可能出现的 magic ball  放在一个数组里(去重),从小到大排列 先不考虑特殊 ...

  5. HDU 4605 Magic Ball Game (dfs+离线树状数组)

    题意:给你一颗有根树,它的孩子要么只有两个,要么没有,且每个点都有一个权值w. 接着给你一个权值为x的球,它从更节点开始向下掉,有三种情况 x=w[now]:停在此点 x<w[now]:当有孩子 ...

  6. HDU 4605 Magic Ball Game 树状数组

    题目大意很简单. 有一颗树(10^5结点),所有结点要么没有子结点,要么有两个子结点.然后每个结点都有一个重量值,根结点是1 然后有一个球,从结点1开始往子孙结点走. 每碰到一个结点,有三种情况 如果 ...

  7. HDU 4605 Magic Ball Game(离线算法)

    题目链接 思路就很难想+代码实现也很麻烦,知道算法后,已经写的很繁琐而且花了很长时间,200+,好久没写过这么长的代码了. #pragma comment(linker, "/STACK:1 ...

  8. HDU 4602 Magic Ball Game(离线处理,树状数组,dfs)

    Magic Ball Game Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

  9. HDU 4417 Super Mario(主席树求区间内的区间查询+离散化)

    Super Mario Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tota ...

随机推荐

  1. StringMVC

    public class FirstController implements Controller { public ModelAndView handleRequest(HttpServletRe ...

  2. javascript中两种基本常用排序算法分析

    备注:内容大部分从网上复制,代码为自己手写.仅做知识的温故知新,并非原创. 1.冒泡排序(Bubble Sort) (1)算法描述 冒泡排序是一种简单的排序算法.它重复地走访过要排序的数列,一次比较两 ...

  3. Typedef 用法

    typedef声明有助于创建平台无关类型,甚至能隐藏复杂和难以理解的语法. 不管怎样,使用 typedef 能为代码带来意想不到的好处,通过本文你可以学习用typedef避免缺欠,从而使代码更健壮.  ...

  4. bug {was not declared in this scope}

    使用自己定义的结构体作为返回值的时候,出现了 ...was not declared in this scope 检查了各种头文件,把缓存也都删掉了还是不行. 结果,发现,应该这样用vector< ...

  5. SQL Server数据库log shipping 灾备(Part2 )

    3.配置步骤: 主服务器(A机)设置 (1) 启用Log Shipping Configuration 右键单击需要配置日志传输的数据库->Tasks-> Ship Transaction ...

  6. POJ 1067 取石子游戏 (威佐夫博奕,公式)

    题意: 有两堆石子,两个人轮流取石子.规定每次有两种取法,一是可以在任意的一堆中取走任意多的石子:二是可以在两堆中同时取走相同数量的石子.最后把石子全部取完者为胜者.给定两堆石子数量,问先手的输赢? ...

  7. [VC]listctrl的基本用法

    1   添加listctrl的头 m_list.setextendedstyle(LVS_EX_FULLROWSELECT||LVS_EX_GRIdLINES); m_list.insertcolum ...

  8. 2012-2013 ACM-ICPC, NEERC, Central Subregional Contest H Milestones1 (暴力)

    预处理+暴力,每个颜色都是独立的,求个前缀和,减一减判断一个在区间内颜色是否存在. 算了算复杂度好像有点勉强,但是还是过了,学了主席树以后用主席树在做一下 #include<bits/stdc+ ...

  9. vue validate多表单验证思考 之前写过一个里外层,现在觉得不合适,应该平行的写,然后都给ret,最后判断ret 再做出反应,这样整体表单的所有验证就都报验证,然后最后提交的时候把组件内的对象合并到总的对象,再提交

    vue validate多表单验证思考 之前写过一个里外层,现在觉得不合适,应该平行的写,然后都给ret,最后判断ret 再做出反应,这样整体表单的所有验证就都报验证,然后最后提交的时候把组件内的对象 ...

  10. 得到本地应用程序的EXE的路径

    实现效果: 知识运用: Application类的ExecutablePath属性 //获取启动了应用程序的可执行文件的路径和可执行文件的名称    public static string Exec ...