博客
关于我
P1908 逆序对(树状数组)
阅读量:707 次
发布时间:2019-03-21

本文共 2138 字,大约阅读时间需要 7 分钟。

对于逆序对的计算,可以借助树状数组(BIT)来高效解决问题。树状数组是一种数据结构,专门用于处理前缀和查询等操作,非常适合逆序对计数的问题。以下是对该问题的详细分析和解决方法:

首先,理解逆序对的定义:逆序对是指在数组中,i < j 且 a[i] > a[j]。计算逆序对的数量在数据处理领域是一个常见问题,传统的双重循环算法在数据量较大时显然效率低下。树状数组可以在这种问题上实现时间复杂度的优化。

核心思想

  • 离散化(Coordinate Compression):由于数组元素可能非常大,直接使用数组索引无法处理,因此需要将原始数值映射到较小的范围内。如何离散化是关键:

    • 将数组中的元素(unique后)按顺序排列,分配给它们新的id,确保相同的数值得到相同的id。
    • 在比较两个元素时,先比较它们的新id,这样离散化后的数据保持了原来的相对顺序。
  • 树状数组的应用

    • 更新操作:在树状数组中记录元素出现的位置。
    • 查询操作:通过树状数组计算前缀和,从而统计逆序对的数量。
  • 详细步骤

  • 离散化过程

    • 将数组中的元素收集起来,去重,并按照从小到大的顺序排列。
    • 为每个唯一元素分配一个新的id,id的顺序与元素的原顺序一致。例如,元素1、3、5映射到id1、id2、id3。
  • 树状数组的初始化

    • 确定离散化后的最大id,然后初始化一个数组tree,大小为当前最大id的两倍,确保 suf(最低有效位)为1。
  • lowbit函数

    • used to determine the next position in the tree for update or query operations.
    • 该函数通过位运算快速找到下一个树状数组的位置,即pos += lowbit(pos)。
  • 更新操作

    • 每次遇到一个元素,调用update函数,将其位置记录到树状数组中。例如,当处理5时,update函数会在id3位置增加计数。
  • 查询操作

    • 为了计算逆序对的数量,按顺序处理每个元素。针对元素a[j],统计前面已经处理的元素中id大于当前元素id的数量,这部分的数量即为与a[j]形成逆序对的数量。
    • 使用query函数获取前缀和,统计满足条件的逆序对数目。
  • 逆序对计算

    • 最终的答案为总组合数减去非逆序对数。例如,示例数组中有5个元素,总共有C(5,2)=10个组合,其中7个是非逆序对,答案是10-7=3个逆序对。
  • 树状数组的实现

    # 树状数组实现def init(max_n):    return [0] * (max_n + 2)def update(tree, index, value):    while index <= max_n:        tree[index] += value        index += lowbit(index)def query(tree, index):    res = 0    while index > 0:        res += tree[index]        index -= lowbit(index)    return resdef query_range(tree, l, r):    return query(tree, r) - query(tree, l - 1)

    代码示例

    def main():    n = len(arr)    sorted_unique = sorted(list(set(arr)))    # 创建映射字典,将每个元素映射到最小的id    id_map = {v: i + 1 for i, v in enumerate(sorted_unique)}    max_id = max(id_map.values())        tree_size = max_id * 2    tree = init(tree_size)        for num in arr:        id_num = id_map[num]        update(tree, id_num, 1)            res = 0    for i in reversed(range(n)):        num = arr[i]        current_id = id_map[num]        res += query(tree, current_id - 1)    return res

    注意事项

    • 优化树状数组大小:确保树状数组足够大以容纳所有最大id。
    • 离散化时的排序:离散化时,排序后的元素顺序必须保持原始数组的顺序,只有当元素相同时才能分配相同的id。
    • 处理重复元素:确保在离散化过程中,重复的元素被映射到该在排序中的正确位置,这样它们会被视为非逆序对。

    总结

    通过离散化和树状数组,可以高效地解决逆序对问题。树状数组的更新和查询操作确保了时间复杂度的优化,这使得问题可以处理非常大的数组。关键是正确实现离散化步骤,保证数据的相对顺序,避免错误地计算逆序对数量。这种方法不仅适合逆序对的问题,还能扩展到其他需要频繁查询和更新操作的场景。

    转载地址:http://zsjrz.baihongyu.com/

    你可能感兴趣的文章
    Nhibernate的第一个实例
    查看>>
    nid修改oracle11gR2数据库名
    查看>>
    NIFI1.21.0/NIFI1.22.0/NIFI1.24.0/NIFI1.26.0_2024-06-11最新版本安装_采用HTTP方式_搭建集群_实际操作---大数据之Nifi工作笔记0050
    查看>>
    NIFI1.21.0_java.net.SocketException:_Too many open files 打开的文件太多_实际操作---大数据之Nifi工作笔记0051
    查看>>
    NIFI1.21.0_Mysql到Mysql增量CDC同步中_日期类型_以及null数据同步处理补充---大数据之Nifi工作笔记0057
    查看>>
    NIFI1.21.0_Mysql到Mysql增量CDC同步中_补充_插入时如果目标表中已存在该数据则自动改为更新数据_Postgresql_Hbase也适用---大数据之Nifi工作笔记0058
    查看>>
    NIFI1.21.0_Mysql到Mysql增量CDC同步中_补充_更新时如果目标表中不存在记录就改为插入数据_Postgresql_Hbase也适用---大数据之Nifi工作笔记0059
    查看>>
    NIFI1.21.0_NIFI和hadoop蹦了_200G集群磁盘又满了_Jps看不到进程了_Unable to write in /tmp. Aborting----大数据之Nifi工作笔记0052
    查看>>
    NIFI1.21.0_Postgresql和Mysql同时指定库_指定多表_全量同步到Mysql数据库以及Hbase数据库中---大数据之Nifi工作笔记0060
    查看>>
    NIFI1.21.0最新版本安装_连接phoenix_单机版_Https登录_什么都没改换了最新版本的NIFI可以连接了_气人_实现插入数据到Hbase_实际操作---大数据之Nifi工作笔记0050
    查看>>
    NIFI1.21.0最新版本安装_配置使用HTTP登录_默认是用HTTPS登录的_Https登录需要输入用户名密码_HTTP不需要---大数据之Nifi工作笔记0051
    查看>>
    NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_增删改数据分发及删除数据实时同步_通过分页解决变更记录过大问题_02----大数据之Nifi工作笔记0054
    查看>>
    NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_增加修改实时同步_使用JsonPath及自定义Python脚本_03---大数据之Nifi工作笔记0055
    查看>>
    NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_插入修改删除增量数据实时同步_通过分页解决变更记录过大问题_01----大数据之Nifi工作笔记0053
    查看>>
    NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表或全表增量同步_实现指定整库同步_或指定数据表同步配置_04---大数据之Nifi工作笔记0056
    查看>>
    NIFI1.23.2_最新版_性能优化通用_技巧积累_使用NIFI表达式过滤表_随时更新---大数据之Nifi工作笔记0063
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_根据binlog实现update数据实时同步_实际操作05---大数据之Nifi工作笔记0044
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_根据binlog实现数据实时delete同步_实际操作04---大数据之Nifi工作笔记0043
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置binlog_使用处理器抓取binlog数据_实际操作01---大数据之Nifi工作笔记0040
    查看>>
    NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置数据路由_实现数据插入数据到目标数据库_实际操作03---大数据之Nifi工作笔记0042
    查看>>