树状数组题目详解 HDU 1166 HDU 1541

2024-02-14 14:18

本文主要是介绍树状数组题目详解 HDU 1166 HDU 1541,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

再次复习树状数组

这里写图片描述

上规律

c1 = a1;
c2 = a1 + a2;
c3 = a3;
c4 = a1 + a2 + a3 + a4;
c5 = a5;
………………
c16 = a1 + a2 + a3 + a4 + a5 + a6 + a7 + a8 + a9 + a10 + a11 + a12 + a13 + a14 + a15 + a16;

树状数组各元素表示的位数

cn = (n - a ^ k + 1) + ………………………… + an

代码版

    for (k = 1; k <= n; k++)for (l = k - lowbit(k) + 1; l <= k; l++)c[k] += b[l];

移位公式

int lowbit (int a)
{return a & (-a);
}

求和公式

int sum (int c[], int n)
{sum = 0;while(n > 0){sum += c[n]n -= lowbit(n);}
}

改变某一个变量公式

int change (int i, int x) // i表示更改的位置, x表示改为的数。
{while(i <= n){c[i] += x;i += lowbit(i);}
}

下面上题 HDU 1166(模板题)

敌兵布阵

Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K
(Java/Others) Total Submission(s): 63224 Accepted Submission(s):
26668

Problem Description
C国的死对头A国这段时间正在进行军事演习,所以C国间谍头子Derek和他手下Tidy又开始忙乎了。A国在海岸线沿直线布置了N个工兵营地,Derek和Tidy的任务就是要监视这些工兵营地的活动情况。由于采取了某种先进的监测手段,所以每个工兵营地的人数C国都掌握的一清二楚,每个工兵营地的人数都有可能发生变动,可能增加或减少若干人手,但这些都逃不过C国的监视。
中央情报局要研究敌人究竟演习什么战术,所以Tidy要随时向Derek汇报某一段连续的工兵营地一共有多少人,例如Derek问:“Tidy,马上汇报第3个营地到第10个营地共有多少人!”Tidy就要马上开始计算这一段的总人数并汇报。但敌兵营地的人数经常变动,而Derek每次询问的段都不一样,所以Tidy不得不每次都一个一个营地的去数,很快就精疲力尽了,Derek对Tidy的计算速度越来越不满:”你个死肥仔,算得这么慢,我炒你鱿鱼!”Tidy想:“你自己来算算看,这可真是一项累人的工作!我恨不得你炒我鱿鱼呢!”无奈之下,Tidy只好打电话向计算机专家Windbreaker求救,Windbreaker说:“死肥仔,叫你平时做多点acm题和看多点算法书,现在尝到苦果了吧!”Tidy说:”我知错了。。。”但Windbreaker已经挂掉电话了。Tidy很苦恼,这么算他真的会崩溃的,聪明的读者,你能写个程序帮他完成这项工作吗?不过如果你的程序效率不够高的话,Tidy还是会受到Derek的责骂的.

Input 第一行一个整数T,表示有T组数据。
每组数据第一行一个正整数N(N<=50000),表示敌人有N个工兵营地,接下来有N个正整数,第i个正整数ai代表第i个工兵营地里开始时有ai个人(1<=ai<=50)。
接下来每行有一条命令,命令有4种形式: (1) Add i j,i和j为正整数,表示第i个营地增加j个人(j不超过30) (2)Sub i
j ,i和j为正整数,表示第i个营地减少j个人(j不超过30); (3)Query i j
,i和j为正整数,i<=j,表示询问第i到第j个营地的总人数; (4)End 表示结束,这条命令在每组数据最后出现;
每组数据最多有40000条命令

Output 对第i组数据,首先输出“Case i:”和回车,
对于每个Query询问,输出一个整数并回车,表示询问的段中的总人数,这个数保持在int以内。

Sample Input 1 10 1 2 3 4 5 6 7 8 9 10 Query 1 3 Add 3 6 Query 2 7 Sub
10 2 Add 6 3 Query 3 10 End

Sample Output Case 1: 6 33 59

分析

首先简单来看这是一个很简单的累加求和的问题 但是过多的数据会使题目超时

当累加的时候有少量的数据改变的时候不需要重新累加 这时候就需要树状数组

当某个数据发生改变的时候树状数组只需要改变大于这个数最近的2的级数的数组 效率为logN

具体方法看复习

核心代码

计算初始的C数组:
for(i = 1; i <= n; i++)for(l = i - lowbit(i) + 1; l <= i;l++)c[i] += b[l];
前n项求和
int sum2 (int c[], int n)
{int sum = 0;while(n >0){sum += a[n];k -= lowbit(k);}
}
更新数组
void add(int i, int x, int c[])
{while(i < n){c[i] += x;i += lowbit(i);}
}

全部代码

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int lowbit(int a);
int sum2(int a[], int k);
void add(int i, int x, int c[]);
int n;
int b[100000];
int c[100000];
char d[10];
int main(void)
{int i, j, k, l, a, change1, change2, sum, v;while (scanf_s("%d", &a) != EOF){for (i = 1; i <= a; i++){memset(b, 0, sizeof(b)); memset(c, 0, sizeof(c)); scanf_s("%d", &n);for (j = 1; j <= n; j++)scanf_s("%d", &b[j]);for (k = 1; k <= n; k++)for (l = k - lowbit(k) + 1; l <= k; l++)c[k] += b[l];printf("Case %d:\n", i);getchar();scanf_s("%s", d, 10);while (d[0] != 'E'){scanf_s("%d %d", &change1, &change2);if (d[0] == 'Q'){sum = 0; v = change1 - 1;sum = sum2(c, change2) - sum2(c, v);//printf("%d %d  ", sum2(c, v), sum2(c, change2));printf("%d\n", sum);}else if (d[0] == 'A'){add(change1, change2, c);}else{add(change1, -(change2), c);}getchar();scanf_s("%s", d, 10);}}}return 0;
}int lowbit(int a)
{return a & (-a);
}int sum2(int a[], int k)
{int sum = 0;while (k > 0){sum += a[k];k -= lowbit(k);}return sum;
}void add(int i, int x, int c[])
{while (i <= n){c[i] += x;i += lowbit(i);}
}

HDU 1541

上题目

Stars Time Limit: 2000/1000 MS (Java/Others) Memory Limit:
65536/32768 K (Java/Others) Total Submission(s): 6737 Accepted
Submission(s): 2691

Problem Description Astronomers often examine star maps where stars
are represented by points on a plane and each star has Cartesian
coordinates. Let the level of a star be an amount of the stars that
are not higher and not to the right of the given star. Astronomers
want to know the distribution of the levels of the stars.

For example, look at the map shown on the figure above. Level of the
star number 5 is equal to 3 (it’s formed by three stars with a numbers
1, 2 and 4). And the levels of the stars numbered by 2 and 4 are 1. At
this map there are only one star of the level 0, two stars of the
level 1, one star of the level 2, and one star of the level 3.

You are to write a program that will count the amounts of the stars of
each level on a given map.

Input The first line of the input file contains a number of stars N
(1<=N<=15000). The following N lines describe coordinates of stars
(two integers X and Y per line separated by a space, 0<=X,Y<=32000).
There can be only one star at one point of the plane. Stars are listed
in ascending order of Y coordinate. Stars with equal Y coordinates are
listed in ascending order of X coordinate.

Output The output should contain N lines, one number per line. The
first line contains amount of stars of the level 0, the second does
amount of stars of the level 1 and so on, the last line contains
amount of stars of the level N-1.

Sample Input 5 1 1 5 1 7 1 3 3 5 5

Sample Output 1 2 1 1 0

翻译

我这弱菜的英语 - -

在 笛卡尔坐标系上有一张星空图。有N颗星 图中的每一颗星都用X,Y坐标表示。

每一颗星的亮度等于它左下角的星的个数

输出1 至 N - 1亮度的星的个数 每个一行

解析

我这弱菜的水平刚开始真没看出来这是一个树状数组的题目

其实从题目给的条件中也可以看出题目说了是从小到大排出X坐标优先

可以看成给出的一个暗示 可以通过树状数组来表示变化

下面我们来想象一下树状数组:

在读入第n个数据时,实际上就是求之前读入的数的X坐标小于第N个数的个数

因为题目给出的数字是默认排好序 x优先的 所以在之前的数不会有Y坐标大于第N个数的

同样因为要频繁的更改数组的第X项并改变前N项的和 所以优先考虑树状数组

伪码解析

从1到N循环读入数据(X0, Y0)用树状数组来计算前X0项的和计入数组 lignt[]中第X0项更新加一继续循环……
从1 到 n - 1 输出light[]数组

完全代码

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;const int maxn = 32050;
int  bit[maxn];
int  level[maxn];
int  n;int lowbit(int x)    //sum lowbit add 为树状数组模板代码 直接调用
{return x & (-x); //等价于x&(-x)
}void add(int x, int val)   //用树状数组对后面节点更新
{while (x<maxn){bit[x] += val;x += lowbit(x);}
}int sum(int x)
{int rank = 0;   //rank代表星星的等级,即它的左下角有的星星个数while (x>0){rank += bit[x];x -= lowbit(x);}return rank;
}int main()
{int   x, y;while (scanf_s("%d", &n) != EOF){memset(bit, 0, sizeof(bit));//清空数组memset(level, 0, sizeof(level));for (int i = 0; i < n; i++){scanf_s("%d%d", &x, &y);level[sum(++x)]++;  // sum函数计算出前x项和  也就是星的亮度add(x, 1);//更新数组}for (int i = 0; i<n; i++)printf("%d\n", level[i]);}return 0;
}

这篇关于树状数组题目详解 HDU 1166 HDU 1541的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/708720

相关文章

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

MyBatis常用XML语法详解

《MyBatis常用XML语法详解》文章介绍了MyBatis常用XML语法,包括结果映射、查询语句、插入语句、更新语句、删除语句、动态SQL标签以及ehcache.xml文件的使用,感兴趣的朋友跟随小... 目录1、定义结果映射2、查询语句3、插入语句4、更新语句5、删除语句6、动态 SQL 标签7、ehc

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input