题目链接##

Mychael vs Kid

题解##

先说说这题的由来及前身

前身##

首先有一个很经典的题目:

维护区间加,查询区间\(gcd\)

如果强行用线段树维护的话,区间加之后就没法直接确定当前区间的\(gcd\),不可直接维护

这个时候就用到了\(gcd\)的一个性质:

\[(a,b) = (a - b,b)
\]

三个的\(gcd\)也是符合的:

\[(a,b,c) = (a,b,c - b) = (a,b - a,c - b)
\]

同样可以推广出\(n\)个的情况

\[gcd\{a_i\} = gcd(a_1,gcd\{a_j - a_{j - 1}\}) \qquad i \in [1,n] \quad j \in [2,n]
\]

也就是说,区间差分了之后,我们要求原区间\(gcd[l,r]\),就相当于求\(a_l\)和差分后区间\([l + 1,r]\)的\(gcd\)

所以我们只需维护差分区间和区间每个位置的值即可

维护区间值用线段树一点问题都没有

在加法下维护差分区间,一个区间加上一个数,仅影响区间端点的值,所以单点修改即可

所以我们开两棵线段树,一棵维护权值,一棵维护差值的\(gcd\),即可\(O(nlog^2n)\)解决这道题

本题##

本题多加入了一个区间乘法操作

本来想\(yy\)分块的,可后来发现线段树依旧可写并且暴艹分块

思考一下发现,区间乘法后,区间差分值也乘上那个数,对应\(gcd\)也乘上一个数,可以使用线段树维护

至于端点的差值改变,只需取出端点的值计算出差值后单点加法修改即可

复杂度依旧是\(O(nlog^2n)\)

部分分##

前\(5\%\),咳,,,

对于\(n,m \le 300\),暴力求即可

对于没有操作\(2\)的,就是原版题目

对于\(n,m \le 3 \times 10^4\)的,可以考虑分块

std

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
#define ls (u << 1)
#define rs (u << 1 | 1)
using namespace std;
const int maxn = 100005,maxm = 100005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
LL n,m,A[maxn],D[maxn];
LL gcd(LL a,LL b){
if (a < 0) a = -a;
if (b < 0) b = -b;
return b ? gcd(b,a % b) : a;
}
struct Seg_Gcd{
LL Gcd[maxn << 2],tag[maxn << 2];
void upd(int u){
Gcd[u] = gcd(Gcd[ls],Gcd[rs]);
}
void pd(int u){
if (tag[u] != 1){
Gcd[ls] *= tag[u]; tag[ls] *= tag[u];
Gcd[rs] *= tag[u]; tag[rs] *= tag[u];
tag[u] = 1;
}
}
void build(int u,int l,int r){
tag[u] = 1;
if (l == r){
Gcd[u] = D[l];
return;
}
int mid = l + r >> 1;
build(ls,l,mid);
build(rs,mid + 1,r);
upd(u);
}
void Add(int u,int l,int r,int pos,int v){
if (l == r){Gcd[u] += v; return;}
pd(u);
int mid = l + r >> 1;
if (mid >= pos) Add(ls,l,mid,pos,v);
else Add(rs,mid + 1,r,pos,v);
upd(u);
}
void Mult(int u,int l,int r,int L,int R,int v){
if (l >= L && r <= R){Gcd[u] *= v; tag[u] *= v; return;}
pd(u);
int mid = l + r >> 1;
if (mid >= L) Mult(ls,l,mid,L,R,v);
if (mid < R) Mult(rs,mid + 1,r,L,R,v);
upd(u);
}
LL query(int u,int l,int r,int L,int R){
if (l >= L && r <= R) return Gcd[u];
pd(u);
int mid = l + r >> 1;
if (mid >= R) return query(ls,l,mid,L,R);
if (mid < L) return query(rs,mid + 1,r,L,R);
return gcd(query(ls,l,mid,L,R),query(rs,mid + 1,r,L,R));
}
}T2;
struct Seg{
LL val[maxn << 2],mult[maxn << 2],add[maxn << 2];
void pd(int u){
if (mult[u] != 1){
val[ls] *= mult[u]; mult[ls] *= mult[u]; add[ls] *= mult[u];
val[rs] *= mult[u]; mult[rs] *= mult[u]; add[rs] *= mult[u];
mult[u] = 1;
}
if (add[u]){
val[ls] += add[u]; add[ls] += add[u];
val[rs] += add[u]; add[rs] += add[u];
add[u] = 0;
}
}
void build(int u,int l,int r){
add[u] = 0; mult[u] = 1;
if (l == r){
val[u] = A[l];
return;
}
int mid = l + r >> 1;
build(ls,l,mid);
build(rs,mid + 1,r);
}
void Add(int u,int l,int r,int L,int R,int v){
if (l >= L && r <= R){val[u] += v; add[u] += v; return;}
pd(u);
int mid = l + r >> 1;
if (mid >= L) Add(ls,l,mid,L,R,v);
if (mid < R) Add(rs,mid + 1,r,L,R,v);
}
void Mult(int u,int l,int r,int L,int R,int v){
if (l >= L && r <= R){val[u] *= v; add[u] *= v; mult[u] *= v; return;}
pd(u);
int mid = l + r >> 1;
if (mid >= L) Mult(ls,l,mid,L,R,v);
if (mid < R) Mult(rs,mid + 1,r,L,R,v);
}
LL query(int u,int l,int r,int pos){
if (l == r) return val[u];
pd(u);
int mid = l + r >> 1;
if (mid >= pos) return query(ls,l,mid,pos);
return query(rs,mid + 1,r,pos);
}
}T1;
int main(){
n = read(); m = read();
for (int i = 1; i <= n; i++){
A[i] = read();
D[i] = A[i] - A[i - 1];
}
T1.build(1,1,n);
T2.build(1,1,n);
LL opt,l,r,v,t1,t2;
while (m--){
opt = read(); l = read(); r = read();
if (opt == 1){
v = read();
T1.Add(1,1,n,l,r,v);
T2.Add(1,1,n,l,v);
if (r + 1 <= n) T2.Add(1,1,n,r + 1,-v);
}
else if (opt == 2){
v = read();
t1 = T1.query(1,1,n,l);
if (r + 1 <= n) t2 = T1.query(1,1,n,r);
if (l < r) T2.Mult(1,1,n,l + 1,r,v);
T2.Add(1,1,n,l,t1 * v - t1);
if (r + 1 <= n) T2.Add(1,1,n,r + 1,t2 - t2 * v);
T1.Mult(1,1,n,l,r,v);
}
else {
if (l < r) printf("%lld\n",gcd(T1.query(1,1,n,l),T2.query(1,1,n,l + 1,r)));
else printf("%lld\n",T1.query(1,1,n,l));
}
}
return 0;
}

Myhchael原创题系列 Mychael vs Kid 【题解】的更多相关文章

  1. Newtonsoft.Json C# Json序列化和反序列化工具的使用、类型方法大全 C# 算法题系列(二) 各位相加、整数反转、回文数、罗马数字转整数 C# 算法题系列(一) 两数之和、无重复字符的最长子串 DateTime Tips c#发送邮件,可发送多个附件 MVC图片上传详解

    Newtonsoft.Json C# Json序列化和反序列化工具的使用.类型方法大全   Newtonsoft.Json Newtonsoft.Json 是.Net平台操作Json的工具,他的介绍就 ...

  2. NOI题库 1768最大子矩阵 题解

    NOI题库 1768最大子矩阵  题解     总时间限制: 1000ms 内存限制: 65536kB   描述   已知矩阵的大小定义为矩阵中所有元素的和.给定一个矩阵,你的任务是找到最大的非空(大 ...

  3. hdu5017:补题系列之西安网络赛1011

    补题系列之西安网络赛1011 题目大意:给定一个椭球: 求它到原点的最短距离. 思路: 对于一个椭球的标准方程 x^2/a^2 + y^2/b^2 +z^2/c^2=1 来说,它到原点的最短距离即为m ...

  4. nyoj 1208——水题系列——————【dp】

    水题系列 时间限制:1000 ms  |  内存限制:65535 KB 难度:2   描述     给你一个有向图,每条边都有一定的权值,现在让你从图中的任意一点出发,每次走的边的权值必须必上一次的权 ...

  5. Mychael原创题 洛谷T23923 Mychaelの水题 【题解】

    原题链接 题目大意: 有来自三个地区的人各a,b,c位,他们排成了一排.请问有多少种不同类型的排法,使得相邻的人都来自不同的地区 \(a,b,c<=200\) 答案取模 题解 弱弱的标程解法 设 ...

  6. NOI题库刷题日志 (贪心篇题解)

    这段时间在NOI题库上刷了刷题,来写点心得和题解 一.寻找平面上的极大点 2704:寻找平面上的极大点 总时间限制:  1000ms  内存限制:  65536kB 描述 在一个平面上,如果有两个点( ...

  7. C#版 - PAT乙级(Basic Level)真题 之 1021.个位数统计 - 题解

    版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. C#版 - P ...

  8. LeetCode算法扫题系列19

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9104677.html LeetCode算法第19题(难度:中等) 题目:给定一个链表,删 ...

  9. LeetCode算法扫题系列83

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9104582.html LeetCode算法第83题(难度:简单) 题目:给定一个排序链表 ...

随机推荐

  1. 「日常训练」Known Notation(ZOJ-3829)

    题意与分析 题意是这样的:给一个字符串,字符串中只包含数字和运算符'*'.现在问字符串是不是一个合法的逆波兰式(后缀表达式).已知逆波兰式的空格消除,也就是说123可以看成123也可以看成1和23.如 ...

  2. 苏醒的巨人----CSRF

    一.CSRF 跨站请求伪造(Cross-Site Request Forgery,CSRF)是指利用 受害者尚未失效的身份认证信息(cookie.会话等),诱骗其点 击恶意链接或者访问包含攻击代码的页 ...

  3. 爬虫1.5-ajax数据爬取

    目录 爬虫-ajax数据爬取 1. ajax数据 2. selenium+chromedriver知识准备 3. selenium+chromedriver实战拉勾网爬虫代码 爬虫-ajax数据爬取 ...

  4. java字符转义

    之前对java字符转义这一块稍作了解,在这里理理自己主观浅显的理解 这里会谈谈字符编码的是另一种问题和转义没有关系 以下面代码做分析 System.out.println("a". ...

  5. 菜鸟之路——机器学习之决策树个人理解及Python实现

    最近开始学习机器学习,以下会记录我学习中遇到的问题以及我个人的理解 决策树算法,网上很多介绍,在这不复制粘贴.下面解释几个关键词就好. 信息熵(entropy):就是信息不确定性的多少 H(x)=-Σ ...

  6. 使用深度学习来破解 captcha 验证码(转)

    使用深度学习来破解 captcha 验证码 本项目会通过 Keras 搭建一个深度卷积神经网络来识别 captcha 验证码,建议使用显卡来运行该项目. 下面的可视化代码都是在 jupyter not ...

  7. POJ 2631 Roads in the North(求树的直径,两次遍历 or 树DP)

    题目链接:http://poj.org/problem?id=2631 Description Building and maintaining roads among communities in ...

  8. rsync+inotify实现实时同步,自动触发同步文件

    本文参考来自:http://chocolee.blog.51cto.com/8158455/1400596 我的需求和他的略有不同,同时做了一下更改,如下: 需求:两台机器相互为主备,搭建相同的两个服 ...

  9. 20162328蔡文琛week03

    学号 2006-2007-2 <程序设计与数据结构>第X周学习总结 教材学习内容总结 在第三章,我学习到了更多有关于java.util包的知识.了解了多个引用变量可以指向同一个对象.而且J ...

  10. LintCode-67.二叉树的中序遍历

    二叉树的中序遍历 给出一棵二叉树,返回其中序遍历. 样例 给出一棵二叉树 {1,#,2,3}, 返回 [1,3,2]. 挑战 你能使用非递归实现么? 标签 递归 二叉树 二叉树遍历 code /** ...