[洛谷P3987]我永远喜欢珂朵莉~

题目大意:

给你\(n(n\le10^5)\)个数\(A_{1\sim n}(A_i\le5\times10^5)\),\(m(m\le5\times10^5)\)次操作,操作包含以下两种:

  1. 将区间\([l,r]\)间所有\(v\)的倍数除以\(v\)。
  2. 求区间\([l,r]\)所有数之和。

思路1:

对范围内要用到的每个质因数开一个set维护包含该质因子的\(A_i\)下标。

对于操作\(1\),将\(v\)分解质因数,在set中查找下标在\([l,r]\)中的满足条件的数。并将这些数\(\div v\)。若操作过后不包含该质因数,则将其从set中删去。

对于操作\(2\),用树状数组维护即可。

源代码1:

#pragma GCC optimize("Ofast")
#pragma GCC optimize("unroll-loops")
#pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
#include<cstdio>
#include<cctype>
#include<vector>
#include<climits>
#include<sys/mman.h>
#include<sys/stat.h>
#include<ext/pb_ds/tree_policy.hpp>
#include<ext/pb_ds/assoc_container.hpp>
class MMapInput {
private:
char *buf,*p;
int size;
public:
MMapInput() {
register int fd=fileno(stdin);
struct stat sb;
fstat(fd,&sb);
size=sb.st_size;
buf=reinterpret_cast<char*>(mmap(0,size,PROT_READ,MAP_PRIVATE,fileno(stdin),0));
p=buf;
}
char getchar() {
return (p==buf+size||*p==EOF)?EOF:*p++;
}
};
MMapInput mmi;
inline int getint() {
register char ch;
while(!isdigit(ch=mmi.getchar()));
register int x=ch^'0';
while(isdigit(ch=mmi.getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return x;
}
typedef __gnu_pbds::tree<int,__gnu_pbds::null_type,std::less<int>,__gnu_pbds::rb_tree_tag,__gnu_pbds::tree_order_statistics_node_update> rbtree;
typedef long long int64;
const int N=1e5+1,M=1e5,A=1e6+1,P=78499;
bool vis[A];
int a[N],n,m,p[P],fail[N];
inline void sieve() {
for(register int i=2;i<A;i++) {
if(vis[i]) continue;
p[++p[0]]=i;
for(register int j=i*2;j<A;j+=i) {
vis[j]=true;
}
}
}
rbtree set[P];
class FenwickTree {
private:
int64 val[N];
int lowbit(const int &x) const {
return x&-x;
}
public:
void modify(int p,const int &x) {
for(;p<=n;p+=lowbit(p)) val[p]+=x;
}
int64 query(int p) const {
int64 ret=0;
for(;p;p-=lowbit(p)) ret+=val[p];
return ret;
}
int64 query(const int &l,const int &r) const {
return query(r)-query(l-1);
}
};
FenwickTree t;
int main() {
sieve();
n=getint(),m=getint();
for(register int i=1;i<=n;i++) {
a[i]=getint();
t.modify(i,a[i]);
int tmp=a[i];
for(register int j=1;p[j]*p[j]<=tmp;j++) {
if(tmp%p[j]==0) {
set[j].insert(i);
while(tmp%p[j]==0) {
tmp/=p[j];
}
}
}
if(tmp!=1) {
set[std::lower_bound(&p[1],&p[p[0]]+1,tmp)-p].insert(i);
}
}
for(register int tim=1;tim<=m;tim++) {
const int opt=getint();
const int l=getint(),r=getint();
if(opt==1) {
const int tmp=getint();
int v=tmp;
for(register int i=1;p[i]*p[i]<=v;i++) {
if(v%p[i]!=0) continue;
int sum=1;
for(;v%p[i]==0;v/=p[i]) sum*=p[i];
const rbtree::iterator begin=set[i].lower_bound(l);
const rbtree::iterator end=set[i].upper_bound(r);
if(v==1) {
for(rbtree::iterator it=begin;it!=end;) {
if(a[*it]%p[i]!=0) {
set[i].erase(it++);
continue;
}
if(a[*it]%tmp==0&&fail[*it]<tim) {
t.modify(*it,a[*it]/tmp-a[*it]);
a[*it]/=tmp;
}
it++;
}
continue;
}
for(rbtree::iterator it=begin;it!=end;) {
if(a[*it]%p[i]!=0) {
set[i].erase(it++);
continue;
}
if(a[*it]%sum!=0) fail[*it]=tim;
it++;
}
}
if(v!=1) {
const int i=std::lower_bound(&p[1],&p[p[0]]+1,v)-p;
const rbtree::iterator begin=set[i].lower_bound(l);
const rbtree::iterator end=set[i].upper_bound(r);
for(rbtree::iterator it=begin;it!=end;) {
if(a[*it]%v!=0) {
set[i].erase(it++);
continue;
}
if(a[*it]%tmp==0&&fail[*it]<tim) {
t.modify(*it,a[*it]/tmp-a[*it]);
a[*it]/=tmp;
}
it++;
}
}
}
if(opt==2) {
printf("%lld\n",t.query(l,r));
}
}
return 0;
}

思路2:

本题是某年CCF-CSP原题,用当时原题的数据,上面的做法是过不了的。

考虑不要按照质因数维护set,将操作离线,对于每个\(v\)维护一个set。然后卡卡常就过了。

源代码:

#pragma GCC optimize("Ofast")
#pragma GCC optimize("unroll-loops")
#pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
#include<cstdio>
#include<cctype>
#include<vector>
#include<climits>
#include<sys/mman.h>
#include<sys/stat.h>
#include<ext/pb_ds/assoc_container.hpp>
class MMapInput {
private:
char *buf,*p;
int size;
public:
MMapInput() {
register int fd=fileno(stdin);
struct stat sb;
fstat(fd,&sb);
size=sb.st_size;
buf=reinterpret_cast<char*>(mmap(0,size,PROT_READ,MAP_PRIVATE,fileno(stdin),0));
p=buf;
}
char getchar() {
return (p==buf+size||*p==EOF)?EOF:*p++;
}
};
MMapInput mmi;
inline int getint() {
register char ch;
while(!isdigit(ch=mmi.getchar()));
register int x=ch^'0';
while(isdigit(ch=mmi.getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return x;
}
typedef long long int64;
const int N=1e5+1,M=1e5,A=5e5+1;
int n,a[N];
struct Opt {
int type,l,r,v;
};
Opt o[M];
std::set<int> set[M];
std::vector<int> vec;
int id[A];
class FenwickTree {
private:
int64 val[N];
int lowbit(const int &x) const {
return x&-x;
}
public:
void modify(int p,const int &x) {
for(;p<=n;p+=lowbit(p)) val[p]+=x;
}
int64 query(int p) const {
int64 ret=0;
for(;p;p-=lowbit(p)) ret+=val[p];
return ret;
}
int64 query(const int &l,const int &r) const {
return query(r)-query(l-1);
}
};
FenwickTree t;
int main() {
n=getint();
const int m=getint();
for(register int i=1;i<=n;i++) {
t.modify(i,a[i]=getint());
}
for(register int i=0;i<m;i++) {
o[i].type=getint();
o[i].l=getint();
o[i].r=getint();
if(o[i].type==1) {
o[i].v=getint();
if(o[i].v==1) continue;
vec.push_back(o[i].v);
}
}
std::sort(vec.begin(),vec.end());
vec.resize(std::unique(vec.begin(),vec.end())-vec.begin());
std::fill(&id[0],&id[A],-1);
for(register unsigned i=0;i<vec.size();i++) id[vec[i]]=i;
for(register int i=1;i<=n;i++) {
for(register int j=1;j*j<=a[i];j++) {
if(a[i]%j!=0) continue;
if(id[j]!=-1) set[id[j]].insert(i);
if(id[a[i]/j]!=-1) set[id[a[i]/j]].insert(i);
}
}
for(register int i=0;i<m;i++) {
const int &opt=o[i].type,&l=o[i].l,&r=o[i].r;
if(opt==1) {
const int &v=o[i].v;
if(v==1) continue;
const std::set<int>::iterator begin=set[id[v]].lower_bound(l);
const std::set<int>::iterator end=set[id[v]].upper_bound(r);
for(register std::set<int>::iterator i=begin;i!=end;) {
if(a[*i]%v!=0) {
set[id[v]].erase(i++);
continue;
}
t.modify(*i,a[*i]/v-a[*i]);
a[*i]/=v;
if(a[*i]%v!=0) {
set[id[v]].erase(i++);
continue;
}
++i;
}
}
if(opt==2) {
printf("%lld\n",t.query(l,r));
}
}
return 0;
}

[洛谷P3987]我永远喜欢珂朵莉~的更多相关文章

  1. 洛谷P3987 我永远喜欢珂朵莉~(set 树状数组)

    题意 题目链接 Sol 不会卡常,自愧不如.下面的代码只有66分.我实在懒得手写平衡树了.. 思路比较直观:拿个set维护每个数出现的位置,再写个线段树维护区间和 #include<bits/s ...

  2. [转]我的数据结构不可能这么可爱!——珂朵莉树(ODT)详解

    参考资料: Chtholly Tree (珂朵莉树) (应某毒瘤要求,删除链接,需要者自行去Bilibili搜索) 毒瘤数据结构之珂朵莉树 在全是珂学家的珂谷,你却不知道珂朵莉树?来跟诗乃一起学习珂朵 ...

  3. 洛谷AT2342 Train Service Planning(思维,动态规划,珂朵莉树)

    洛谷题目传送门 神仙思维题还是要写点东西才好. 建立数学模型 这种很抽象的东西没有式子描述一下显然是下不了手的. 因为任何位置都以\(k\)为周期,所以我们只用关心一个周期,也就是以下数都在膜\(k\ ...

  4. 洛谷P4344 [SHOI2015]脑洞治疗仪(珂朵莉树)

    传送门 看到区间推倒……推平就想到珂朵莉树 挖脑洞直接assign,填坑先数一遍再assign再暴力填,数数的话暴力数 //minamoto #include<iostream> #inc ...

  5. 洛谷P2787 语文1(chin1)- 理理思维(珂朵莉树)

    传送门 一看到区间推倒……推平操作就想到珂朵莉树 区间推平直接assign,查询暴力,排序的话开一个桶统计,然后一个字母一个字母加就好了 开桶统计的时候忘了保存原来的左指针然后挂了233 //mina ...

  6. 洛谷P2082 区间覆盖(加强版)(珂朵莉树)

    传送门 虽然是黄题而且还是一波离散就能解决的东西 然而珂朵莉树还是很好用 相当于一开始区间全为0,然后每一次区间赋值,问最后总权值 珂朵莉树搞一搞就好了 //minamoto #include< ...

  7. 洛谷P2572 [SCOI2010]序列操作(珂朵莉树)

    传送门 珂朵莉树是个吼东西啊 这题线段树代码4k起步……珂朵莉树只要2k…… 虽然因为这题数据不随机所以珂朵莉树的复杂度实际上是错的…… 然而能过就行对不对…… (不过要是到时候noip我还真不敢打… ...

  8. 洛谷$P2572\ [SCOI2010]$ 序列操作 线段树/珂朵莉树

    正解:线段树/珂朵莉树 解题报告: 传送门$w$ 本来是想写线段树的,,,然后神仙$tt$跟我港可以用珂朵莉所以决定顺便学下珂朵莉趴$QwQ$ 还是先写线段树做法$QwQ$? 操作一二三四都很$eas ...

  9. 洛谷 P5350 序列 珂朵莉树

    题目描述 分析 操作一.二.三为珂朵莉树的基本操作,操作四.五.六稍作转化即可 不会珂朵莉树请移步至这里 求和操作 把每一段区间分别取出,暴力相加 ll qh(ll l,ll r){ it2=Spli ...

随机推荐

  1. 【译】第二篇 SQL Server代理作业步骤和子系统

    本篇文章是SQL Server代理系列的第二篇,详细内容请参考原文. SQL Server代理作业由一系列的一个或多个作业步骤组成.一个作业步骤分配给一个特定的作业子系统(确定作业步骤去完成的工作). ...

  2. BBScan — 一个信息泄漏批量扫描脚本

    github:https://github.com/lijiejie/BBScan 有些朋友手上有几十万甚至上百万个域名,比如,乌云所有厂商的子域名. 如果把这30万个域名全部扔给wvs,APPsca ...

  3. 在html页面中引入公共的头部和底部

    参考链接: http://www.cnblogs.com/jason-star/p/3345225.html http://blog.csdn.net/jsxzzliang/article/detai ...

  4. Python语言库pyttsx3

    这是一个文字转语音的python模块. 1. macos下安装的时候出现问题:  后来发现,Foundation, AppKit, PyObjCTools都不存在,主要原因是缺少依赖模块pyobjc, ...

  5. Unity 软件使用事项

    打开旧版工程 目前发现两种方式来触发升级程序: 1.Unity软件启动时选择旧版工程,触发更新 2.直接打开旧版工程的场景文件,触发更新   在使用中发现一种错误做法,不知道是不是共性问题,在此先记录 ...

  6. LeetCode312. Burst Balloons

    Given n balloons, indexed from 0 to n-1. Each balloon is painted with a number on it represented by ...

  7. PHP获取机器mac代码

    废话不多话,直接上代码 <?php class GetMac { public $result = array(); public $macAddrs = array(); //所有mac地址 ...

  8. Web前端开发最佳实践(4):在页面中添加必要的meta信息

    meta标签放置在HTML页面的head中,主要用于标识网站.其中基本上包含了网站的一些描述信息,例如,简介.作者等.这些信息有助于搜索引擎更准确地识别网页的内容,也有助于第三方工具抓取网站基本信息. ...

  9. 4. CMake 系列 - 项目添加编译选项

    目录 1. 项目目录结构 2. 相关代码 2.1 add 模块 2.2 sub 模块 2.3 example 模块 2.4 顶层 CMakeLists.txt 3. 配置&编译 1. 项目目录 ...

  10. 查看loadrunner代码行号

    运行前报错,如Syntax error on line 133 near ";"那么如何查看代码的行号呢?解决方法:看代码行号时,直接将鼠标在代码的某处单击,在窗体的最下方右侧能看 ...