题意

有一个长度为\(n\)的数组W;

对于每一个\(i\)(\(1<=i<=n\)),你可以选择中任意一些元素W[k] (\(1<=k<i\)),将他们的值改变为0,使得\(\sum_{j=1}^{i-1}W[j] <= m\),

所以输出n个数字,代表对于每一个\(i\),要满足以上条件,至少改变多少个元素.

想法

做这个题的时候,我一直想着怎样贪心,然后一直想不出.

一般来说第一反应应该会是先改变大的元素,但是也可以去考虑保留小的元素.

可以权值线段树+离散化.

离散化之后,用一颗权值线段树维护在前\(i-1\)个元素中,每一种元素出现了多少次.

线段树上每个节点记录每个区间内的元素出现过多少次,和每个区间内所有元素的和.

对于每一个\(i\),我们在前\(i-1\)个元素里面找到尽量多的元素,使它们的和不大于\(m-W[i]\).

在查询的时候,

如果一个节点的和小于\(m-W[i]\),那么答案直接加上这个区间上的元素的个数,即这个区间的元素都保留.

否则如果左子树上的和大于\(m-W[i]\),那么就从左子树中找应该保留哪些数,

如果左子树上的和小于\(m-W[i]\),,那么就保留左子树上的所有元素并且在右子树中找还需要哪些元素,即在右子树中查询不大于\(m-W[i]-\sum左子树上的数\).

代码

#include<stdio.h>
#include<vector>
#include<algorithm> typedef long long ll;
const int MAXN = 200100;
class Node{
public:
int l, r, num;//l,r是区间的范围,num是区间上的数的出现次数
ll val;//val是区间上的数的和
Node *lson, *rson;//左右子树
Node(int _l, int _r);
int mid();
int query(int k);
void update(int pos, int _val);
void clear(); };
Node::Node(int _l, int _r){//建线段树
l = _l, r = _r, val = 0, num = 0;
if(l != r){
lson = new Node(l, mid());
rson = new Node(mid()+1, r);
}
}
void Node::clear(){//删除线段树
if(l!=r){
lson->clear();
rson->clear();
delete lson;
delete rson;
}
}
int Node::mid(){//求l和r的平均值
return (l+r)>>1;
}
int Node::query(int _val){//查询,在不大于_val的情况下,应该保留哪些数字
if(val <= _val) return num;//保留整个区间,直接返回num
if(l == r) return (_val*num/val);//区间范围为1(即只有一个数)的时候,返回能保留多少个这个数字
if(lson->val>_val) return lson->query(_val);
else return lson->num+rson->query(_val-lson->val);//保留左子树上的所有数字,查询右子树
}
void Node::update(int pos, int _val){
if(l == r){
val+=_val,num++;
return;
}
if(pos<l || pos>r) return; if(pos<=mid()) lson->update(pos, _val);
else rson->update(pos, _val);
num = lson->num+rson->num;
val = lson->val+rson->val;
}
int W[MAXN], uniqueW[MAXN]; int main(){
int t;
scanf("%d", &t);
while(t--){
int maxW, n, m;
Node *seqTree = 0;
scanf("%d%d", &n, &m);
for(int i = 1; i <= n; ++i)
scanf("%d", W+i), uniqueW[i] = W[i];
std::sort(uniqueW+1, uniqueW+n+1);//离散化
maxW = std::unique(uniqueW+1, uniqueW+n+1)-(uniqueW+1);
seqTree = new Node(1, maxW);//建立一颗空的线段树,一开始没有任何的元素
for(int i = 1; i <= n ; ++i){
int pos = std::lower_bound(uniqueW+1, uniqueW+maxW+1, W[i])-(uniqueW);//找W[i]离散化之后的位置
printf("%d ", i-seqTree->query(m-W[i])-1);//query返回的是保留的数,i-query-1就是应该改变的数
seqTree->update(pos, uniqueW[pos]);//把W[i]插入线段树
}
seqTree->clear();//删除一下,以免mle
printf("\n");
}
}

HDU 6609 离散化+权值线段树的更多相关文章

  1. R - Weak Pair HDU - 5877 离散化+权值线段树+dfs序 区间种类数

    R - Weak Pair HDU - 5877 离散化+权值线段树 这个题目的初步想法,首先用dfs序建一颗树,然后判断对于每一个节点进行遍历,判断他的子节点和他相乘是不是小于等于k, 这么暴力的算 ...

  2. BZOJ_3224 Tyvj 1728 普通平衡树 【离散化+权值线段树】

    一 题面 Tyvj 1728 普通平衡树 二 分析 比较明显是可以用平衡二叉搜索树(splay)做的. 用权值线段树做,前提就是要先离散化,因为权值线段树维护的值域信息. 板子. 三 AC代码 #in ...

  3. hdu 6703 array(权值线段树)

    Problem Description You are given an array a1,a2,...,an(∀i∈[1,n],1≤ai≤n). Initially, each element of ...

  4. 2019年CCPC网络赛 HDU 6703 array【权值线段树】

    题目大意:给出一个n个元素的数组A,A中所有元素都是不重复的[1,n].有两种操作:1.将pos位置的元素+1e72.查询不属于[1,r]中的最小的>=k的值.强制在线. 题解因为数组中的值唯一 ...

  5. bzoj1588: [HNOI2002]营业额统计(权值线段树)

    1588: [HNOI2002]营业额统计 Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 16863  Solved: 6789[Submit][Sta ...

  6. 线段树(单标记+离散化+扫描线+双标记)+zkw线段树+权值线段树+主席树及一些例题

    “队列进出图上的方向 线段树区间修改求出总量 可持久留下的迹象 我们 俯身欣赏” ----<膜你抄>     线段树很早就会写了,但一直没有总结,所以偶尔重写又会懵逼,所以还是要总结一下. ...

  7. HDU 6464 免费送气球 【权值线段树】(广东工业大学第十四届程序设计竞赛)

    传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6464 免费送气球 Time Limit: 2000/1000 MS (Java/Others)    M ...

  8. HDU 6464 权值线段树 && HDU 6468 思维题

    免费送气球 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submi ...

  9. 区间第k大问题 权值线段树 hdu 5249

    先说下权值线段树的概念吧 权值平均树 就是指区间维护值为这个区间内点出现次数和的线段树 用这个加权线段树 解决第k大问题就很方便了 int query(int l,int r,int rt,int k ...

随机推荐

  1. Python decode()方法

    描述 Python decode() 方法以 encoding 指定的编码格式解码字符串.默认编码为字符串编码.高佣联盟 www.cgewang.com 语法 decode()方法语法: str.de ...

  2. PHP rename() 函数

    定义和用法 rename() 函数重命名文件或目录. 如果成功,该函数返回 TRUE.如果失败,则返回 FALSE. 语法 rename(oldname,newname,context) 参数 描述 ...

  3. 教你在 Linux 下时光穿梭

    时光穿梭?电影里的桥段吧?良许你又在唬人? 非也非也,良许在这里要给大家介绍 touch 命令,有了它你就可以改变时间戳,达到时光穿梭的目的. touch 命令在我们的工作中使用也相当频繁,我们就由浅 ...

  4. luogu P3409 值日班长值周班长 exgcd

    LINK:值日班长值周班长 题目描述非常垃圾. 题意:一周5天 每周有一个值周班长 每天有一个值日班长 值日班长日换 值周班长周换. 共n个值日班长 m个值周班长 A是第p个值日班长 B是第q个值日班 ...

  5. [转]Maven类包冲突终极三大解决技巧

    举例 A依赖于B及C,而B又依赖于X.Y,而C依赖于X.M,则A除引B及C的依赖包下,还会引入X,Y,M的依赖包(一般情况下了,Maven可通过<scope>等若干种方式控制传递依赖).这 ...

  6. @property@classmethod@staticmethod

    一.静态属性@property将方法标记成数据属性:可以访问实例和类的属性 @classmethod标记成类的方法,不需要实例化,可以类直接调用的方法.可以访问类的属性方法,不能访问实例的 class ...

  7. CSS页面布局与网格(上)

    1.布局规划 1.1 网格 网格系统是设计师在切分布局时作为参照的一组行和列. 1.2 布局辅助类 类名用于为布局添加样式.为了让样式可以重用,让类名表达其意图. .column { /* 一般列的样 ...

  8. GitLab 系列文章

    GitLab 系列文章 记录 GitLab 的相关文章 列表 Docker 搭建 GitLab GitLab CI/CD 配置 GitLab 配置模板 访问 GitLab 数据库 GitLab 转让所 ...

  9. EasyExcel的基本使用方法

    在Java语言领域,说到Excel处理工具,大家首先想到的可能是阿帕奇的poi,poi在处理数据量不大的excel文件上确实非常强大,但是随着后来excel从03(一个excel文件中最多有65536 ...

  10. 线程通讯wait&notify

    目录 相关概念 生产者&消费者模型 相关概念 锁:解决线程间冲突的问题 wait&notify:解决线程间协作的问题 wait和sleep的区别 wait期间对象锁是释放的,而slee ...