【伪暴力+智商剪枝】Codeforces Round #489 (Div. 2) D
失踪人口突然回归……orz。题解还是有必要写的,虽然估计只有自己(?自己也不一定看得懂)看得懂。
题目链接:http://codeforces.com/contest/992/problem/D
题目大意:给出n个数字a和一个k,求数列a中的子区间aa满足aa的和乘k等于aa的积。(a<=1e8,n<=2e5,k<=1e5)
这道题没找到官方题解,所以看了一下standing rank1的dalao的代码。 鸣谢 dotorya
第一眼就觉得短,非常短。赛上把我卡得很恶心的1的情况竟然被两行代码解决了……再度%大佬的机智。
暴力解决这道题(枚举所有区间)的时间复杂度是n^2,在看了一眼数据范围后被我放弃了。
先说一下赛上思路过程,前缀和是第一反应,然后考虑前缀积。再一看范围,计算量直接T掉高精度。不过转念一想,k*sum_aa的最大值也就2e18,好像是在暗示着什么。用和来寻找积而不用积来寻找和。在草稿本上演算了一下,发现积的增长速度十分快,一旦在某个瞬间大于sum_aa*k了之后就不会再小于了。当然是在没有1的情况。想了半天不知道怎么破解这个1,最终GG退赛。
看了dalao的代码之后发现dalao机智地把1折叠了起来……既然乘1等于不乘那我们就不乘,跳过就行了,加1的影响用前缀和搞定。然后……就是暴力,对,暴力(Formiko的智商真是碾压我呀。)
%%%,我初步估计时间复杂度在nlog2e18,因为乘积的增长速度真的十分十分快,log级别的。此处鸣谢Formiko点醒了我。这确实是基本的数学素质,我竟然忘掉了。
中途写完代码的时候发现不能完全忽略1的影响,再次感谢formiko,他的一句“1加着加着就蹦出来一个解”成功帮我AC了这道题,同时%dalao的二分思路。
废话太多我精简地说一下这道题的题解。
首先用前缀和处理数列,把这个数列映射到另一个数列上,类似于链表(但是要保留原下标,方便前缀和以及1的影响)。映射方式就是折叠1,只保留非1的数,也就是对乘积有影响的数,对于固定的左端点,这个区间长度不会超过63,时间复杂度是可以承受的。接下来暴力枚举每个左端点,在映射的数组上跳跃,最多条约63次。中途会有1出现的情况。也就是分右端点是1和右端点不是1两种情况。如果右端点不是1,那么直接在映射数组上验证是否满足条件就行了,如果右端点是1,那么在一段连续1中,由于乘积一直乘的1,值不变,和一直在增加,所以如果出现满足条件的解,只会有一个,而且满足单调性!!那么二分就可以了。总时间复杂度在63*nlogn。
下面放代码:
/* by Lstg */ #include<stdio.h>
#include<iostream>
#define MAXN 200105
#define inf 3000000000000000000
using namespace std; int b[MAXN];
long long a[MAXN],sum[MAXN]; bool _find(int k,int l,int r,long long tmp){ if(l>r)return false;
int mid;
while(l<=r){
mid=(l+r)>>;
if(sum[mid]-sum[k]==tmp)return true;
if(sum[mid]-sum[k]>tmp)r=mid-;
else l=mid+;
}
return false;
} int main(){ int n,i,j;
long long k,tmp;
scanf("%d%I64d",&n,&k);
for(i=;i<=n;i++){
scanf("%I64d",&a[i]);
sum[i]=sum[i-]+a[i];
}
b[i]=i;
for(i=n;i>=;i--){
if(a[i]>)b[i]=i;
else b[i]=b[i+];
} long long cnt=;
for(i=;i<=n;i++){
j=b[i];
tmp=1ll;
if(a[i]==&&k==)cnt++;
while(j<=n){
if(inf/a[j]<tmp)break;
tmp*=a[j];
if(tmp==k*(sum[j]-sum[i-]))cnt++;
if(tmp%k==&&_find(i-,j+,b[j+]-,tmp/k))cnt++;
j=b[j+];
}
}
printf("%I64d",cnt);
return ;
}
【伪暴力+智商剪枝】Codeforces Round #489 (Div. 2) D的更多相关文章
- 并查集+bfs+暴力滑窗 Codeforces Round #356 (Div. 2) E
http://codeforces.com/contest/680/problem/E 题目大意:给你一个n*n的图,然后图上的 . (我们下面都叫做‘点’)表示可以走,X表示不能走,你有如下的操作, ...
- Codeforces Round #489 (Div. 2) E. Nastya and King-Shamans(线段树)
题意 给出一个长度为 \(n\) 的序列 \(\{a_i\}\) , 现在会进行 \(m\) 次操作 , 每次操作会修改某个 \(a_i\) 的值 , 在每次操作完后你需要判断是否存在一个位置 \(i ...
- Codeforces Round #489 (Div. 2)
A. Nastya and an Array time limit per test 1 second memory limit per test 256 megabytes input standa ...
- Codeforces Round #489 (Div. 2) E - Nastya and King-Shamans
E - Nastya and King-Shamans 题目大意:有n个数,每一次操作更改一个数,每次操作之后问你是否有一个数等于其前面所有数的和. 思路:好题,想了很久没想出来,看了题解,主要思想就 ...
- Codeforces Round #489 (Div. 2) B、C
B. Nastya Studies Informatics time limit per test 1 second memory limit per test 256 megabytes input ...
- [Codeforces]Codeforces Round #489 (Div. 2)
Nastya and an Array 输出有几种不同的数字 #pragma comment(linker, "/STACK:102400000,102400000") #ifnd ...
- 数论/暴力 Codeforces Round #305 (Div. 2) C. Mike and Frog
题目传送门 /* 数论/暴力:找出第一次到a1,a2的次数,再找到完整周期p1,p2,然后以2*m为范围 t1,t2为各自起点开始“赛跑”,谁落后谁加一个周期,等到t1 == t2结束 详细解释:ht ...
- 暴力 Codeforces Round #305 (Div. 2) B. Mike and Fun
题目传送门 /* 暴力:每次更新该行的num[],然后暴力找出最优解就可以了:) */ #include <cstdio> #include <cstring> #includ ...
- 暴力 Codeforces Round #183 (Div. 2) A. Pythagorean Theorem II
题目传送门 /* 暴力:O (n^2) */ #include <cstdio> #include <algorithm> #include <cstring> # ...
随机推荐
- Atcoder #014 agc014_D 树形DP+nim变形
LINK 题意:两人在一颗树上做游戏,先手可以将树上一个节点染白,后手染黑,到最后时,所有与黑色相邻的白色同时变黑.如果还存在白色,先手胜,否则后手胜. 思路:首先不考虑树上,单独为链时,不管找规律也 ...
- C++类四个默认函数&深复制&浅复制
学习C++语言的同学都知道,C++中类是有默认的几个函数的,主要是有四个函数: 四个函数 默认构造函数:A(void),无参构造函数 拷贝(复制)构造函数:A(const A&a).用一个对象 ...
- Eclipse自动代码补全
Windows——>Preferences——>Java-->Editor-->Content Asist, 在Auto activation triggers for Jav ...
- 使用Forms Authentication
using System; using System.Web; using System.Web.Security; namespace AuthTest { public class Aut ...
- bootstrap-select,selectpicker 用法详细:通过官方文档翻译
用过selectpicker的都说好~但是网上中文的教程又找不到比较完整的用法,于是去官网看了下 顺便弄过来翻译一下: 选项可以通过数据属性或JavaScript传递.对于数据属性,附加选项名称dat ...
- 【leetcode 简单】第十五题 加一
给定一个非负整数组成的非空数组,在该数的基础上加一,返回一个新的数组. 最高位数字存放在数组的首位, 数组中每个元素只存储一个数字. 你可以假设除了整数 0 之外,这个整数不会以零开头. 示例 1: ...
- matlab求逆运算:左除( \ )和右除( / ),inv,pinv
矩阵求逆可以使用左除(\)和右除(/),inv,pinv 首先了解需要求逆的矩阵A是否为奇异方阵 inv 若A为非奇异方阵,则存在逆矩阵,可利用inv求逆: inv(A) pinv 若需要求逆的矩阵A ...
- Java八种基本类型
boolean 二进制位: true ,false byte 二进制位:8 -128 - 127 -2的7次方到2的7次方-1 char 二进制位:16 0 - 65535 short 二 ...
- flask插件系列之SQLAlchemy实用技巧
下面记录一下SQLAlchemy使用的技巧. 在多模块下定义models 如果由多个蓝图下读定义了model模块,在初始化的时候需要加载到上下文中. 当使用flask_Migrate迁移数据库的时候, ...
- 145.Binary Tree Postorder Traversal---二叉树后序非递归遍历
题目链接 题目大意:后序遍历二叉树. 法一:普通递归,只是这里需要传入一个list来存储遍历结果.代码如下(耗时1ms): public List<Integer> postorderTr ...