hdu-4027线段树练习
title: hdu-4027线段树练习
date: 2018-10-10 18:07:11
tags:
- acm
- 算法
- 刷题
categories: - ACM-线段树
概述
这道线段树的题可以说是我这一段时间复习线段树后第一次认认真真的做的第一道线段树的题了吧,,,,
然后,,,看似很简单的题翻车了,,,,QAQ
题意和分析
题意大概就是给你一些数,,然后对[l , r]这个区间里的所有数进行开平方根运算,,,其中还有一些询问[l , r]的区间和,,,
看到一排数列的区间和还有更新询问操作的题型铁定是要用线段树来维护这个数列了,,,
一开始我想着结点保存两个区间和,,一个是现在未更新的区间和,,另一个是每个开方后的区间和,,,然后用lazy来延迟更新,,,然后貌似在更新时这样会少更新,,,最后的答案肯定就不对了,,
最后,,,坑了两个多小时的我还是去找别人的做法了,,,
这道题首先一点就是即使数字很大,,,但是 \(2^{63} - 1\) 也就最多开8次平方根,,,而且开到1时再开平方根还是1,,,,
所以再开到区间所有数都为1时就不再对这个区间更新,,,也就是当 node[rt].sum == node[rt].r - node[rt].l + 1 时就返回上一层,,,这样就减小了更新时的操作,,,
最终的代码,,,
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <algorithm>
using namespace std;
typedef long long ll;
const int maxn = 1e5 + 10;
int n, q;
ll a[maxn];
struct tree
{
int l;
int r;
ll sum;
}node[maxn << 2];
void pushup(int rt)
{
node[rt].sum = node[rt << 1].sum + node[rt << 1 | 1].sum;
}
void build(int rt , int l , int r)
{
node[rt].l = l;
node[rt].r = r;
if (l == r)
{
node[rt].sum = a[l];
return; //要记得return,,,最近写这个总是忘记写
}
int mid = (l + r) >> 1;
build(rt << 1 , l , mid);
build(rt << 1 | 1 , mid + 1 , r);
pushup(rt);
return;
}
void update(int rt , int L , int R)
{
if (node[rt].sum == node[rt].r - node[rt].l + 1)
return; //区间全为1时返回,,,不然会tle
if (node[rt].l == node[rt].r)
{
node[rt].sum = (ll)(sqrt(node[rt].sum)); //直接开方就行了
return;
}
int mid = (node[rt].l + node[rt].r) >> 1;
if (L <= mid) update(rt << 1 , L , R);
if (R > mid) update(rt << 1 | 1 , L , R);
pushup(rt);
return;
}
ll query(int rt , int L , int R)
{
if (L <= node[rt].l && node[rt].r <= R)
{
return node[rt].sum;
}
int mid = (node[rt].l + node[rt].r) >> 1;
ll ans = 0;
if (L <= mid) ans += query(rt << 1 , L , R);
if (R > mid) ans += query(rt << 1 | 1 , L , R);
return ans;
}
int main(int argc, char const *argv[])
{
int i = 0;
while(scanf("%d" , &n) != EOF)
{
printf("Case #%d:\n" , ++i);
for (int i = 1; i <= n; ++i)
scanf("%lld" , &a[i]);
build(1 , 1 , n);
scanf("%d" , &q);
while(q--)
{
int t , l , r;
scanf("%d%d%d" , &t , &l , &r);
if (l > r)
swap(l , r); //l , r不一定保证 l <= r 所以要判断
if (t)
printf("%lld\n" , query(1 , l , r));
else
update(1 , l , r);
}
printf("\n"); //每组测试样例之间加空行
}
return 0;
}
总结
- 没看出来一个数最多开方8次啊,,,
- 还是不能仔细的读题,,,比如那个输入的l , r比如那个每一组测试样例之间加空行
- 想的太多了,,,而且最主要的是还是想着套模板解题,,,而不是就题而选择怎么写
- ,,,,
还是做的题太少了,,还是有点像暑假时见到线段树就套板子,,套板子,,已经好几次套板子是没用的情况了,,,而且除了入门的线段树的题,,,都不是之套板子就能出结果的,,,都要在某些地方加一些判断,,,或者对数据的处理,,,线段树只是众多工具之一啊,,不是万能的呐,,,,
(end)
动手总比只想所得到的多一些,,即使结果不尽人意呐~
hdu-4027线段树练习的更多相关文章
- Can you answer these queries? HDU 4027 线段树
Can you answer these queries? HDU 4027 线段树 题意 是说有从1到编号的船,每个船都有自己战斗值,然后我方有一个秘密武器,可以使得从一段编号内的船的战斗值变为原来 ...
- V - Can you answer these queries? HDU - 4027 线段树 暴力
V - Can you answer these queries? HDU - 4027 这个题目开始没什么思路,因为不知道要怎么去区间更新这个开根号. 然后稍微看了一下题解,因为每一个数开根号最多开 ...
- HDU 4027 <线段树,区间√>
题目连接 题意 给出一个区间,每次把[l,r]内的值√,维护区间和. 坑: £:l会比r大,swap. £: 当f[i].sum=f[i].r-f[i].l+1;,不修改.因为保证每个数都大于等于1, ...
- HDU - 4027 线段树减枝
这题太坑了...满满的都是坑点 1号坑点:给定左右区间有可能是反的...因为题目上说x,y之间,但是没有说明x,y的大小关系(害我一直RE到怀疑人生) 2号坑点:开根号的和不等于和开根号(还好避开了) ...
- hdu 5877 线段树(2016 ACM/ICPC Asia Regional Dalian Online)
Weak Pair Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total ...
- hdu 3974 线段树 将树弄到区间上
Assign the task Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- hdu 3436 线段树 一顿操作
Queue-jumpers Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- hdu 3397 线段树双标记
Sequence operation Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- hdu 4578 线段树(标记处理)
Transformation Time Limit: 15000/8000 MS (Java/Others) Memory Limit: 65535/65536 K (Java/Others) ...
- hdu 4533 线段树(问题转化+)
威威猫系列故事——晒被子 Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others) Tot ...
随机推荐
- Mongodb开启远程连接并认证
环境: Mongodb版本:3.4.6 步骤: 1. mongo创建管理员: 在mongo shell下: use admin db.createUser( { user: "testus ...
- 20155232 2016-2017-3 《Java程序设计》第7周学习总结
20155232 2016-2017-3 <Java程序设计>第7周学习总结 教材学习内容总结 第十三章 1.Greenwich MeanTime,格林威治时间,简称GMT时间,由观察太阳 ...
- func_get_args()获取参数
php中func_get_args()可以获取多个参数,讲多个参数放在数组里面. <?php function show() { $attr = func_get_args();//获取输入的参 ...
- HDU 2594 KMP
题目链接 题意:给定两个字符串s1,s2,求最长的s1前缀s使得s为s2的最长后缀,输出该字符串和其长度. 题解:调换s1和s2的顺序,用KMP求解即可. #include <bits/stdc ...
- c++刷题(24/100)正则匹配与位运算
题目1:正则表达式匹配 请实现一个函数用来匹配包括'.'和'*'的正则表达式.模式中的字符'.'表示任意一个字符,而'*'表示它前面的字符可以出现任意次(包含0次). 在本题中,匹配是指字符串的所有字 ...
- def 的传参数的问题
测试了一下.关于python 的函数def里面的传参数问题: 1.传参顺序: # codig:utf-8 def function(a,b=1,*args,**kwargs): print(a,'\n ...
- 【Nginx】 Nginx实现端口转发
什么是端口转发 当我们在服务器上搭建一个图书以及一个电影的应用,其中图书应用启动了 8001 端口,电影应用启动了 8002 端口.此时如果我们可以通过 localhost:8001 //图书 loc ...
- Meltdown论文翻译【转】
转自:http://www.wowotech.net/basic_subject/meltdown.html#6596 摘要(Abstract) The security of computer sy ...
- textarea保留换行和空格
<style> pre {white-space: pre-wrap;} </style> //替换textare <pre class="feedback_q ...
- JDK 6和JDK 7的intern方法之不同
首先介绍下intern方法: 如果常量池中存在当前字符串, 就会直接返回当前字符串. 如果常量池中没有此字符串, 会将此字符串放入常量池中后, 再返回. 1 2 在<深入理解Java虚拟机> ...