解题报告 P4588 [TJOI2018]数学计算

P4588 [TJOI2018]数学计算

一道不错的线段树练手题。

我们开一棵线段树,下标维护操作次序,节点值即是对应区间的积

设当前操作是第 \(i\) 次操作。

每次操作 \(1\) 我们都把 \(i\) 对应叶节点的值改为 \(m\) 。然后查询 \([1,i]\) 的乘积。

每次操作 \(2\) 我们都把 \(pos\) 对应的叶节点的值改为 \(1\) ,然后查询 \([1,i]\) 的乘积。

记得初始化为 \(1\)

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;

const int N=1e6+10;

ll tree[N<<2];

#define lnode node<<1
#define rnode node<<1|1
#define DEFMID int mid=(start+end)>>1

ll Q,M;

void push_up(int node)
{
	tree[node]=tree[lnode]*tree[rnode]%M;
}

void build(int node,int start,int end)
{
	if(start==end)
	{
		tree[node]=1;
		return ;
	}
	DEFMID;
	build(lnode,start,mid);
	build(rnode,mid+1,end);
	push_up(node);
	return ;
}

void update(int node,int start,int end,int pos,ll val)
{
	if(start==end) {tree[node]=val; return ;}
	DEFMID;
	if(pos<=mid) update(lnode,start,mid,pos,val);
	else update(rnode,mid+1,end,pos,val);
	push_up(node);
	return;
}

ll query(int node,int start,int end,int l,int r)
{
	if(end<l||start>r) return 1;
	if(l<=start&&end<=r) return tree[node]%M;

	DEFMID;
	ll res=1;
	res=(res%M)*query(lnode,start,mid,l,r)%M;
	res=(res%M)*query(rnode,mid+1,end,l,r)%M;
	push_up(node);
	return (res%M+M)%M;
}

int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		scanf("%lld%lld",&Q,&M);
		build(1,1,Q+10);
		for(int i=1;i<=Q;i++)
		{
			int opt,k;
			scanf("%d%d",&opt,&k);
			if(opt==1)
			{
				update(1,1,Q+10,i,k);
				ll res=query(1,1,Q+10,1,i);
				printf("%lld\n",res);
			}
			else
			{
				update(1,1,Q+10,k,1); update(1,1,Q+10,i,1);
				ll res=query(1,1,Q+10,1,i);
				printf("%lld\n",res);
			}
		}
	}
	return 0;
}
posted @ 2021-03-28 17:34  RemilaScarlet  阅读(50)  评论(0编辑  收藏  举报