题目描述:

给出一棵n个节点的树,每个点初始m值为1。

你有三种操作:

1.Add l r k ,将l到r路径上所有点m值加k。

2.Multi l r k ,将l到r路径上所有点m值乘k。

3.Query l r ,设x是l到r路径上的点,y是x的m值。假设有1~y共y个点,随机打乱,求形成错排的概率。

(k<=1000,n<=80000)

题解:

树剖正解?

(反正我用的dfs序+并查集)

首先1000^80000错排怎么搞啊?

要明白我们真正要的并不是错排数,而是错排数/阶乘。

打表后发现他是:

0.0,0.5,0.333333333,0.375,0.366666667,0.368055556,0.367857143,0.367881944,0.367879464

最后一位用的一般值。

(后来发现这个精度依然不够,要用错排递推直接打表。不然会卡精。)

树链怎么修改啊?

其实可以用单点修改……

重点在于一共只有80000个点,如果m值超过15就不用处理了,保留15位就够了……

所以单点最多修改80000*15次……

并查集维护。

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
//lgl AK in NOIp
#define N 80050
double num[],tmp[];
void init()
{
double now = ;
tmp[]=0.0,tmp[]=1.0;
num[]=0.5;
for(int i=;i<;i++)
{
now*=(double)i;
tmp[i]=(double)(i-)*(tmp[i-]+tmp[i-]);
num[i]=(double)tmp[i]/now;
}
}
int n,hed[N],cnt,q;
char ch[];
struct EG
{
int to,nxt;
}e[*N];
void ae(int ff,int t)
{
e[++cnt].to = t;
e[cnt].nxt = hed[ff];
hed[ff] = cnt;
}
struct aaafku
{
int s[*N];
void ins(int x,int d)
{
while(x<*N)
{
s[x]+=d;
x+=(x&(-x));
}
}
int qry(int x)
{
int ret = ;
while(x)
{
ret+=s[x];
x-=(x&(-x));
}
return ret;
}
}f[];
int las[N];
int dep[N],siz[N],fa[N],son[N],top[N],typ[N];
int tin[N],tout[N],tim;
void dfs1(int u,int ff)
{
tin[u]=++tim;
typ[u]=;
siz[u]=;
dep[u]=dep[ff]+;
las[u]=u;
f[].ins(tin[u],);
for(int j=hed[u];j;j=e[j].nxt)
{
int to = e[j].to;
if(to==ff)continue;
fa[to]=u;
dfs1(to,u);
siz[u]+=siz[to];
if(siz[to]>siz[son[u]])son[u]=to;
}
tout[u]=tim;
f[].ins(tout[u]+,-);
}
void dfs2(int u,int tp)
{
top[u] = tp;
if(!son[u])return ;
dfs2(son[u],tp);
for(int j=hed[u];j;j=e[j].nxt)
{
int to = e[j].to;
if(to!=fa[u]&&to!=son[u])
dfs2(to,to);
}
}
int get_lca(int x,int y)
{
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])swap(x,y);
x=fa[top[x]];
}
return dep[x]<dep[y]?x:y;
}
int get_las(int x)
{
if(las[x]==x)return x;
return las[x]=get_las(las[x]);
}
void deala(int u,int k,int lim)
{
if(dep[u]<=lim)return ;
f[typ[u]].ins(tin[u],-);
f[typ[u]].ins(tout[u]+,);
typ[u] = min(,typ[u]+k);
f[typ[u]].ins(tin[u],);
f[typ[u]].ins(tout[u]+,-);
deala(get_las(fa[u]),k,lim);
if(typ[u]==)las[u]=fa[u];
}
void dealb(int u,int k,int lim)
{
if(dep[u]<=lim)return ;
f[typ[u]].ins(tin[u],-);
f[typ[u]].ins(tout[u]+,);
typ[u] = min(,typ[u]*k);
f[typ[u]].ins(tin[u],);
f[typ[u]].ins(tout[u]+,-);
dealb(get_las(fa[u]),k,lim);
if(typ[u]==)las[u]=fa[u];
}
int main()
{
// freopen("yuyuko.in","r",stdin);
// freopen("yuyuko.out","w",stdout);
init();
scanf("%d",&n);
for(int ff,t,i=;i<n;i++)
{
scanf("%d%d",&ff,&t);
ae(ff,t),ae(t,ff);
}
dfs1(,),dfs2(,);
scanf("%d",&q);
for(int l,r,k,i=;i<=q;i++)
{
scanf("%s",ch+);
if(ch[]=='Q')
{
scanf("%d%d",&l,&r);
int lca = get_lca(l,r);
int ff = fa[lca];
double ans = 0.0;
for(int j=;j<;j++)
{
int sum = f[j].qry(tin[l])+f[j].qry(tin[r])-f[j].qry(tin[lca])-f[j].qry(tin[ff]);
ans+=(double)sum*num[j];
}
printf("%.1lf\n",ans);
}else
{
scanf("%d%d%d",&l,&r,&k);
int lca = get_lca(l,r);
if(ch[]=='A')
{
deala(las[l],k,dep[lca]);
deala(las[r],k,dep[lca]-);
}else
{
dealb(las[l],k,dep[lca]);
dealb(las[r],k,dep[lca]-);
}
}
}
return ;
}

NOIp模拟赛 西行妖下的更多相关文章

  1. NOIP模拟赛20161022

    NOIP模拟赛2016-10-22 题目名 东风谷早苗 西行寺幽幽子 琪露诺 上白泽慧音 源文件 robot.cpp/c/pas spring.cpp/c/pas iceroad.cpp/c/pas ...

  2. NOIP模拟赛 by hzwer

    2015年10月04日NOIP模拟赛 by hzwer    (这是小奇=> 小奇挖矿2(mining) [题目背景] 小奇飞船的钻头开启了无限耐久+精准采集模式!这次它要将原矿运到泛光之源的矿 ...

  3. 大家AK杯 灰天飞雁NOIP模拟赛题解/数据/标程

    数据 http://files.cnblogs.com/htfy/data.zip 简要题解 桌球碰撞 纯模拟,注意一开始就在袋口和v=0的情况.v和坐标可以是小数.为保险起见最好用extended/ ...

  4. 【noip模拟赛5】细菌 状压dp

    [noip模拟赛5]细菌   描述 近期,农场出现了D(1<=D<=15)种细菌.John要从他的 N(1<=N<=1,000)头奶牛中尽可能多地选些产奶.但是如果选中的奶牛携 ...

  5. NOI.AC NOIP模拟赛 第六场 游记

    NOI.AC NOIP模拟赛 第六场 游记 queen 题目大意: 在一个\(n\times n(n\le10^5)\)的棋盘上,放有\(m(m\le10^5)\)个皇后,其中每一个皇后都可以向上.下 ...

  6. NOI.AC NOIP模拟赛 第四场 补记

    NOI.AC NOIP模拟赛 第四场 补记 子图 题目大意: 一张\(n(n\le5\times10^5)\)个点,\(m(m\le5\times10^5)\)条边的无向图.删去第\(i\)条边需要\ ...

  7. NOI.AC NOIP模拟赛 第三场 补记

    NOI.AC NOIP模拟赛 第三场 补记 列队 题目大意: 给定一个\(n\times m(n,m\le1000)\)的矩阵,每个格子上有一个数\(w_{i,j}\).保证\(w_{i,j}\)互不 ...

  8. 【BZOJ 2957】楼房重建&&Codechef COT5 Count on a Treap&&【NOIP模拟赛】Weed 线段树的分治维护

    线段树是一种作用于静态区间上的数据结构,可以高效查询连续区间和单点,类似于一种静态的分治.他最迷人的地方在于“lazy标记”,对于lazy标记一般随我们从父区间进入子区间而下传,最终给到叶子节点,但还 ...

  9. 2016-06-19 NOIP模拟赛

          2016-06-19 NOIP模拟赛 by coolyangzc 共3道题目,时间3小时 题目名 高级打字机 不等数列 经营与开发 源文件 type.cpp/c/pas num.cpp/c ...

随机推荐

  1. Pascal输出星星

    program Project2; {$APPTYPE CONSOLE} uses SysUtils; var i,j:integer; begin { TODO -oUser -cConsole M ...

  2. centos配置nodejs和mysql

    我使用的是centos7.2 64位,弄了一大晚上试了各种方法,安装的nodejs就是启动不了服务器.全是IP能ping通,浏览器不能访问.端口都是打开了的.安全组也设置了,就是不行.最后阿里云客服电 ...

  3. SAS基础 -- SAS编程入门

    SAS语言 -- 简介   SAS语言是一种专用的数据管理与分析语言,它提供了一种完善的编程语言.类似于计算机的高级语言,SAS用户只需要熟悉其命令.语句及简单的语法规则就可以做数据管理和分析处理工作 ...

  4. SGU 176 Flow construction【有上下界最小流】

    正好考到了所以翻一些题来做--猛然发现搞了半个月的网络流却没做两道上下界(不过这种题好像是比较少233) 首先建立超级源汇ss,tt,没限制的边照常连,对于有限制的边(u,v,mn,mx),连接(u, ...

  5. git上拉取tag,识别最新tag在此版本上新增tag

    通过shell 脚本自动获取最新tag,并输入最新版本后,推到git上 # 拉取分支上现有的tags git fetch --tags echo -e "所有tag列表" git ...

  6. 51nod 1116 K进制下的大数

    你万万想不到,Long Long 就能存下的数据 #include <iostream> #include <cstdio> #include <cstdlib> ...

  7. TCP/IP网络协议基础

    实验楼学习网络协议传送门 一.TCP/IP简介 TCP/IP(Transmission Control Protocol/Internet Protocol)是传输控制协议和网络协议的简称,它定义了电 ...

  8. (五)Python面向对象编程

    根据廖雪峰老师的python教程写一些学习总结! 面向对象编程 面向对象编程--Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元, ...

  9. 题解报告:hdu 5750 Dertouzos(最大真约数、最小素因子)

    Problem Description A positive proper divisor is a positive divisor of a number n, excluding n itsel ...

  10. synchronized(1)用法简介:修饰方法,修饰语句块

    注意: 同一个对象或方法在不同线程中才出现同步问题,不同对象在不同线程互相不干扰. synchronized方法有2种用法:修饰方法,修饰语句块 1.synchronized方法 是某个对象实例内,s ...