bzoj 2217 [Poi2011]Lollipop 乱搞 贪心

2217: [Poi2011]Lollipop

Time Limit: 15 Sec  Memory Limit: 64 MBSec  Special Judge
Submit: 383  Solved: 159
[Submit][Status][Discuss]

Description

有一个长度为n的序列a1,a2,...,an。其中ai要么是1("W"),要么是2("T")。
现在有m个询问,每个询问是询问有没有一个连续的子序列,满足其和为q。

Input

第一行n,m (1<=n,m<=1000000)
第二行这个序列,起始编号为1,终止编号为n
下面每行一个询问q,询问有没有一个连续的子序列,满足其和为q (1<=q<=2000000)

Output

对于每个询问,输出一行,如果有,输出这个序列的起点和终点(如果有多个输出任意一个);如果没有,输出“NIE”。

Sample Input

5 3
TWTWT
5
1
7

Sample Output

1 3
2 2
NIE

HINT

尚无SPJ,请不要提交

 

这题有一个性质,如果存在一个连续的序列和为k,那么在前缀和中一定存在k或者k+1

为什么?

可以证明,如果l-r和为k,那么后面部分,可以不断减去,前面部分不断加入,如果相差为2,那么后面部分减去一个1,或者2,使其

差<2就可以了,这样是保证了这个性质。

 

所以对于这道题,对于k判断前缀和十分有k,有的话直接输出,否则判断是否有k+1,如果有的话,就记录前面二的个数,后面哪个位置是1,

然后搞一搞就好了。

 1 #pragma GCC optimize(2)
 2 #pragma G++ optimize(2)
 3 #include<iostream>
 4 #include<algorithm>
 5 #include<cmath>
 6 #include<cstdio>
 7 #include<cstring>
 8 
 9 #define N 27
10 using namespace std;
11 inline int read()
12 {
13     int x=0,f=1;char ch=getchar();
14     while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}
15     while(isdigit(ch)){x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
16     return x*f;
17 }
18 
19 #define MAXN 1000010
20 #define MAXM 1010
21 #define INF 1000000000
22 #define MOD 1000000007
23 #define eps 1e-8
24 #define ll long long
25 
26 int n,m;
27 char a[MAXN];
28 int s[MAXN],nxt[MAXN];
29 int v[MAXN<<1];
30 int main()
31 {
32     int i,x;
33     scanf("%d%d%s",&n,&m,a+1);
34     for(i=1;i<=n;i++)
35     {
36         s[i]=s[i-1]+1+(a[i]=='T');
37         v[s[i]]=i;
38     }
39     x=n+1;
40     for(i=n;i;i--)
41     {
42         if(a[i]=='W')x=i;
43         nxt[i]=x-i;
44     }
45     while(m--)
46     {
47         scanf("%d",&x);
48         if(v[x])printf("%d %d\n",1,v[x]);
49         else if(v[x+1])
50         {
51             int l=1,r=v[x+1];
52             if(nxt[l]<nxt[r])
53             {
54                 r+=nxt[l];
55                 l+=nxt[l]+1;
56                 printf("%d %d\n",l,r);
57             }else if(r+nxt[r]<=n)
58             {
59                 l+=nxt[r];
60                 r+=nxt[r];
61                 printf("%d %d\n",l,r);
62             }else printf("NIE\n");
63         }else printf("NIE\n");
64     }
65 }

 

转载于:https://www.cnblogs.com/fengzhiyuan/p/8486673.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值