线段树 + 区间更新 ----- HDU 4902 : Nice boat
Nice boat
Time Limit: 30000/15000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)
Total Submission(s): 968 Accepted Submission(s): 441
Let us continue our story, z*p(actually you) defeat the 'MengMengDa' party's leader, and the 'MengMengDa' party dissolved. z*p becomes the most famous guy among the princess's knight party.
One day, the people in the party find that z*p has died. As what he has done in the past, people just say 'Oh, what a nice boat' and don't care about why he died.
Since then, many people died but no one knows why and everyone is fine about that. Meanwhile, the devil sends her knight to challenge you with Algorithm contest.
There is a hard data structure problem in the contest:
There are n numbers a_1,a_2,...,a_n on a line, everytime you can change every number in a segment [l,r] into a number x(type 1), or change every number a_i in a segment [l,r] which is bigger than x to gcd(a_i,x) (type 2).
You should output the final sequence.
For each test case, the first line contains a integers n.
The next line contains n integers a_1,a_2,...,a_n separated by a single space.
The next line contains an integer Q, denoting the number of the operations.
The next Q line contains 4 integers t,l,r,x. t denotes the operation type.
T<=2,n,Q<=100000
a_i,x >=0
a_i,x is in the range of int32(C++)
Please output a single more space after end of the sequence
10
【题目大意】
给你一串数字,有两个操作,1表示把一段区间内的数变成x,2表示把一段区间内的数如果这个数大于x则变为这个数与x的最小公约数,否则不变。最后输出变化后的一组数。
【题目分析】
这题和其他线段树有一些区别,这题是在所有的处理结束后才全部输出。
我们在每个结点中加一个flag标记该区间内的数字是否都是同一个,如果区间是同一个数的话我们就可以进行批处理,这将会大大降低时间复杂度。
再用一个temp来存储该结点的val,然后在pushdown函数将temp的值一层一层的传递下去。
这里的temp既起到了该结点是否已经向下更新的作用(相当于lazy),又起到了记录子节点需要更新的值的作用。
temp只有在val的值改变的时候才改变。
//Memory Time
// 5376K 651MS
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<vector>
#include<queue>
#include<stack>
#include<iomanip>
#include<string>
#include<climits>
#include<cmath>
#define MAX 100100
#define LL long long
using namespace std;
int n,m;
int ans;
int num[MAX];
struct Tree
{
int l,r;
bool flag;
int val,temp;
};
Tree tree[MAX<<2]; int gcd(int x,int y)
{
return y?gcd(y,x%y):x;
} void pushup(int x)
{
int tmp=x<<1;
tree[x].val=max(tree[tmp].val,tree[tmp+1].val);
tree[x].flag=(tree[tmp].val==tree[tmp+1].val&&tree[tmp].flag&&tree[tmp+1].flag);
} void pushdown(int x)
{
if(tree[x].temp==-1)return;
int tmp=x<<1;
int mid=(tree[x].l+tree[x].r)>>1;
tree[tmp].val=tree[tmp+1].val=tree[tmp].temp=tree[tmp+1].temp=tree[x].temp;
tree[x].temp=-1;
} void build(int l,int r,int x)
{
tree[x].flag=0;
tree[x].temp=-1;
tree[x].l=l,tree[x].r=r;
if(l==r)
{
scanf("%d",&tree[x].val);
tree[x].flag=1;
return;
}
int tmp=x<<1;
int mid=(l+r)>>1;
build(l,mid,tmp);
build(mid+1,r,tmp+1);
pushup(x);
} void update(int l,int r,int num,int x)
{
if(r<tree[x].l||l>tree[x].r)return;
if(l<=tree[x].l&&r>=tree[x].r)
{
tree[x].flag=1;
tree[x].val=num;
tree[x].temp=num;
return;
}
pushdown(x);
int tmp=x<<1;
int mid=(tree[x].l+tree[x].r)>>1;
if(r<=mid)
update(l,r,num,tmp);
else if(l>mid)
update(l,r,num,tmp+1);
else
{
update(l,mid,num,tmp);
update(mid+1,r,num,tmp+1);
}
pushup(x);
} void change(int l,int r,int num,int x)
{
if(r<tree[x].l||l>tree[x].r)return;
if(tree[x].flag&&tree[x].val<=num)return;
if(l<=tree[x].l&&r>=tree[x].r&&tree[x].flag)
{
tree[x].val=gcd(tree[x].val,num);
tree[x].temp=tree[x].val;
return;
}
pushdown(x);
int tmp=x<<1;
int mid=(tree[x].l+tree[x].r)>>1;
if(r<=mid)
change(l,r,num,tmp);
else if(l>mid)
change(l,r,num,tmp+1);
else
{
change(l,mid,num,tmp);
change(mid+1,r,num,tmp+1);
}
pushup(x);
} void query(int l,int r,int k,int x)
{
if(k<tree[x].l||k>tree[x].r)return;
if(tree[x].flag)
{
ans=tree[x].val;
return;
}
pushdown(x);
int tmp=x<<1;
int mid=(tree[x].l+tree[x].r)>>1;
if(k<=mid)
query(l,mid,k,tmp);
else
query(mid+1,r,k,tmp+1);
} int main()
{
int T;
int t,l,r,num;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
build(1,n,1);
scanf("%d",&m);
while(m--)
{
scanf("%d %d %d %d",&t,&l,&r,&num);
if(t==1)
update(l,r,num,1);
else
change(l,r,num,1);
}
for(int i=1;i<=n;i++)
{
ans=0;
query(1,n,i,1);
printf("%d ",ans);
}
puts("");
}
return 0;
}
【解法二】
这题不知道出题人怎么搞的,数据弱爆了,直接暴力还比线段树还快。
当然也不是单纯的暴力,从后往前搜,用一个数组prime来记录需要求gcd的值,遇到type=1就退出,最后来求一下gcd即可。
//Memory Time
// 2232K 250MS
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<vector>
#include<queue>
#include<stack>
#include<iomanip>
#include<string>
#include<climits>
#include<cmath>
#define MAX 100100
#define LL long long
using namespace std;
int T,n,q;
int num[MAX];
int t[MAX],l[MAX],r[MAX],x[MAX];
int prime[MAX];
int gcd(int x,int y)
{
return y?gcd(y,x%y):x;
}
int main()
{
// freopen("cin.txt","r",stdin);
// freopen("cout.txt","w",stdout);
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
for(int i=1;i<=n;++i)
scanf("%d",&num[i]);
scanf("%d",&q);
for(int i=1;i<=q;++i)
scanf("%d %d %d %d",&t[i],&l[i],&r[i],&x[i]);
for(int i=1;i<=n;++i)
{
int tp=num[i];
int index=-1;
for(int j=q;j>=1;--j)
{
if(i>=l[j]&&i<=r[j])
{
if(t[j]==1)
{
tp=x[j];
break;
}
else prime[++index]=x[j];
}
}
for(int j=index;j>=0;--j)
{
if(tp>prime[j])
tp=gcd(tp,prime[j]);
}
printf("%d ",tp);
}
puts("");
}
return 0;
}
线段树 + 区间更新 ----- HDU 4902 : Nice boat的更多相关文章
- 线段树-区间更新-HDU 1689
#include <iostream> #include <cstdio> #include <string> #include <cstring> # ...
- HDU 4902 Nice boat 2014杭电多校训练赛第四场F题(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4902 解题报告:输入一个序列,然后有q次操作,操作有两种,第一种是把区间 (l,r) 变成x,第二种是 ...
- HDU 5023 A Corrupt Mayor's Performance Art(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5023 解题报告:一面墙长度为n,有N个单元,每个单元编号从1到n,墙的初始的颜色是2,一共有30种颜色 ...
- hdu 4031 attack 线段树区间更新
Attack Time Limit: 5000/3000 MS (Java/Others) Memory Limit: 65768/65768 K (Java/Others)Total Subm ...
- hdu 3966(树链剖分+线段树区间更新)
传送门:Problem 3966 https://www.cnblogs.com/violet-acmer/p/9711441.html 学习资料: [1]线段树区间更新:https://blog.c ...
- HDU.1556 Color the ball (线段树 区间更新 单点查询)
HDU.1556 Color the ball (线段树 区间更新 单点查询) 题意分析 注意一下pushdown 和 pushup 模板类的题还真不能自己套啊,手写一遍才行 代码总览 #includ ...
- HDU 1556 Color the ball(线段树区间更新)
Color the ball 我真的该认真的复习一下以前没懂的知识了,今天看了一下线段树,以前只会用模板,现在看懂了之后,发现还有这么多巧妙的地方,好厉害啊 所以就应该尽量搞懂 弄明白每个知识点 [题 ...
- HDU 1698 线段树 区间更新求和
一开始这条链子全都是1 #include<stdio.h> #include<string.h> #include<algorithm> #include<m ...
- (简单) HDU 1698 Just a Hook , 线段树+区间更新。
Description: In the game of DotA, Pudge’s meat hook is actually the most horrible thing for most of ...
随机推荐
- RocketMq 学习记录
最近因为工作需求,领导让我安装一下RocketMQ 这里简单记录一下 这里我的操作系统是centos 6.5 64位 我们看一下官网的RocketMQ安装要求 Prerequisite The fol ...
- MYSQL拒绝访问:not allowed to connect解决方法
分享下MYSQL拒绝访问报错not allowed to connect的解决方法. 可以在其它任何的主机上以root身份登录 mysql报如下错误,截取部分, message from server ...
- eclipse 安装properties编辑器,显示中文
如图添加,地址为: propedit.sourceforge.jp/eclipse/updates/ 选择红框,只安装这个即可 然后一直安装,再接受同意,最后重启eclipse就安装好了 重启后发现文 ...
- android category
本章节翻译自<Beginning-Android-4-Application-Development>,如有翻译不当的地方,敬请指出. 原书购买地址http://www.amazon.co ...
- Ubuntu下golang环境搭建
参考官方文档:https://golang.org/doc/editors.html 1.安装golang apt install golang 2.配置GOPATH mkdir ~/gopath v ...
- Vue之vuex实现简易计算器
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Leetcode:Scramble String 解题报告
Scramble String Given a string s1, we may represent it as a binary tree by partitioning it to two no ...
- Django admin 继承user表后密码为明文,继承UserAdmin,重写其方法
Django用户继承AbstractUser后密码为明文 其实本不应该有这个问题,却花了我很久的时间,因为还是初学阶段. 造成这个原因是因为在admin注册的生活没有指定Admin 在app的admi ...
- 【Linux】crontab 定时启动sh
crontab -e 20点59分启动脚本 59 20 * * * sh /home/fzuir/xingye4crawl/endXingYe4Crawl.sh >/home/fzuir/xin ...
- AIX上打包排除某些文件/文件夹
Syntax X/Open Standards: tar {-c|-r|-t|-u|-x} [-B] [ -d ] [ -E ] [ -F ] [-h ] [ -i ] [ -l ] [ -m ] [ ...