leetcode 699. Falling Squares 线段树的实现
线段树实现。很多细节值得品味 都在注释里面了
class SegTree:
    def __init__(self,N,query_fn,update_fn):
        self.tree=[0]*(2*N+2) # 最后一个节点有可能无用 ,但是确保树是完全的
        self.lazy=[0]*(N+1)
        self.N=N
        self.h=0
        while ((1<<self.h) < N ):
            self.h=self.h+1 
        self.query_fn=query_fn
        self.update_fn=update_fn 
    #这里实现只有单个节点的更改,对于整个节点来说以上的节点没有杯更新到 所以要使用 pull 做进一步的更改
    def _apply(self,x,val):
        self.tree[x] = self.update_fn(val,self.tree[x])
        if(x<self.N):
            self.lazy[x]=self.update_fn(self.lazy[x] , self.tree[x])
    '''
    pull
    从叶部到根部 ,用于更新后的维护
    push
    从根部到叶部 ,查询前的维护 
    '''
    def pull(self,x):
        while x>1 :  # 先while 再除 保证了 (x==1 仍然会执行&&第一次执行的时候不是叶子节点)
            x = x // 2
            self.tree[x] = self.query_fn(self.tree[x*2+1],self.tree[x*2 ] ) #针对区间之间的信息 是 query_fn 
            self.tree[x] = self.update_fn(self.tree[x],self.lazy[x])
    def push(self,x):
        for i in range(self.h,0,-1):
            y = x >> i
            if(self.lazy[ y ]):                  #巧妙的使用了 lazy 前空出无用的 lazy[0]
                self._apply(y*2, self.lazy[y])   #_apply 代码的复用 (修改值 并且置 lazy )
                self._apply(y*2+1, self.lazy[y])
                self.lazy[y]=0
    def update(self,left,right,val):
        #print("update "+str(left )+" "+str(right )+" "+str(val))
        left =left +self.N
        right=right+self.N
        init_left = left
        init_right = right
        while(left<=right): #追溯最邻近父节点的写法 
            if( left & 1 ):
                self._apply(left , val )
                left = left +1
            if((right & 1) ==0):
                self._apply(right, val)
                right = right -1 
            left = left // 2
            right = right // 2  
        #以上只更新到左右节点的最邻近父节点 所以需要进行pull 更新上面的所有节点
        self.pull( init_left )
        self.pull( init_right )
    def query(self,left,right):
        #print("query "+str(left )+" "+str(right ))
        left = left +self.N
        right = right + self.N
        self.push( left )
        self.push( right )
        ans = 0
        while ( left <= right ):#追溯最邻近父节点
            #print("left "+str(left)+" right"+str(right))
            if(left & 1):
                ans= self.query_fn(ans , self.tree[left] )
                left = left + 1 #这个+1 是为了对称 没有实际意义?
            if((right & 1) ==0 ):
                #print(right)
                ans = self.query_fn(ans , self.tree[right ])
                right = right -1                 
            left = left   // 2
            right = right // 2
        return ans 
class Solution:
    def fallingSquares(self, positions):
        """
        :type positions: List[List[int]]
        :rtype: List[int]
        """
        position = [i[0] for i in positions]
        position . extend([i[0]+i[1]-1 for i in positions])
        position=sorted(set(position))
        pos2idx ={pos:i+1 for i,pos in enumerate(position)}
        N=len(pos2idx)
        #print("N is "+str(N))
        segtree=SegTree(N,max,max)
        best = -1
        ans_list = []
        #print(pos2idx)
        for block in positions:
            l,r=pos2idx[block[0]],pos2idx[block[0]+block[1]-1]
            height = segtree.query(l,r) + block[1]
            #print("query height"+str(height-block[1])+"   blcok size:"+str(block[1]))
            segtree.update(l,r,height)
            best=max(best,height)
            ans_list.append(best)
        return ans_list
leetcode 699. Falling Squares 线段树的实现的更多相关文章
- 【leetcode】699. Falling Squares
		题目如下: On an infinite number line (x-axis), we drop given squares in the order they are given. The i- ... 
- 699. Falling Squares
		On an infinite number line (x-axis), we drop given squares in the order they are given. The i-th squ ... 
- 【LeetCode】线段树 segment-tree(共9题)+ 树状数组 binary-indexed-tree(共5题)
		第一部分---线段树:https://leetcode.com/tag/segment-tree/ [218]The Skyline Problem [307]Range Sum Query - Mu ... 
- HDU 1264 Counting Squares (线段树-扫描线-矩形面积并)
		版权声明:欢迎关注我的博客.本文为博主[炒饭君]原创文章,未经博主同意不得转载 https://blog.csdn.net/a1061747415/article/details/25471349 P ... 
- C#LeetCode刷题-线段树
		线段树篇 # 题名 刷题 通过率 难度 218 天际线问题 32.7% 困难 307 区域和检索 - 数组可修改 42.3% 中等 315 计算右侧小于当前元素的个数 31.9% 困难 4 ... 
- Sum of Squares of the Occurrence Counts解题报告(后缀自动机+LinkCutTree+线段树思想)
		题目描述 给定字符串\(S(|S|\le10^5)\),对其每个前缀求出如下的统计量: 对该字符串中的所有子串,统计其出现的次数,求其平方和. Sample Input: aaa Sample Out ... 
- 【Codeforces720D】Slalom       线段树 + 扫描线  (优化DP)
		D. Slalom time limit per test:2 seconds memory limit per test:256 megabytes input:standard input out ... 
- poj 1195:Mobile phones(二维线段树,矩阵求和)
		Mobile phones Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 14391 Accepted: 6685 De ... 
- Codeforces Round #337 (Div. 2) D. Vika and Segments 线段树 矩阵面积并
		D. Vika and Segments Vika has an infinite sheet of squared paper. Initially all squares are whit ... 
随机推荐
- day  61  Django基础之django分页
			Django基础之django分页 一.Django的内置分页器(paginator) view from django.shortcuts import render,HttpRespons ... 
- low版九九乘法表
			# while循环实现九九乘法表num_one = 1while num_one <= 9: num_two = 1 while num_two <= num_one: print(&qu ... 
- java-day04
			IntelliJ快捷键 导入包 alt + enter 删除光标所在行 ctrl + y 复制光标所在行 ctrl + d 格式代码 ctrl + alt + l 单行注释 ctrl + / 多行注释 ... 
- 初识OpenCV-Python - 009: 图像梯度
			本节学习找到图像的梯度和边界.只要用到的函数为: cv2.Sobel(), cv2.Scharr(), cv2.Laplacian() 1. Laplacian 和 Sobel的对比 import c ... 
- POJ 2398 map /// 判断点与直线的位置关系
			题目大意: poj2318改个输出 输出 a: b 即有a个玩具的格子有b个 可以先看下poj2318的报告 用map就很方便 #include <cstdio> #include < ... 
- 关于jar包启动遇到的问题
			一.找不到propertites文件,错误如下 原因是打成的jar不包含classpath信息,需要运行时指定,命令为 -Xbootclasspath/a: 后缀在核心class搜索路径后面.常用! ... 
- linq学习(第二部分)
			8.匿名方法 (1)源起 在上面的例子中 为了得到序列中较大的值 我们定义了一个More方法 var d1 = new Predicate<int>(More); 然而这个方法,没有太多逻 ... 
- 廖雪峰Java12maven基础-1maven入门-1maven介绍
			1.Java项目: * 1.1.首先确定引入哪些依赖包,如commons-logging, log4j, selenium * 1.2.确定项目的目录结构 * 1.3.配置环境:如JDK的版本号 * ... 
- Tensorflow入门篇
			参考Tensorflow中文网(http://www.tensorfly.cn/tfdoc/get_started/introduction.html) ,写一个入门. 1.打开pyCharm,新建 ... 
- Apache Flink 1.9.0版本新功能介绍
			摘要:Apache Flink是一个面向分布式数据流处理和批量数据处理的开源计算平台,它能够基于同一个Flink运行时,提供支持流处理和批处理两种类型应用的功能.目前,Apache Flink 1.9 ... 
