首页 > 其他分享 >题解:CSP-S2020] 函数调用

题解:CSP-S2020] 函数调用

时间:2024-04-19 22:24:06浏览次数:24  
标签:int 题解 函数调用 rd 1ll mul nw CSP mod

题解:CSP-S2020] 函数调用

  • 一句话题意:给定一个有初始值的序列,支持如下三种操作:

    • 1、单点加
    • 2、全局乘
    • 3、递归某些操作1、2、3
  • 求最终的序列。

  • 标签:topsort,动态规划,转化贡献统计(集中贡献),主客翻转

  • 关于topsort:部分分里的树结构基本上直接暗示了正解要使用topsort,而且本来函数关系就会组成一张DAG。

  • 关于贡献转化

    • 加、乘运算的独立比较容易想到:即某个元素的终值=初值*全局乘因子+扩倍后每个加法因子
      • 简单来说,

\[((a+b)\times c+d)\times e=a \times c \times d+b \times c \times e +d \times e \]

  • 感觉难点在于想到贡献的统计要全部转化到操作1上

  • 如果针对每个操作统计贡献,那意味着每个操作上要挂原序列中的 \(n\) 个数,显然不管是空间还是时间,这都是无法承受的。而我们可以通过动态规划的方式将所有贡献转到操作1上。(搞了半天精髓还是在动规上!!!)

    • 全局乘因子可以一遍倒序topsort求得
    • 扩倍后每个加法因子可以一遍正序top求得,思想见这篇博客
      • 简单来说就是同父亲的儿子的乘法因子受到更靠右的儿子的影响,链式前向星的遍历顺序天然地满足我们从右到左遍历儿子的需求。
    • 最后合并贡献

    a[b[i].p]=((a[b[i].p]+1ll*b[i].v*b[i].sum)%mod+mod)%mod;

    (存疑:为何还要乘 b[i].v ???sum中没有包含吗?)

#include<bits/stdc++.h>
#define F(i,l,r) for(int i(l);i<=r;++i)
#define G(i,r,l) for(int i(r);i>=l;--i)
using namespace std;
using ll = long long;
char buf[100],*p1=buf,*p2=buf;
inline int gc(){return (p1==p2)&&(p2=(p1=buf)+fread(buf,1,100,stdin),p1==p2)?EOF:*p1++;}
inline int rd(){
	int x=0; char ch;
	while(!isdigit(ch=gc()));
	do x=(x<<3)+(x<<1)+(ch^48); while(isdigit(ch=gc()));
	return x;
}
const int N=1e6+5;
const int mod=998244353;
struct node{
	int op,p,v,mul,sum;
}b[N];
struct edge{
	int v,ne;
}e[N<<1];
int n,m,q,idx=0,cnt=0;
int in[N],a[N],first[N],ord[N],f[N];
void add(int x,int y){e[++idx]=(edge){y,first[x]};first[x]=idx;}
void topsort(){
	queue<int> q;
	F(i,1,m) if(!in[i]) q.emplace(i);
	while(q.size()){
		int u=q.front(); q.pop();
		ord[++cnt]=u;
		for(int i=first[u];i;i=e[i].ne){
			int v=e[i].v;
			if(!(--in[v])) q.emplace(v);
		}
	}
}
void getmul(){
	G(o,m,1){
		int u=ord[o];
		for(int i=first[u];i;i=e[i].ne){
			int v=e[i].v;
			b[u].mul=(1ll*b[u].mul*b[v].mul)%mod;
//			nw=(1ll*nw*b[u].mul)%mod;
		}
	}
}
void getsum(){
	F(o,1,m){
		int u=ord[o],nw=1;
		for(int i=first[u];i;i=e[i].ne){
			int v=e[i].v;
			b[v].sum=(1ll*b[v].sum+1ll*nw*b[u].sum)%mod;
			nw=(1ll*nw*b[v].mul)%mod;
		}
	}
}
signed main(){
	n=rd();
	F(i,1,n) a[i]=rd();
	m=rd();
	F(i,1,m){
		b[i].op=rd();
		if(b[i].op==1){
			b[i].p=rd(),b[i].v=rd();
			b[i].mul=1;
		}
		else if(b[i].op==2){
			b[i].v=rd();
			b[i].mul=b[i].v;
		}
		else{
			b[i].p=rd();
			b[i].mul=1;
			F(j,1,b[i].p){
				int o=rd();
				add(i,o);
				in[o]++;
			}
		}
	}
	topsort();
	getmul();
	q=rd(); int nw=1;
	F(i,1,q) f[i]=rd();
	G(i,q,1){
		int x=f[i];
		b[x].sum=(b[x].sum+1ll*nw)%mod;
		nw=(1ll*nw*b[x].mul)%mod;
	}
	getsum();
	F(i,1,n) a[i]=(1ll*a[i]*nw)%mod;
	F(i,1,m){
		if(b[i].op==1){
			a[b[i].p]=((a[b[i].p]+1ll*b[i].v*b[i].sum)%mod+mod)%mod;
		}
	}
	F(i,1,n) printf("%d ",a[i]);
	return 0;
} 
  • 总结:答案贡献分散,难以统计时,考虑转化统计主体,以集中贡献,优化算法。

标签:int,题解,函数调用,rd,1ll,mul,nw,CSP,mod
From: https://www.cnblogs.com/superl61/p/18146891

相关文章

  • CF1713F Lost Array 题解
    题目链接点击打开链接题目解法很牛的题!!!先考虑\((0,i)\)对\((j,n)\)的贡献,因为是异或,所以只要考虑奇偶性问题可以抽象成一条路径对应\(a_i\)的贡献,所以是否有\(a_i\)的贡献看\(\binom{n-i+j-1}{j-1}\)的奇偶性根据\(kummer\)定理,这个组合数是奇数当且仅当\(n-i+......
  • [题解][洛谷P1412] 经营与开发
    题目描述给定n,k,c,w,然后输入n组数据,数据分为两种:1ai:可以选择获得aiw的价值,但w会变成w(1-0.01*k)2bi:可以选择损失biw的价值,但w会变成w(1+0.01*c)求可获得的最大价值是多少。题解看到这个题,我的第一思路是求后缀和,然后让新得到的系数乘后缀和判断是否进行操作。但问题在于,对于......
  • 高斯消元学习笔记——P304题解
    如果你觉得这篇太啰嗦问题[SDOI2006]线性方程组题目描述已知\(n\)元线性一次方程组。\[\begin{cases}a_{1,1}x_1+a_{1,2}x_2+\cdots+a_{1,n}x_n=b_1\\a_{2,1}x_1+a_{2,2}x_2+\cdots+a_{2,n}x_n=b_2\\\cdots\\a_{n,1}x_1+a_{n,2}x......
  • ABC349F题解
    思想看到LCM想到质因数分解。首先,我们先把\(M\)质因数分解了,根号复杂度刚好1e8级别。然后我们发现一个很显然的性质:如果一个数不是\(M\)的因数那他肯定没用。所以此处我们就把不是因数地踢掉。我们惊奇地发现因为\(M\)的质因数分解最多\(13\)个不同的质数,然后我......
  • 题解 LGP5397【[Ynoi2018] 天降之物】/ 第四分块
    题解P5397【[Ynoi2018]天降之物】/第四分块题目描述一个长为\(n\)的序列\(a\)。你需要实现\(m\)个操作,操作有两种:把序列中所有值为\(x\)的数的值变成\(y\)。找出一个位置\(i\)满足\(a_i=x\),找出一个位置\(j\)满足\(a_j=y\),使得\(|i-j|\)最小,并输出\(|i-......
  • vsCode无法连接服务器问题解决及思考
    背景早上刚打开电脑,准备开始一天的工作。但是发现VSCode无法连接上我的虚拟机了,导致无法工作了,这让我十分头疼。最终花了将近一天的时间将问题解决,但是其中的过程走了不少弯路,浪费了不少时间,也进行了反思。我们作为开发人员,应该要用软件思维去理解这款产品,帮助我们去思考问题。......
  • P4168 [Violet] 蒲公英(题解)
    题目题目描述输入格式输出格式数据范围![]样例输入:63123212153615输出:121思路暴力本题求区间内的最小众数,容易想到去用数组sum[i]表示第i种花的个数,在去便利比较,但是复杂度nm一定会T,这时候就要对暴力进行优化。分块优化1如果我们将所......
  • Codeforces Round 932 (Div. 2)题解(c、d)
    CodeforcesRound932(Div.2)C.MessengerinMAC题目大意给定一些\(a_i\)\(和b_i\),选出尽量多的\(a_i和b_i\),使得\(\sum_{i=1}^ka_{p_i}+\sum_{i=1}^{k-1}\left|b_{p_i}-b_{p_{i+1}}\right|\)小于给定的\(l\)。题目解析由于题目没有要求\(\{p\}\)是升序排列的序列,因此......
  • P6018 [Ynoi2010] Fusion tree 题解
    题目链接:Fusiontree大部分人貌似用的边权01Trie,实际这题用点权01Trie类似文艺平衡树去写更方便。考虑两种常见的区间维护:线段树。使用的是父节点信息是归并了左右区间的信息,适用于不需要考虑父节点的贡献的信息。文艺平衡树。每个点就是一个信息,归并左右子树,外加当......
  • kafka消息只能在一台服务器消费的问题解决过程
    场景:kafka消费端应用部署在两台机器上,其中一台能消费到生产端发出的kafka消息,另一台服务器接收不到任何消息。解决过程:一、从消费端启动日志中找出所有消费端线程2024-04-2320:04:44,726[xx_xxapp03-1556011171628-976bc2af_watcher_executor]INFOkafka.consumer.RangeA......