【NOIP2013模拟联考5】军训
题目
HYSBZ 开学了!今年HYSBZ 有n 个男生来上学,学号为1…n,每个学生都必须参加军训。在这种比较堕落的学校里,每个男生都会有Gi 个女朋友,而且每个人都会有一个欠扁值Hi。学校为了保证军训时教官不会因为学生们都是人生赢家或者是太欠扁而发生打架事故,所以要把学生们分班,并做出了如下要求:
1.分班必须按照学号顺序来,即不能在一个班上出现学号不连续的情况。
2.每个学生必须要被分到某个班上。
3.每个班的欠扁值定义为该班中欠扁值最高的那名同学的欠扁值。所有班的欠扁值之和不得超过Limit。
4.每个班的女友指数定义为该班中所有同学的女友数量之和。在满足条件1、2、3 的情况下,分班应使得女友指数最高的那个班的女友指数最小。
请你帮HYSBZ 的教务处完成分班工作,并输出女友指数最高的班级的女友指数。
输入数据保证题目有解。
分析
要求最小的最大值,
很容易让我们想到想到二分。
那么二分ans,
定义\(f_{i}\)表示,前\(i\)个人都分好班了,最小的欠扁值H为多少。
显然,如果\(f_{n}<=limit\),那么二分出来的ans是满足分班条件的。
那么怎么转移呢?,\(f_{i}=min(f_{j}+max(H_{j+1},H_{j+1},H_{j+3}......H_{i}))\)
但这样是\(O(N^{2})\)的,会超时,
那么我们就要像如何优化dp。
用个线段树来记录这个区间的
最大的h值mxh
最小的h值mnh
最小的f值mnf
还有答案mn,显然mn=mnf+mxh(方便而已)
接着随便搞搞就可以了。
#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const int maxlongint=2147483647;
using namespace std;
int f[25000],sum[27000],g[27000],h[27000],limit,n,m,ans;
int mxh[100000],mnh[100000],mnf[100000],mn[100000],lazy[100000];
int spread(int x,int y)
{
if(!lazy[x])
return 0;
lazy[y]=max(lazy[y],lazy[x]);
mxh[y]=max(lazy[x],mxh[y]);
mnh[y]=mxh[y];
mn[y]=mnf[y]+mxh[y];
}
int deeply(int v,int l,int r,int z)
{
if(mxh[v]<z)
{
mxh[v]=mnh[v]=z;
mn[v]=mnf[v]+z;
lazy[v]=z;
return 0;
}
if(l==r)
return 0;
spread(v,v*2);
spread(v,v*2+1);
lazy[v]=0;
int mid=(l+r)/2;
if(mnh[v*2]<z)
deeply(v*2,l,mid,z);
if(mnh[v*2+1]<z)
deeply(v*2+1,mid+1,r,z);
mn[v]=min(mn[v*2],mn[v*2+1]);
mnf[v]=min(mnf[v*2],mnf[v*2+1]);
mnh[v]=min(mnh[v*2],mnh[v*2+1]);
mxh[v]=max(mxh[v*2],mxh[v*2+1]);
}
int change(int v,int l,int r,int x,int y,int z)
{
if(l==x && y==r)
{
deeply(v,l,r,z);
return 0;
}
spread(v,v*2);
spread(v,v*2+1);
lazy[v]=0;
int mid=(l+r)/2;
if(y<=mid)
change(v*2,l,mid,x,y,z);
else
if(x>mid)
change(v*2+1,mid+1,r,x,y,z);
else
change(v*2,l,mid,x,mid,z),change(v*2+1,mid+1,r,mid+1,y,z);
mn[v]=min(mn[v*2],mn[v*2+1]);
mnf[v]=min(mnf[v*2],mnf[v*2+1]);
mnh[v]=min(mnh[v*2],mnh[v*2+1]);
mxh[v]=max(mxh[v*2],mxh[v*2+1]);
}
int put(int v,int l,int r,int x,int z)
{
if(l==r)
{
mxh[v]=mnh[v]=h[l];
mnf[v]=z;
mn[v]=mnf[v]+mxh[v];
return 0;
}
spread(v,v*2);
spread(v,v*2+1);
lazy[v]=0;
int mid=(l+r)/2;
if(x<=mid)
put(v*2,l,mid,x,z);
else
put(v*2+1,mid+1,r,x,z);
mn[v]=min(mn[v*2],mn[v*2+1]);
mnf[v]=min(mnf[v*2],mnf[v*2+1]);
mnh[v]=min(mnh[v*2],mnh[v*2+1]);
mxh[v]=max(mxh[v*2],mxh[v*2+1]);
}
int find(int v,int l,int r,int x,int y,int p)
{
if(l==x && y==r)
{
f[p]=min(f[p],mn[v]);
return 0;
}
spread(v,v*2);
spread(v,v*2+1);
lazy[v]=0;
int mid=(l+r)/2;
if(y<=mid)
find(v*2,l,mid,x,y,p);
else
if(x>mid)
find(v*2+1,mid+1,r,x,y,p);
else
find(v*2,l,mid,x,mid,p),find(v*2+1,mid+1,r,mid+1,y,p);
}
bool ok(int gg)
{
memset(mxh,0,sizeof(mxh));
memset(mnh,0,sizeof(mnh));
memset(mnf,0,sizeof(mnf));
memset(mn,0,sizeof(mn));
memset(lazy,0,sizeof(lazy));
f[0]=0;
int j=1,ans;
for(int i=1;i<=n;i++)
{
while(sum[i]-sum[j-1]>gg)
j++;
put(1,1,n,i,f[i-1]);
change(1,1,n,j,i,h[i]);
f[i]=maxlongint;
find(1,1,n,j,i,i);
}
if(f[n]<=limit)
return true;
return false;
}
int rf()
{
int l=1,r=sum[n];
while(l<r)
{
int mid=(l+r)/2;
if(ok(mid))
r=mid;
else
l=mid+1;
}
printf("%d",l);
}
int main()
{
scanf("%d%d",&n,&limit);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&h[i],&g[i]);
sum[i]=sum[i-1]+g[i];
}
rf();
}
【NOIP2013模拟联考5】军训的更多相关文章
- JZOJ 3463. 【NOIP2013模拟联考5】军训
3463. [NOIP2013模拟联考5]军训(training) (Standard IO) Time Limits: 2000 ms Memory Limits: 262144 KB Deta ...
- JZOJ 3493. 【NOIP2013模拟联考13】三角形
3493. [NOIP2013模拟联考13]三角形(triangle) (File IO): input:triangle.in output:triangle.out Time Limits: 10 ...
- JZOJ 3487. 【NOIP2013模拟联考11】剑与魔法(dragons)
3487. [NOIP2013模拟联考11]剑与魔法(dragons) (Standard IO) Time Limits: 1000 ms Memory Limits: 131072 KB De ...
- JZOJ 3470. 【NOIP2013模拟联考8】最短路(path)
470. [NOIP2013模拟联考8]最短路(path) (Standard IO) Time Limits: 1000 ms Memory Limits: 262144 KB Detailed ...
- JZOJ 3462. 【NOIP2013模拟联考5】休息(rest)
3462. [NOIP2013模拟联考5]休息(rest) (Standard IO) Time Limits: 1000 ms Memory Limits: 262144 KB Detailed ...
- JZOJ 3461. 【NOIP2013模拟联考5】小麦亩产一千八(kela)
3461. [NOIP2013模拟联考5]小麦亩产一千八(kela) (Standard IO) Time Limits: 1000 ms Memory Limits: 262144 KB Det ...
- 【NOIP2013模拟联考7】OSU
[NOIP2013模拟联考7]OSU 描述 Description osu 是一款群众喜闻乐见的休闲软件. 我们可以把osu的规则简化与改编成以下的样子: 一共有n次操作,每次操作只有成功与失败之分, ...
- JZOJ【NOIP2013模拟联考14】隐藏指令
JZOJ[NOIP2013模拟联考14]隐藏指令 题目 Description 在d维欧几里得空间中,指令是一个长度为2N的串.串的每一个元素为d个正交基的方向及反方向之一.例如,d = 1时(数轴) ...
- [jzoj]3468.【NOIP2013模拟联考7】OSU!(osu)
Link https://jzoj.net/senior/#main/show/3468 Description osu 是一款群众喜闻乐见的休闲软件. 我们可以把osu的规则简化与改编成以下的样子: ...
随机推荐
- Python Module_openpyxl_处理Excel表格
目录 目录 前言 软件系统 Install openpyxl module Sample code load_workbook加载Excel文件 wbObjectget_sheet_names 获取E ...
- 2018.03.29 python-pandas 数据读取
#数据读取# read_table,read_csv,read_excel #读取普通分隔数据:read_table #可以读取txt,csv import os import pandas as p ...
- 使用cesium中的scene.open中遇到的几个问题
有些服务是发在场景(scene)下的,超图提供了一个很方便的方法:scene.open,这个方法会将场景中所有的图层(无论是OSGB还是影像和地形)加载进来.同时这个方法会自带一个自动地位功能,具体实 ...
- bigdata数据分析(二):关闭防火墙&安装telnet
先检查CentOS7.0是否已经安装以下两个安装包:telnet-server.xinetd.命令如下: rpm -qa telnet-server rpm -qa xinetd 如果没有安装,则先安 ...
- 算法之美&数据结构与算法复习
1.归并两个有序链表(归并排序) 2.最小路径和--BP解法 3.计算int sqrt(x)--二分解法 4.趣味面试题 5.跳步游戏(Jump_Game)--后向回溯算法 6.Excel列号转十进制 ...
- python的浅复制,深复制
1.a = b是将b的id复制给b,然后a与b指向同一个对象 import numpy as np a = np.arange(5) print(a) b = a print(id(a)) print ...
- 【Linux-驱动】驱动策略----自旋锁
自旋锁 自旋锁最多只能被一个内核任务持有.要是锁未被持有,请求它的内核任务便会立即得到它并继续执行.如果一个内核任务试图请求一个已经被别的内核任务持有的自旋锁,那么CPU就会一直尽心循环---旋转-- ...
- debian上安装mysql server
1 将mysql添加到apt的repository中 第一步,下载mysql提供的ppa文件 wget https://dev.mysql.com/get/mysql-apt-config_0.8.1 ...
- c++ const 用法
1. 修饰一般变量,const int a = 10; 表示此变量不能被修改,简单易懂,不多说 2. 修饰指针,主要是下面三种 const int *cp1 = &a; // ...
- c++多线程并发学习笔记(2)
等待一个时间或其他条件 在一个线程等待完成任务时,会有很多选择: 1. 它可以持续的检查共享数据标志(用于做保护工作的互斥量),直到另一个线程完成工作时对这个标志进行重设.缺点:资源浪费,开销大 2. ...