HDU - 4578 线段树+三重操作
这道题自己写了很久,还是没写出来,也看了很多题解,感觉多数还是看的迷迷糊糊,最后面看到一篇大佬的才感觉恍然大悟。
先上一篇大佬的题解:https://blog.csdn.net/aqa2037299560/article/details/82872866?tdsourcetag=s_pcqq_aiomsg (既简单又高效 代码还短!%%%)
先说下题意:
就是给你n个数,每个数的初始值都是为0
然后给你m个操作
每个操作有 4 个数 op x y c
当 op==1 的时候,把 x到y 范围内的数 都 加上 c
当 op==2 的时候,把 x到y 范围内的数 都 乘以 c
当 op==3 的时候,把 x到y 范围内的数 都 等于 c
当 op==4 的时候,把 x到y 范围内的 每一个数 的 c 次方的和 输出(注意,当op等于4的时候,c的范围为1~3)
下面说说思路:
关键就是在于这个 懒惰值的传递,和 区间的每一个值是否都相等的问题
一.先说说树的数据
这个大家参考下就可以,实现的方法有很多,不一定需要这么写,把这个先放上来是便于理解。(我这么写是因为我太菜了)
struct Data {
ll l, r, val;//分别是左边界,右边界,懒惰值(也是该区间每一个叶节点的值)
bool dif;//判断区间内每一个区间是否的相同
}tree[M << ];
二.再说这个区间值都相等
我们可以知道,如果区间内的每一个值都相等,那么我们只要 求 其中一个的值的c次方,然后把该数乘以(右边界 减去 左边界 再加 1 ),便是该区间值的次方总和了
如果该区间的每一个值不相等,那么我们必须接着向下探索,直到 找到 一个 区间内的每一个值都相等 的区间,最坏的情况也就是找到叶节点。
三.懒惰值的传递
如果这个区间的每一个值都相等,那么它的左右子区间肯定也都是相等的。
如果这个区间不是全等区间,那么我们就没必要传递懒惰值,因为你这个区间每一个值不一定相等 ; 但是如果是全等区间,就要传递懒惰值。
如果我们在更新数据的过程中,需要用到传递懒惰值,那么肯定是要修改这个区间的某一个子区间,所以传递后,这个区间肯定不会再是全等区间
四.数据的更新
我们在更新完值后,肯定也需要更新区间是否相等的信息
有三种情况:
1.如果该区间的左右子区间 都不是 全等区间的话,那这个区间肯定也 不是 全等区间
2.如果该区间的左右子区间 都是 全等区间, 但是它们的 叶节点的值都 不相等,那么这个区间肯定 也不是 全等区间
3.如果该区间的左右区间 都是 全等区间,并且 它们的 叶节点的值都 全等,那么这个区间 肯定是 全等区间
下面上代码:
#include <algorithm>
#include <iostream>
#include<sstream>
#include<iterator>
#include<cstring>
#include<string>
#include<cstdio>
#include<cctype>
#include<vector>
#include<deque>
#include<queue>
#include<stack>
#include<map>
#include<set> #define M 100005
#define mod 10007
#define inf 0x3f3f3f3f
#define left k<<1
#define right k<<1|1
#define ms(a,b) memset(a,b,sizeof(a))
typedef long long ll;
using namespace std; struct Data {
ll l, r, val; // 分别是左边界,右边界,懒惰值(也是该区间每一个叶节点的值
bool dif;//判断区间内每一个区间是否的相同
}tree[M << ]; ll ans, temp;
ll op, c, p;
int x, y, n, m; void built(int l, int r, int k) {
tree[k].l = l, tree[k].r = r, tree[k].val = , tree[k].dif = true;//初始化,每一个区间肯定都是全等的
if (l == r)return;
int mid = (l + r) >> ;
built(l, mid, left);
built(mid + , r, right);
} void down(int k) {
if (tree[k].l == tree[k].r)return;//如果这个数就是子区间那么没必要往下传递了
tree[k].dif = false;//传递后可能不是全等区间了
tree[left].val = tree[right].val = tree[k].val;//传递懒惰值
tree[left].dif = tree[right].dif = true;//暂时都是全等区间
} void updata(int k) {
if (tree[k].l >= x && tree[k].r <= y && tree[k].dif) {//在要操作的区间范围内,并且是全等区间就可以直接操作
if (op == ) {//操作一,加上c
tree[k].val = (tree[k].val + c) % mod;
}
else if (op == ) {//操纵二,乘以c
tree[k].val = (tree[k].val * c) % mod;
}
else {//操作三,等于c
tree[k].val = c % mod;
}
return;
}
if (tree[k].dif)down(k);//如果该区间是全等区间,那么就要传递懒惰值,并且传递后该区间肯定是 不全等区间
int mid = (tree[k].l + tree[k].r) >> ; if (x <= mid)updata(left);
if (y > mid)updata(right);
//传递后三种情况分析更新
if (!tree[left].dif || !tree[right].dif)tree[k].dif = false;
else {
if (tree[left].val != tree[right].val)tree[k].dif = false;
else {
tree[k].dif = true;
tree[k].val = tree[left].val;
}
}
} void query(int k) {
if (tree[k].l >= x && tree[k].r <= y && tree[k].dif) {//如果是全等区间,那么每一个数的值都相等
temp = pow(tree[k].val, c);
temp *= (tree[k].r - tree[k].l + );
ans = (ans + temp) % mod;
return;
}
if (tree[k].dif)down(k);//如果该区间是全等区间,那么就要传递懒惰值,并且传递后该区间肯定是 ”全等区间“
int mid = (tree[k].l + tree[k].r) >> ;
if (x <= mid)query(left);
if (y > mid)query(right);
} int main() {
while (scanf("%d%d", &n, &m) != EOF && n && m) {
built(, n, );
for (int i = ; i <= m; i++) {
scanf("%d%d%d%d", &op, &x, &y, &c);
if (op == ) {
ans = ;
query();
cout << ans << endl;
}
else {
updata();
}
}
}
return ;
}
HDU - 4578 线段树+三重操作的更多相关文章
- hdu 4578 线段树(标记处理)
Transformation Time Limit: 15000/8000 MS (Java/Others) Memory Limit: 65535/65536 K (Java/Others) ...
- hdu 2871 线段树(各种操作)
Memory Control Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...
- HDU 4578 线段树玄学算法?
Transformation 题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=4578 Problem Description Yuanfang is p ...
- HDU 4578 线段树复杂题
题目大意: 题意:有一个序列,有四种操作: 1:区间[l,r]内的数全部加c. 2:区间[l,r]内的数全部乘c. 3:区间[l,r]内的数全部初始为c. 4:询问区间[l,r]内所有数的P次方之和. ...
- hdu 4578 线段树 ****
链接:点我 1
- K - Transformation HDU - 4578 线段树经典题(好题)
题意:区间 加 变成定值 乘 区间查询:和 平方和 立方和 思路:超级超级超级麻烦的一道题 设3个Lazy 标记分别为 change 改变mul乘 add加 优先度change>m ...
- hdu 3436 线段树 一顿操作
Queue-jumpers Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- hdu 3974 线段树 将树弄到区间上
Assign the task Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- hdu 3397 线段树双标记
Sequence operation Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
随机推荐
- 远程控制使用kill软件映射内网进行远程控制(9.28 第十四天)
1.能ping通IP情况下远程控制 设置kill软件中的端口.密码.上线列表 2.在软件的Bin\Plugins目录下找到Consys21.dll复制到/phpstudy/www目录下留作生成软件 3 ...
- python_re正则表达
re模块就本质而言,正则表达式(或RE)是一种小型的.高度专业化的编程语言,(在python中)它内嵌在Python中,并通过re模块实现,正则表达式模块被编译成一系列的字节码,然后由用C编写的匹配引 ...
- C# 对CSV 读写
下面这篇博客只介绍了简单的 用“,”隔开的方式, 不是很推荐,但是对于符合的数据类型还是挺好的 https://www.cnblogs.com/Clin/archive/2013/03/14/2959 ...
- UVA 10003 区间DP
这个题目蛮有新意的,一度导致我没看透他是区间DP 给一个0-L长度的木板,然后给N个数,表示0-L之间的某个刻度,最后要用刀把每个刻度都切一下 使其断开,然后每次分裂的cost是分裂前的木板的长度.求 ...
- UVALive 6763 / CSU 1446
今天比赛的时候拿到的第一道题,其实挺简单的,求两等差序列中相同元素的个数,我想了一下就觉得,只要找到了第一个相等的点,然后后面求最大公约数就可以直接得到结果了 网上叫什么拓展欧几里得,我反正是按照我们 ...
- (5)opencv的基础操作和矩阵的掩模操作
不懂的,可以简单,看看这个网址:https://blog.csdn.net/xiongwen_li/article/details/78503491 图片放到了桌面,所以,图片的路径就是桌面了,剩余的 ...
- 14. react 基础 redux 的编写 TodoList 功能
1. 安装 redux 监听工具 ( 需要翻墙 ) 打开 谷歌商店 搜索 redux devtool 安装第一个即可 2. 安装 redux yarn add redux 3. 创建 一个 store ...
- 工程日记之ChildLost(2) :如何编写一个多线程的程序
Dispatch Dispatch结合语言特性,运行时,和系统的特点,提供了系统的,全面的高层次API来提升多核多线程编程的能力. Dispatch会自动的根据CPU的使用情况,创建线程来执行任务,并 ...
- 移动端 之 触摸事件、Tap事件和swipe事件
触摸事件 touch是一个事件组,意思不止一个事件,是移动端滑动事件组,touchstart touchmove touchend touchcancel touchstart 当刚刚触摸屏幕的时候触 ...
- maven项目编译报错处理
1.问题一: [ERROR] Failed to execute goal on project data-common: Could not resolve dependencies for pro ...