子序列个数

N - 子序列个数
Time Limit:2000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u
Submit  Status

Description

子序列的定义:对于一个序列a=a[1],a[2],......a[n]。则非空序列a'=a[p1],a[p2]......a[pm]为a的一个子序列,其中1<=p1

例如4,14,2,3和14,1,2,3都为4,13,14,1,2,3的子序列。

对于给出序列a,请输出不同的子序列的个数。(由于答案比较大,请将答案mod 1000000007)

Input

输入包含多组数据。每组数据第一行为一个整数n(1<=n<=1,000,000),表示序列元素的个数。

第二行包含n个整数a[i] (0<=a[i]<=1,000,000)表示序列中每个元素。

Output

输出一个整数占一行,为所求的不同子序列的个数。由于答案比较大,请将答案mod 1000000007。

Sample Input

4
1 2 3 2

Sample Output

13

Hint

其中40%数据点1<=n<=1000。
由于英语比较差,所以就选了个汉语简单的题目来做。。。嘿嘿。这个题目最主要的是分析,
用代码来模拟过程,要计算一个数列的子序列,我们的想法,是一个一个接入,

第i个数据的子序列为sum[i].当接入的数据b在前面没有的时候,他的子序列个数2*sum[i-1]+1;
因为,当接入一个数据,不计最后一个数据的大小为sum[i-1].用这个数据进行代换又产生个
sum[i-1],再加上总的一个,就为2*sum[i-1]+1;

当接入的数据b在前面已经出现,那么sum[i]=2*sum[i-1]-sum[b-1];其中b为前面i个中最后一
个出现接入数据的位置。因为不算自身b,前面的数据为sum[i-1];而加上最后一个数,在前
面一个数b1之前的数都已经算过了,而新加的数据的功能只在b之后所以为sum[i-1]-sun[b-1];
当然,这个情况也包括了总的一个,所以这种情况下为sum[i-1]*2-sum[b-1];

下面是实现代码:
#include
#include
 __int64 b[1000001],c[1000001],sum[1000001];
int main()
{

    int a,i;
    while(scanf("%d",&a)!=EOF)
    {
        for(i=1;i<=a;i++)
        scanf("%d",&b[i]);
        memset(c,0,sizeof(c));
        sum[0]=0;
        for(i=1;i<=a;i++)
        {
            if(!c[b[i]])
            sum[i]=(2*sum[i-1]+1);
            else
            sum[i]=(2*sum[i-1]-sum[c[b[i]]-1]);
            sum[i]=(sum[i]+1000000007)%1000000007;
            c[b[i]]=i;
        }
        printf("%I64d\n",sum[a]);
    }
}


你可能感兴趣的:(DP)