题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4027
这道线段树的题目我并没有按照线段树的模板写,这里把要点讲出来就行了。
第一:区间开根号不像区间乘或者除,能通过区间和然后一次性乘除求出一个区间的乘除,所以区间开根号只能一个一个的开,但是如果一个一个的开不会超时吗? 假设一个数在long long的范围里面,也就是2^63这个范围内,那么对这个数开根号最多要开几次? 63次。也就是说,开根号能够在很少的次数内,让一个数一直趋近于一个固定值1,这是区间开根号的一个突破口,只要知道当前这个区间的值都等于或者小于1,那就不需要对这个区间做开根号的操作了。虽然一个一个的开根号看似很费时,但是事实上是可行的,我们假设有 10^5(题目N的上限)个数,每个数都是2^63,每次都对整个区间开根号我们需要花费多少时间,因为线段树是一个把数组二叉树化的结构,所以每次访问一个数,需要log(N)的时间复杂度,有N个数,并且每个数最多开log(2^63)次,那么时间复杂度大约是 15*63*10^5;而且题目有说明,所有数的总和是不超过 2^63的,所以实际的时间复杂度还比上面说的要小很多;
所以具体的代码要怎么写?
首先需要一个记录区间最大值的数组,只要当前区间的最大值是小于或者等于1,那么就不用继续访问下面的节点了,因为1开根号仍然是1,这能为我们省下不少时间,其次是一个区间和的数组。
每次区间修改&#