zoukankan      html  css  js  c++  java
  • 【BZOJ4499】线性函数 线段树

    【BZOJ4499】线性函数

    Description

    小C最近在学习线性函数,线性函数可以表示为:f(x) = kx + b。现在小C面前有n个线性函数fi(x)=kix+bi ,他对这n个线性函数执行m次操作,每次可以:
    1.M i K B 代表把第i个线性函数改为:fi(x)=kx+b 。
    2.Q l r x 返回fr(fr-1(...fl(x)))  mod  10^9+7 。

    Input

    第一行两个整数n, m (1 <= n, m <= 200,000)。
    接下来n行,每行两个整数ki, bi。
    接下来m行,每行的格式为M i K B或者Q l r x。

    Output

    对于每个Q操作,输出一行答案。

    Sample Input

    5 5
    4 2
    3 6
    5 7
    2 6
    7 5
    Q 1 5 1
    Q 3 3 2
    M 3 10 6
    Q 1 4 3
    Q 3 4 4

    Sample Output

    1825
    17
    978
    98

    HINT

    1 <= n, m <= 200,000,0 <= k, b, x < 1000,000,007

    题解:这个式子显然是可以用线段树维护的。

    维护区间k的乘积sk,区间的那个式子的值sb,则合并时sb=sb1*sk2+sb2,sk=sk1*sk2。答案就是x*sk+sb。

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #define lson x<<1
    #define rson x<<1|1
    using namespace std;
    typedef long long ll;
    const ll P=1000000007;
    const int maxn=200010;
    int n,m;
    ll B[maxn],K[maxn];
    char str[5];
    struct node
    {
    	ll b,k;
    	node() {}
    	node(ll x,ll y) {b=x,k=y;}
    	node operator + (const node &a) const {return node((b*a.k+a.b)%P,k*a.k%P);}
    }s[maxn<<2];
    void build(int l,int r,int x)
    {
    	if(l==r)
    	{
    		s[x]=node(B[l],K[l]);
    		return ;
    	}
    	int mid=(l+r)>>1;
    	build(l,mid,lson),build(mid+1,r,rson);
    	s[x]=s[lson]+s[rson];
    }
    void updata(int l,int r,int x,int a)
    {
    	if(l==r)
    	{
    		s[x]=node(B[l],K[l]);
    		return ;
    	}
    	int mid=(l+r)>>1;
    	if(a<=mid)	updata(l,mid,lson,a);
    	else	updata(mid+1,r,rson,a);
    	s[x]=s[lson]+s[rson];
    }
    node query(int l,int r,int x,int a,int b)
    {
    	if(a<=l&&r<=b)	return s[x];
    	int mid=(l+r)>>1;
    	if(b<=mid)	return query(l,mid,lson,a,b);
    	if(a>mid)	return query(mid+1,r,rson,a,b);
    	return query(l,mid,lson,a,b)+query(mid+1,r,rson,a,b);
    }
    inline int rd()
    {
    	int ret=0,f=1;	char gc=getchar();
    	while(gc<'0'||gc>'9')	{if(gc=='-')	f=-f;	gc=getchar();}
    	while(gc>='0'&&gc<='9')	ret=ret*10+gc-'0',gc=getchar();
    	return ret*f;
    }
    int main()
    {
    	int i,a,b;
    	ll x;
    	n=rd(),m=rd();
    	for(i=1;i<=n;i++)	K[i]=rd(),B[i]=rd();
    	build(1,n,1);
    	for(i=1;i<=m;i++)
    	{
    		scanf("%s",str);
    		if(str[0]=='M')
    		{
    			a=rd(),K[a]=rd(),B[a]=rd();
    			updata(1,n,1,a);
    		}
    		else
    		{
    			a=rd(),b=rd(),x=rd();
    			node tmp=query(1,n,1,a,b);
    			printf("%lld
    ",(x*tmp.k+tmp.b)%P);
    		}
    	}
    	return 0;
    }
  • 相关阅读:
    C语言第0次作业
    C博客作业01分支、顺序结构
    C博客第02次作业循环结构
    关于编写有效用例的12秘诀
    关于调用FTP中遇到的问题以及解决方案
    关于FtpWebRequest.Timeout属性的理解
    WPF中四种不同的测量单位
    关于检查Oracle表及列是否存在SQL语句
    ArcSDE configuration files
    C#判断不同版本的Excel
  • 原文地址:https://www.cnblogs.com/CQzhangyu/p/7671055.html
Copyright © 2011-2022 走看看