HDU 6070 (线段树)(统计颜色)
HDU 6070 Partition
Problem :
给一段长度为n的序列,要求找出一段区间,使得这段区间的数字种类除以区间长度最小。输出最后的答案即可。(n <= 60000)(9s时限)
Solution :
显然,答案是0~1中的一个数字,可以很自然的想到二分答案的做法。假设目前二分到的答案为mid,那么需要判断
\]
其中cnt(l,r)为l到r这个区间内的数字种类。变化一下式子可以得到:
\]
通过枚举有端点r,使用线段树维护左边的式子,每当右端点r向右移动1时,所影响的区间为r到对应颜色上一次出现的位置,区间整体加1就行了。
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <string>
#include <cmath>
#include <algorithm>
#include <vector>
#include <map>
#include <queue>
using namespace std;
#define eps 1e-10
const int N = 1e5 + 8;
int a[N], pre[N];
int n;
struct Segment_Tree
{
double tag[N << 2];
double lazy[N << 2];
void pushup(int rt)
{
int l = rt << 1, r = rt << 1 | 1;
tag[rt] = min(tag[l], tag[r]);
}
void pushdown(int rt)
{
int l = rt << 1, r = rt << 1 | 1;
if (lazy[rt])
{
tag[l] += lazy[rt];
tag[r] += lazy[rt];
lazy[l] += lazy[rt];
lazy[r] += lazy[rt];
lazy[rt] = 0;
}
}
void build(int l, int r, int rt, double x)
{
tag[rt] = lazy[rt] = 0;
if (l == r)
{
tag[rt] = l * x;
return;
}
int m = l + r >> 1;
build(l, m, rt << 1, x);
build(m + 1, r, rt << 1 | 1, x);
pushup(rt);
}
void update(int L, int R, int val, int l, int r, int rt)
{
if (L <= l && r <= R)
{
tag[rt] = tag[rt] + val;
lazy[rt] += val;
return;
}
pushdown(rt);
int m = l + r >> 1;
if (L <= m) update(L, R, val, l, m, rt << 1);
if (m < R) update(L, R, val, m + 1, r, rt << 1 | 1);
pushup(rt);
}
double query(int L, int R, int l, int r, int rt)
{
if (L <= l && r <= R)
{
return tag[rt];
}
pushdown(rt);
int m = l + r >> 1;
double ans = 1e12;
if (L <= m) ans = min(ans, query(L, R, l, m, rt << 1));
if (m < R) ans = min(ans, query(L, R, m + 1, r, rt << 1 | 1));
return ans;
}
}T;
void init()
{
cin >> n;
for (int i = 1; i <= n; ++i) cin >> a[i];
}
int sgn(double x)
{
if (fabs(x) < eps) return 0;
if (x > 0) return 1;
return -1;
}
bool check(double mid)
{
for (int i = 1; i <= n; ++i) pre[i] = 0;
T.build(1, n, 1, mid);
for (int i = 1; i <= n; ++i)
{
T.update(pre[a[i]] + 1, i, 1, 1, n, 1);
pre[a[i]] = i;
if (sgn(mid * (i + 1) - T.query(1, i, 1, n, 1) >= 0)) return 1;
}
return 0;
}
void solve()
{
double l = 0, r = 1;
while (l + eps < r)
{
double mid = (l + r) / 2;
if (check(mid)) r = mid; else l = mid;
}
printf("%.6f\n", l);
}
int main()
{
cin.sync_with_stdio(0);
int T; cin >> T;
for (int cas = 1; cas <= T; ++cas)
{
init();
solve();
}
}
HDU 6070 (线段树)(统计颜色)的更多相关文章
- HDU 6070 线段树
题意:求AC率,x/y 的最小值,x是区间数字的种类数,y是区间的长度. 分析: 二分答案比率.ans, 动态插入结点,一些区间的size会发生变化,是那些前面暂时没有新的结点的区间 size + 1 ...
- HDU 6035 (虚树)(统计颜色)
HDU 6035 Colorful Tree Problem : 给一棵树,每个结点有一种颜色,定义每条路径的权值为这条路径上颜色的种数,询问所有路径(C(n,2)条)的权值之和. Solution ...
- hdu 5877 线段树(2016 ACM/ICPC Asia Regional Dalian Online)
Weak Pair Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total ...
- hdu 3974 线段树 将树弄到区间上
Assign the task Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- hdu 3436 线段树 一顿操作
Queue-jumpers Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- hdu 3397 线段树双标记
Sequence operation Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- hdu 4578 线段树(标记处理)
Transformation Time Limit: 15000/8000 MS (Java/Others) Memory Limit: 65535/65536 K (Java/Others) ...
- hdu 4533 线段树(问题转化+)
威威猫系列故事——晒被子 Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others) Tot ...
- hdu 2871 线段树(各种操作)
Memory Control Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...
随机推荐
- slimScroll的应用(一)
本类文章依旧是针对初学者来说的,希望大家看到后觉得有用的能给个赞~~ 什么是slimScroll? 一.官网介绍: slimScroll is a small (4.6KB) jQuery plugi ...
- hdu4003/蓝桥杯 金属采集
思路: 树形dp + 分组背包dp. 参考https://www.cnblogs.com/kuangbin/archive/2012/08/29/2661928.html 实现: #include & ...
- PHP安全之 register_globals
一.register_globals = Off 和 register_globals = On的区别 register_globals是php.ini里的一个配置,这个配置影响到php如何接收传递过 ...
- CAD交互绘制样条线(网页版)
在CAD设计时,需要绘制样条线,用户可以设置样条线线重及颜色等属性. 主要用到函数说明: _DMxDrawX::SendStringToExecuteFun 把命令当着函数执行,可以传参数.详细说明如 ...
- centos7下LVM挂载和扩容
说明:此操作在centos7下进行,如果是centos6发行版,需要注意格式化LV的文件系统类型(centos7.0开始默认文件系统是xfs,centos6是ext4).最后一步写入系统的类型,其中文 ...
- 硬盘写入 iso
https://www.jb51.net/softjc/508796.html WinImage 正确操作是要有两个or以上-硬盘. 这样才能写入你要装的操作系统 测试或者安装
- sql中group by
某图书馆收藏有书籍具有不同的出版年份,管理员需要做一下统计工作: (1)每一年书籍的数目,如: 2000年有10本书, 2001年有5本书... (2)每一种书籍的数目,如: 西游记有10本, 三国演 ...
- vue在传值的时候经常遇到的问题
在我用vue编写程序的时候,在传值的时候,经常会遇到些问题,像今天遇到了两个问题,在用父传子的方法去传值,当父组件中的要传的数据是for循环出来的或者是列表的时候,你想每次运行的事件,都去传某一行,或 ...
- 基于jquery的自定义显示消息数量
根据需求简单的实现一个小功能控件,暂时不支持扩展 $("xxxxxxx").iconCountPlugin(options, start, isOffset) {//三个参数,自定 ...
- ubuntu18.04 server配置静态ip
最新发布的ubuntu18.04 server,启用了新的网络工具netplan,对于命令行配置网络参数跟之前的版本有比较大的差别,现在介绍如下:1.其网络配置文件是放在/etc/netplan/50 ...