序列问题

【题目描述】

小H是个善于思考的学生,她正在思考一个有关序列的问题。

她的面前浮现出了一个长度为n的序列{ai},她想找出两个非空的集合S、T。

这两个集合要满足以下的条件:

1. 两个集合中的元素都为整数,且都在 [1, n] 里,即Si,Ti ∈ [1, n]。

2. 对于集合S中任意一个元素x,集合T中任意一个元素y,满足x < y。

3. 对于大小分别为p, q的集合S与T,满足

            a[s1] xor a[s2] xor a[s3] ... xor a[sp] = a[t1] and a[t2] and a[t3] ... and a[tq].

小H想知道一共有多少对这样的集合(S,T),你能帮助她吗?

【输入格式】

       第一行,一个整数n

       第二行,n个整数,代表ai。

【输出格式】

       仅一行,表示最后的答案。

【样例输入】

4

1 2 3 3

【样例输出】

4

【样例解释】

S = {1,2}, T = {3}, 1 ^ 2 = 3 = 3 (^为异或)

S = {1,2}, T = {4},  1 ^ 2 = 3 = 3

S = {1,2}, T = {3,4}  1 ^ 2 = 3 & 3 = 3 (&为与运算)

S = {3}, T = {4}  3 = 3 = 3 

【数据范围】

    30%: 1 <= n <= 10

    60%: 1 <= n <= 100

    100%: 1 <= n <= 1000, 0 <= ai < 1024

/*
  暴力30分。
  正解是DP,考试时也想到了,但是状态定义的不好没所以很慢,题解的状态很巧妙,异或和与的情况分别定义。
   两个数相等就相当于两个数的 xor 为 0。设 f[i][j][k=0..2]代表 处理到第i个数,如果 k=1 代表 and 值为 j,如果 k=2 代表 xor 值为 j,如果 k= 0 则代表一个元素都没取。所以很容易得到方程:
f[i][j][0] = f[i + 1][j][0]
f[i][j & ai][1] = f[i + 1][j][1] + f[i + 1][j][0] + f[i + 1][j & ai][1]
f[i][j ^ ai][2] = f[i + 1][j][1] + f[i + 1][j][2] + f[i + 1][j ^ ai][2];
最后 f[1][0][2]就是答案。
由于高精度要压位,所以没打。
*/ #include<cstdio> #include<iostream> #define N 1500 #define lon long long using namespace std; int a[N],n; lon f[2][N][3]; int main(){ //freopen("sequence.in","r",stdin); //freopen("sequence.out","w",stdout); freopen("jh.in","r",stdin); scanf("%d",&n); for(int i=n;i;--i) scanf("%d",&a[i]); f[0][1023][0]=1LL; int nxt=0,cur=0; for(int i=0;i<n;++i){ nxt=cur^1; for(int j=0;j<=1023;++j) for (int k=0;k<=2;++k) f[nxt][j][k]=f[cur][j][k]; for (int j=0;j<=1023;++j) { int va=a[i+1]&j,vx=a[i+1]^j; f[nxt][va][1]+=f[cur][j][0];f[nxt][va][1]+=f[cur][j][1]; f[nxt][vx][2]+=f[cur][j][1];f[nxt][vx][2]+=f[cur][j][2]; } cur^=1; } cout<<f[cur][0][2]; return 0; }
原文地址:https://www.cnblogs.com/harden/p/6063286.html