题意

给定n个物品,和一个容量为C的桶

需要求出为了装下这些物品,分别使用首次适应算法(FF)、最佳适应算法(BF)需要的桶的数量

\(n \leq 10^6\)

思路

BF:容易想到可以用set维护,每次二分查找能放进去的第一个位置,如果都放不下就另外开一个桶存放

注意因为可能存在重复元素,所以应该使用multiset

FF:容易想到利用数据结构维护区间最大值(我用的是线段树

首先想到了 \(O(nlog^2n)\) 做法,对于每个ai,二分[1, mid]最大能放下ai的位置,又因为查找也是log的,所以是log方

然而怎么改都tle

正解是一个log的做法,线段树update的时候就可以更新最大值、统计答案;查找哪个位置能放下ai时,query里判断一下,如果t[rt << 1]能放下,就往左区间询问,否则往右子树区间询问,当l=r时返回l

code

#include<bits/stdc++.h>
using namespace std;
const int N = 1e6 + 10;
int n, m, T, C;
int a[N];
int t[N << 2];
int mp[N], vis[N], Ans;
inline int maxx (int x, int y) {
return x > y ? x : y;
}
inline void pushup(int rt) {
t[rt] = maxx(t[rt << 1], t[rt << 1 | 1]);
}
inline void build(int l, int r, int rt) {
if(l == r) {
t[rt] = C;
mp[l] = rt;
vis[l] = 0;
return;
}
int mid = (l + r) >> 1;
build(l, mid, rt << 1);
build(mid + 1, r, rt << 1 | 1);
pushup(rt);
}
inline void update(int i, int l, int r, int x, int val) {
if (l == r) {
t[i] = val;
if(val < C && !vis[l]) {
vis[l] = 1;
Ans++;
}
return;
}
int mid = (l + r) >> 1;
if (x <= mid) update(i << 1, l, mid, x, val);
else update(i << 1 | 1, mid + 1, r, x, val);
pushup(i);
}
inline int query(int i, int l, int r, int v) {
if (l == r) return l;
int mid = (l + r) >> 1;
if(t[i << 1] >= v) return query(i << 1, l, mid, v);
else return query(i << 1 | 1, mid + 1, r, v);
} inline void solve_1() {
build(1, n, 1);
Ans = 0;
for(int i = 1; i <= n; i++) {
int pos = query(1, 1, n, a[i]);
int v = t[mp[pos]];
update(1, 1, n, pos, v - a[i]);
}
printf("%d ", Ans);
} multiset<int> S; inline void solve_2() {
S.clear();
int tot = 0;
for (int i = 1; i <= n; ++i) {
if (S.empty()) {if(C > a[i]) S.insert(C - a[i]); ++tot; continue;}
auto tmp = S.lower_bound(a[i]);
if (tmp == S.end()) {
if(C > a[i]) S.insert(C - a[i]); ++tot;
} else {
int V = *tmp; S.erase(tmp);
if (a[i] < V) S.insert(V - a[i]);
}
}
printf("%d\n", tot);
} int main() {
scanf("%d", &T);
while(T--) {
scanf("%d%d", &n, &C);
for(int i = 1; i <= n; i++) scanf("%d", &a[i]);
solve_1();
solve_2();
}
system("pause");
return 0;
}

最新文章

  1. java 处理XML(dom4j-1.6.1)
  2. Java虚拟机JVM学习03 连接过程:验证、准备、解析
  3. 仿造slither.io第一步:先画条蛇
  4. hdu 2602
  5. 把int放在一个char数组里(用于处理每一位数字)
  6. ORM之Dapper操作Sql Server和MySql数据库
  7. Mac 下显示隐藏文件
  8. poj1584 A Round Peg in a Ground Hole 判断多边形凹凸,点到线的距离【基础计算几何】
  9. .net三步配置错误页面,让你的站点远离不和谐的页面
  10. sed中求公共前缀
  11. 转换函数TO_CHAR,TO_DATE,TO_NUMBER
  12. 【Netty】(4)—源码AbstractBootstrap
  13. 转:// LINUX下为ORACLE数据库设置大页--hugepage
  14. JQ判断在不同分辨率电脑下使用不同的banner尺寸
  15. 原生JavaScript运动功能系列(一):运动功能剖析与匀速运动实现
  16. Python进阶-配置文件
  17. docker maven 出错:Failed to execute goal com.spotify:docker-maven-plugin:...: Request error: POST https://192.168.99.100:2376/build?t=
  18. MySQL监控、性能分析——工具篇(转载)
  19. 如何在官网下载JDK(版本、系统类型、字节位等)
  20. Mysql数据操作《三》多表查询

热门文章

  1. [EULAR文摘] 利用蛋白组学技术开发一项蛋白评分用于预测TNFi疗效
  2. map方法整理数据,接口返回值进行处理
  3. keep-alive详解
  4. 从源码MessageSource的三个实现出发实战spring&#183;i18n国际化
  5. Vue 组件VueComponent中_ _proto_ _ 原型对象的指向(指向Vue的原型对象 _ _proto_ _)
  6. elasticsearch相关概念及常用操作汇总
  7. Vue npm run test 错误 (node:16672) UnhandledPromiseRejectionWarning: CssSyntaxError:xxxx.Unknown word
  8. 杭电oj 偶数求和
  9. VsCode轻松使用docker容器-Remote Containers
  10. kubeSphere+kubernetes 集群更新证书