首页 > 其他分享 >NOIP2024/7/25模拟赛

NOIP2024/7/25模拟赛

时间:2024-07-25 19:20:10浏览次数:7  
标签:25 20 16 int siz son fa NOIP2024 模拟

T4

题意:

答案对 \(2^{16}\) 取模。

分析:

根节点 \(1\) 选到 \(1\) 的概率为 \(\frac{1}{n}\),然后随便把剩下的 \(n-1\) 分配给它的所有子树,记 \(1\) 的其中一个儿子为 \(y\),那么 \(y\) 选到它所被分配到的数中最小值的概率为 \(\frac{1}{siz_{y}}\),然后 \(y\) 再继续分配给它的子树,以此类推。因此祈求一次成功的概率为 \(\frac{1}{\prod siz_{i}}\),期望操作次数即为 \(\prod siz_{i}\)。

由于只会操作 \(n\) 次,因此最后 \(siz_{i}>1\) 的点只会有 \(n\) 个,所以可以离线先把这棵树建出来,初始权值 \(S_{i}=1\)。需要支持两个操作:将一条到根上的链的 \(siz\) 全部加上某个数;计算所有 \(S_{i}\) 的乘积。

把每个点看成一个 \(x+S_{i}\) 的多项式,最后答案即为 \((x+S_{1})(x+S_{2})(x+S_{3})...\) 中 \(x_{0}\) 的系数,拆开后记为 \(a_{0}x^0+a_{1}x^1+a_{2}x^2+a_{3}x^3+...\),此时要对每一项全部加 \(k\),即 \((x+S_{1}+k)(x+S_{2}+k)(x+S_{3}+k)...\),不难发现其恰好等于 \(a_{0}(x+k)^0+a_{1}(x+k)^1+a_{2}(x+k)^2+a_{3}(x+k)^3+...\)。

可以证明只需要维护 \(a_{0},a_{1},\dots,a_{15}\),其他项不会对 \(x^{0}\) 产生影响。为什么呢?设有一项为 \(a_{n}(x+k)^{n}\),其中 \(n \ge 16\)。拆开后的多项式记为 \(b_{0}k^{n}x^{0}+b_{1}k^{n-1}x^1+...b_{n}k^{0}x^{n}\),这一次操作一定不会对 \(x^{0}\) 产生影响,因为 \(k^{n} \mod 2^{16} =0\)。假设它新产生了一项 \(b_{i}k^{n-i}x^{i}\),记下一次操作为 \(k_{2}\),假设该项又产生了一项 \(c_{j}k^{n-i}k_{2}^{i-j}x^{j}\),不难观察出它所产生的所有项 \(x^{j}\) 的系数一定是 \(2^{n-j}\) 倍数,当 \(j=0\) 时,一定是 \(2^{n}\) 的倍数。所以不需要管 \(x^{n} \ (n \ge 16)\) 的系数。

因此可以使用线段树+树剖,线段树每个节点分别维护 \(a_{0},a_{1},\dots,a_{15}\),修改时直接二项式定理拆开,懒标记可以直接合并。因此时间复杂度 \(O(n \log^{2} V \log^{2} n)\)。

代码:

#include<bits/stdc++.h>
#define int unsigned long long
#define N 100005
using namespace std;


#define getchar() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 1 << 21, stdin), p1 == p2) ? EOF : *p1++)
char buf[1 << 23], *p1 = buf, *p2 = buf, ubuf[1 << 23], *u = ubuf;

int read() {
    int p = 0, flg = 1;
    char c = getchar();
    while (c < '0' || c > '9') {
        if (c == '-') flg = -1;
        c = getchar();
    }
    while (c >= '0' && c <= '9') {
        p = p * 10 + c - '0';
        c = getchar();
    }
    return p * flg;
}

void print(int x) {
	if(x < 0) putchar('-'), x = -x;
	if(x > 9) print(x / 10);
	putchar('0' + x % 10);
}

int Q, n, cnt, Num; 
int X[N], K[N], C[20][20];

struct tp {
	int fa, v;
	friend bool operator < (tp x, tp y) {
		if(x.v != y.v) return x.v < y.v;
		else return x.fa < y.fa; 
	}
};
vector<tp>s; 
map<int, int>z;
vector<int>p[N];
int fa[N];

void init() {
	for(int i = 0; i <= 15; i++) C[i][0] = 1;
	for(int i = 1; i <= 15; i++)
	for(int j = 1; j <= i; j++) C[i][j] = (C[i - 1][j] + C[i - 1][j - 1]);
	Q = read();
	cnt = 1;
	for(int i = 1; i <= Q; i++) {
		X[i] = read(), K[i] = read();
		if(!z[X[i]]) z[X[i]] = ++Num;
		if(X[i] != 1 && !fa[z[X[i]]]) { //确定它的父亲 
			int h = (*(--upper_bound(s.begin(), s.end(), ((tp){1000000000, X[i]})))).fa;
			p[h].push_back(z[X[i]]);
			fa[z[X[i]]] = h;
		}
		s.push_back((tp){z[X[i]], cnt + 1}); cnt += K[i];
	}
	n = Num;
	/*for(int i = 1; i <= n; i++)
	for(auto y : p[i]) cout << i << " -> " << y << endl;*/
}

int siz[N], dfn[N], top[N], son[N], tot;
void dfs1(int x) {
	siz[x] = 1; int Maxson = 0;
	for(auto y : p[x]) {
		dfs1(y);
		siz[x] += siz[y];
		if(siz[y] > Maxson) {
			Maxson = siz[y];
			son[x] = y;
		}
	}
}
void dfs2(int x, int topf) {
	top[x] = topf; dfn[x] = ++tot;
	if(son[x]) dfs2(son[x], topf);
	for(auto y : p[x]) {
		if(top[y]) continue;
		dfs2(y, y);
	}
}

struct node {
	int a[18], tag;
}t[N * 4];

int B[20], h[20];
void pushup(int u) {
	for(int i = 0; i <= 15; i++) t[u].a[i] = 0;
	for(int i = 0; i <= 15; i++)
	for(int j = 0; j <= 15 - i; j++) t[u].a[i + j] = (t[u].a[i + j] + t[u * 2].a[i] * t[u * 2 + 1].a[j]);
}
void build(int u, int L, int R) {
	if(L == R) {
		t[u].a[0] = t[u].a[1] = 1;
		return;
	}
	int mid = (L + R) / 2;
	build(u * 2, L, mid);
	build(u * 2 + 1, mid + 1, R);
	pushup(u);
}
void maketag(int u, int k) {
	t[u].tag += k;
	for(int i = 0; i <= 15; i++) B[i] = 0;
	h[0] = 1;
	for(int i = 1; i <= 15; i++) h[i] = h[i - 1] * k;
	for(int n = 0; n <= 15; n++)
	for(int i = 0; i <= n; i++) B[i] = (B[i] + C[n][i] * t[u].a[n] * h[n - i]);
	for(int i = 0; i <= 15; i++) t[u].a[i] = B[i];
}
void pushdown(int u) {
	if(!t[u].tag) return;
	maketag(u << 1, t[u].tag);
	maketag(u << 1 | 1, t[u].tag);
	t[u].tag = 0;
}
void update(int u, int L, int R, int l, int r, int k) {
	if(R < l || r < L) return;
	if(l <= L && R <= r) {
		//for(int i = 0; i <= 15; i++) cout << t[u].a[i] << " ";
		//cout << endl;
		//cout << t[u].a[0] << " " << t[u].a[1] << " " << k << endl;
		maketag(u, k);
		//cout << t[u].a[0] << endl;
		return;
	}
	int mid = (L + R) / 2;
	pushdown(u);
	update(u << 1, L, mid, l, r, k);
	update(u << 1 | 1, mid + 1, R, l, r, k);
	pushup(u);
}
void add(int u, int k) {
	while(u) {
		//cout << "upd : " << dfn[top[u]] << " " << dfn[u] << endl;
		update(1, 1, n, dfn[top[u]], dfn[u], k);
		u = fa[top[u]]; 
	}
}
void Sol() {
	build(1, 1, n);
	for(int i = 1; i <= Q; i++) {
		add(z[X[i]], K[i]);
		print(t[1].a[0] & 65535);
		printf("\n");
	}
}

signed main() {
	//freopen("counting6.in", "r", stdin);
	//freopen("counting.out", "w", stdout);
	ios::sync_with_stdio(0);
	cin.tie(0); cout.tie(0);
	init();
	dfs1(1);
	dfs2(1, 1);
	Sol();
	return 0;
}
/*
1
1 2
*/

标签:25,20,16,int,siz,son,fa,NOIP2024,模拟
From: https://www.cnblogs.com/xcs123/p/18323971

相关文章

  • 2024.7.25(Git、gitlab以及分支管理)
    分布式版本控制系统一、Git概述Git是一种分布式版本控制系统,用于跟踪和管理代码的变更。它是由LinusTorvalds创建的,最初被设计用于Linux内核的开发。Git允许开发人员跟踪和管理代码的版本,并且可以在不同的开发人员之间进行协作。Github用的就是Git系统来管理它们的网......
  • 7.25第二周周四学习总结
    算法竞赛(差分)(上午)初始化#include<algorithm>intarr[100];std::fill(arr,arr+100,0);//比memset更高效intarr[100]={};//所有元素都初始化为0栈溢出为局部变量每次运行时都在运行栈中分配,如果数组很大,结果会造成运行栈溢出,自然就运行不了另外,使用全局变......
  • zyx青岛实训day14 7/25
    Git一种分布式版本控制系统,用于跟踪和管理代码的变更一.Git的主要功能:二.准备git机器修改静态ip,主机名三.git仓库的建立:1.安装git[root@git~]#yum-yinstallgit2.创建一个目录----用来放置git文件[root@git~]#mkdir/yy0003.使用git指令,一定要cd到初始化之后的目......
  • Sqli-labs-master的21—25通关教程
    目录Less-21('闭合)查询数据库名查询数据库中的表查询表中字段名查询表中数据Less-22("闭合)查询数据库名查询数据库中的表查询表中字段名查询表中数据Less-23查询数据库名查询数据库中的表查询表中字段名查询表中数据Less-24(二次注入)Less-25查询数据库名......
  • 模拟建造游戏:城市:天际线2(都市天际线2)中文免安装,解压即撸
    《城市:天际线2》(Cities:SkylinesII)是一款模拟经营游戏,由ColossalOrder开发,ParadoxInteractive发行。下载地址:https://pan.quark.cn/s/84e69332ec3e更多游戏:https://kdocs.cn/l/cuHMLqjlrCK7深度模拟体验:《城市:天际线2》提供深度模拟体验和生动运转的经济系统,考验玩......
  • SPONGE常用教程:蛋白+配体模拟2
    前序课程目录应用场景简述;-[Done]DSDP:蛋白-配体对接;-[Done]XPONGE:蛋白-配体建模,加溶剂;-[Done]SPONGE:能量极小化-NVT-NPT-正式模拟;XPONGE:数据简单后处理。4.SPONGE:能量极小化-NVT-NPT-正式模拟文件下载进入建模文件目录,如下图:SPONGE输入文件采用独特的参......
  • 暑假集训csp提高模拟7
    赛时rank42,T180,T213,T30,T420在T4挂死了,赛时胡了一个挺没有前途的\(O(n\log_2^3n)\)的做法,结果赛后发现假了,没有时间打T2,T3的暴力了。糖T1Permutations&PrimesPermutations&Primes赛时有一个特判\(n=3\)错了,挂了20pts结论非常显然,1放中间,2、3各放一边,剩下的随便......
  • 20240724模拟赛订正题笔记
    (T1)lnsyoj2208逆流而上/P10737[SEERC2020]ReverseGame考虑到失败时字符串应为前面都是0,后面都是1(例如"0000001111111")所以可以将原串的逆序对数求出,记为m,对于每个可翻转的串进行分类讨论:1."10"->"01"可以将原串的逆序对减1。2."100"->"001""110"->"011......
  • (三)复习第三课(07.20- 07.25第二轮):HTML标签元素练习大全
    <!DOCTYPEhtml><!--练习时间:2024.07.20-2024.07.25--><htmllang="en"><!--添加了en可以让你的网站打开时会提示翻译--><head> <pid="head1"></p><metacharset="utf-8"><!--对于中文网页需要使用此标签声明编码,否则会出现......
  • 暑假集训CSP提高模拟7
    暑假集训CSP提高模拟7组题人:@KafuuChinocpp|@Chen_jr\(T1\)P122.Permutations&Primes\(20pts\)原题:CF1844BPermutations&Primes假的构造策略,拿到了\(20pts\)。若\(n\)为奇数,则将\(1\)放在\(\left\lceil\frac{n}{2}\right\rceil\)的位置上,前一半......