HDU 6155 Subsequence Count 线段树维护矩阵

Subsequence Count

Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 256000/256000 K (Java/Others)

Problem Description

Given a binary string S[1,...,N] (i.e. a sequence of 0‘s and 1‘s), and Q queries on the string.

There are two types of queries:

1. Flipping the bits (i.e., changing all 1 to 0 and 0 to 1) between l and r (inclusive).
2. Counting the number of distinct subsequences in the substring S[l,...,r].

Input

The first line contains an integer T, denoting the number of the test cases.

For each test, the first line contains two integers N and Q.

The second line contains the string S.

Then Q lines follow, each with three integers type, l and r, denoting the queries.

1≤T≤5

1≤N,Q≤105

S[i]∈{0,1},∀1≤i≤N

type∈{1,2}

1≤l≤r≤N

Output

For each query of type 2, output the answer mod (109+7) in one line.

Sample Input

2
4 4
1010
2 1 4
2 2 4
1 2 3
2 1 4
4 4
0000
1 1 2
1 2 3
1 3 4
2 1 4

Sample Output

11
6
8
10

题解:

  设定dp[i][0/1] 到第i个字符以0/1结尾的子序列方案

  若s[i] = =1 : dp[i][1] = dp[i-1][0] + dp[i-1][1] + 1;

        dp[i][0] = dp[i-1][0];

若是s[i] == 0: dp[i][0] =  dp[i-1][0] + dp[i-1][1] + 1;

        dp[i][1] = dp[i-1][1];

  写成矩阵,用线段树维护一段连续矩阵乘积,有点卡常数

#include<bits/stdc++.h>
using namespace std;
#define ls i<<1
#define rs ls | 1
#define mid ((ll+rr)>>1)
#define pii pair<int,int>
#define MP make_pair
typedef long long LL;
typedef unsigned long long ULL;
const long long INF = 1e18+1LL;
const double pi = acos(-1.0);
const int N=5e5+20,M=1e6+10,inf=2147483647;

inline LL read(){
    LL x=0,f=1;char ch=getchar();
    while(ch<‘0‘||ch>‘9‘){if(ch==‘-‘)f=-1;ch=getchar();}
    while(ch>=‘0‘&&ch<=‘9‘){x=x*10+ch-‘0‘;ch=getchar();}
    return x*f;
}

const LL mod = 1e9+7;
char s[N];

struct Matix {
    LL arr[3][3];
}E,F,again,EE;
inline Matix mul(Matix a,Matix b) {
    Matix ans;
    memset(ans.arr,0,sizeof(ans.arr));
    for(int i = 0; i < 3; i++) {
        for(int j = 0; j < 3; j++) {
            for(int k = 0; k < 3; k++)
                ans.arr[i][j] += a.arr[i][k] * b.arr[k][j],ans.arr[i][j] %= mod;
        }
    }
    return ans;
}

Matix v[N * 4],now,facE[N],facF[N];
int lazy[N * 4],fi[N * 4],se[N * 4];

void change(int i) {
    swap(v[i].arr[0][0],v[i].arr[1][0]);
    swap(v[i].arr[0][1],v[i].arr[1][1]);
    swap(v[i].arr[0][2],v[i].arr[1][2]);
    swap(v[i].arr[0][0],v[i].arr[0][1]);
    swap(v[i].arr[1][0],v[i].arr[1][1]);
    swap(v[i].arr[2][0],v[i].arr[2][1]);
}
void push_down(int i,int ll,int rr) {
    if(!lazy[i]) return;
    lazy[ls] ^= 1;
    lazy[rs] ^= 1;
    change(ls);change(rs);
    lazy[i] ^= 1;
}
inline void push_up(int i,int ll,int rr) {
    v[i] = mul(v[ls],v[rs]);
}
void build(int i,int ll,int rr) {
    lazy[i] = 0;
    if(ll == rr) {
        if(s[ll] == ‘1‘) v[i] = E,fi[i] = 1,se[i] = 0;
        else v[i] = F,fi[i] = 0,se[i] = 1;
        return ;
    }
    build(ls,ll,mid);
    build(rs,mid+1,rr);
    push_up(i,ll,rr);
}
inline void update(int i,int ll,int rr,int x,int y) {
    push_down(i,ll,rr);
    if(ll == x && rr == y) {
        lazy[i] ^= 1;
        change(i);
        return ;
    }
    if(y <= mid) update(ls,ll,mid,x,y);
    else if(x > mid) update(rs,mid+1,rr,x,y);
    else update(ls,ll,mid,x,mid),update(rs,mid+1,rr,mid+1,y);
    push_up(i,ll,rr);
}
inline Matix ask(int i,int ll,int rr,int x,int y) {
    push_down(i,ll,rr);
    if(ll == x && rr == y) {
        return v[i];
    }
    if(y <= mid) return ask(ls,ll,mid,x,y);
    else if(x > mid) return ask(rs,mid+1,rr,x,y);
    else return mul(ask(ls,ll,mid,x,mid),ask(rs,mid+1,rr,mid+1,y));
    push_up(i,ll,rr);
}

int main() {
    EE.arr[0][0] = 1,EE.arr[1][1] = 1,EE.arr[2][2] = 1;

    E.arr[0][0] = 1;E.arr[0][1] = 1;E.arr[0][2] = 1;
    E.arr[1][1] = 1;E.arr[2][2] = 1;

    F.arr[0][0] = 1;F.arr[1][0] = 1;F.arr[1][1] = 1;
    F.arr[1][2] = 1;F.arr[2][2] = 1;

    again.arr[0][2] = 1;

    int T;
    T = read();
    while(T--) {
        int n,Q;
        n = read();
        Q = read();
        scanf("%s",s+1);
        build(1,1,n);
        while(Q--) {
            int op,l,r;
            op = read();
            l = read();
            r = read();
            if(op == 1)
                update(1,1,n,l,r);
            else {
                now = mul(again,ask(1,1,n,l,r));
                printf("%lld\n",(now.arr[0][0]+now.arr[0][1])%mod);
            }
        }
    }
    return 0;
}

先考虑怎么算 s_1, s_2, \ldots, s_ns?1??,s?2??,…,s?n?? 的答案。设 dp(i, 0/1)dp(i,0/1) 表示考虑到 s_is?i??,以 0/10/1 结尾的串的数量。那么 dp(i, 0) =dp(i - 1, 0) + dp(i - 1, 1) + 1dp(i,0)=dp(i−1,0)+dp(i−1,1)+111 也同理。
那么假设在某个区间之前,dp(i, 0/1) = (x, y)dp(i,0/1)=(x,y) 的话,过了这段区间,就会变成 (ax + by + c, dx + ey + f)(ax+by+c,dx+ey+f) 的形式,只要用线段树维护这个线性变化就好了。

时间: 2024-12-21 04:14:53

HDU 6155 Subsequence Count 线段树维护矩阵的相关文章

HDU.6155.Subsequence Count(线段树 矩阵)

题目链接 首先考虑询问[1,n]怎么做 设 f[i][0/1]表示[1,i]以0/1结尾的不同子序列个数 则\(if(A[i]) f[i][1] = f[i-1][0] + f[i-1][1] + 1 , f[i][0] = f[i-1][0]\) \(\ \ if(!A[i]) f[i][0] = f[i-1][0] + f[i-1][1] + 1 , f[i][1] = f[i-1][1]\) 很整齐,我们来写成矩阵的样子: \(f[i,0]\ f[i,1]\ 1=f[i-1,0]\ f[i

Subsequence Count 2017ccpc网络赛 1006 dp+线段树维护矩阵

Problem Description Given a binary string S[1,...,N] (i.e. a sequence of 0's and 1's), and Q queries on the string.There are two types of queries:1. Flipping the bits (i.e., changing all 1 to 0 and 0 to 1) between l and r (inclusive).2. Counting the

HDU 1542 Atlantis (线段树求矩阵覆盖面积)

题意:给你n个矩阵求覆盖面积. 思路:看了别人的结题报告 给定一个矩形的左下角坐标和右上角坐标分别为:(x1,y1).(x2,y2),对这样的一个矩形,我们构造两条线段,一条定位在x1,它在y坐标的区间是[y1,y2],并且给定一个cover域值为1:另一条线段定位在x2,区间一样是[y1,y2],给定它一个cover值为-1.根据这样的方法对每个矩形都构造两个线段,最后将所有的线段根据所定位的x从左到右进行排序 #include <iostream> #include <stdio.h

CF718C Sasha and Array(线段树维护矩阵)

题解 (不会矩阵加速的先去学矩阵加速) 反正我想不到线段树维护矩阵.我太菜了. 我们在线段树上维护一个区间的斐波那契的列矩阵的和. 然后询问时提取每个符合题意列矩阵的答案项(不是列矩阵存了两项吗,一个是当前项,一个是用来递推的) 因为矩阵乘有结合律所以区间加这个操作就直接区间乘变换矩阵的x次方就行. 然后记得开long long 1 #include<iostream> 2 #include<cstring> 3 #include<cstdio> 4 #include&

HDU 6155 Subsequence Count(矩阵 + DP + 线段树)题解

题意:01串,操作1:把l r区间的0变1,1变0:操作2:求出l r区间的子序列种数 思路:设DP[i][j]为到i为止以j结尾的种数,假设j为0,那么dp[i][0] = dp[i - 1][1] + dp[i -1][0] (0结尾新串) + dp[i - 1][0] (0结尾旧串) - dp[i - 1][0] (重复) + 1(0本身被重复时去除). 那么可以得到转移时的矩阵 $$ \left( \begin{matrix} dp[i - 1][0] & dp[i - 1][1] &am

2017中国大学生程序设计竞赛 - 网络选拔赛 HDU 6155 Subsequence Count 矩阵快速幂

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6155 题意: 题解来自:http://www.cnblogs.com/iRedBean/p/7398272.html 先考虑dp求01串的不同子序列的个数. dp[i][j]表示用前i个字符组成的以j为结尾的01串个数. 如果第i个字符为0,则dp[i][0] = dp[i-1][1] + dp[i-1][0] + 1,dp[i][1] = dp[i-1][1] 如果第i个字符为1,则dp[i][1

[HDU6155]Subsequence Count(线段树+矩阵)

DP式很容易得到,发现是线性递推形式,于是可以矩阵加速.又由于是区间形式,所以用线段树维护. https://www.cnblogs.com/Miracevin/p/9124511.html 关键在于证明区间操作中,可以直接在打标记的位置翻转矩阵两行两列. 上面网址用代数形式证了一遍,这里考虑从矩阵本身解释. 由线代内容可知,将一个矩阵作初等行变换,相当于将其左乘一个作了相应初等列变换的单位矩阵.同理将一个矩阵作初等列变换,相当于将其又乘一个作了相应初等行变换的单位矩阵. 这里,左乘的矩阵$T=

hdu 6155 -&#160;Subsequence Count

话说这题比赛时候过的好少,连题都没读TOT 先考虑dp求01串的不同子序列的个数. dp[i][j]表示用前i个字符组成的以j为结尾的01串个数. 如果第i个字符为0,则dp[i][0] = dp[i-1][1] + dp[i-1][0] + 1,dp[i][1] = dp[i-1][1] 如果第i个字符为1,则dp[i][1] = dp[i-1][1] + dp[i-1][0] + 1,dp[i][0] = dp[i-1][0] 显然这是线性递推,我们考虑如何用矩阵表示这种递推关系. 下面分别

codeforces CF718C Sasha and Array 线段树维护矩阵

$ \Rightarrow $ 戳我进CF原题 C. Underground Lab time limit per test: 1 second memory limit per test: 256 megabytes input: standard input output: standard output The evil Bumbershoot corporation produces clones for gruesome experiments in a vast undergroun