Description
子序列的定义:对于一个序列a=a[1],a[2],......a[n]。则非空序列a'=a[p1],a[p2]......a[pm]为a的一个子序列,其中1<=p1<p2<.....<pm<=n。
例如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<stdio.h>
#include<string.h>
__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]);
}
}