NOIP2015普及组T4推销员(暴力+线段树)
题目:阿明是一名推销员,他奉命到螺丝街推销他们公司的产品。螺丝街是一条死胡同,出口与入口是同一个,街道的一侧是围墙,另一侧是住户。螺丝街一共有N家住户,第i家住户到入口的距离为Si米。由于同一栋房子里可以有多家住户,所以可能有多家住户与入口的距离相等。阿明会从入口进入,依次向螺丝街的X家住户推销产品,然后再原路走出去。
阿明每走1米就会积累1点疲劳值,向第i家住户推销产品会积累Ai点疲劳值。阿明是工作狂,他想知道,对于不同的X,在不走多余的路的前提下,他最多可以积累多少点疲劳值
输入格式:
第一行有一个正整数N,表示螺丝街住户的数量。
接下来的一行有N个正整数,其中第i个整数Si表示第i家住户到入口的距离。数据保证S1≤S2≤…≤Sn<10^8。
接下来的一行有N个正整数,其中第i个整数Ai表示向第i户住户推销产品会积累的疲劳值。数据保证Ai<10^3。
输出格式:
输出N行,每行一个正整数,第i行整数表示当X=i时,阿明最多积累的疲劳值。
【数据说明】
对于20%的数据,1≤N≤20;
对于40%的数据,1≤N≤100;
对于60%的数据,1≤N≤1000;
对于100%的数据,1≤N≤1000000。
NOIP之前一直以为NOIP不会考线段树(其实真的没考,这题正解是贪心但是我不会%%%),所以写T4的时候打完暴力根本就没想到写线段树。暴力思路为一重循环枚举推销多少家,第二重循环找出消耗疲劳值最大的,然后改为0,输出答案,效率O(n^2)。
暴力代码如下:
var
len,tired:array[..]of longint;
n,i,j,max,s,ans,now,t,sum:longint; begin
readln(n);
for i:= to n do read(len[i]);
for i:= to n do read(tired[i]);
ans:=;
max:=;
for i:= to n do
begin
max:=-maxlongint;
for j:= to n do
begin
if j>now then s:=(*len[j])-(*sum)+tired[j]
else s:=tired[j];
if s>max then
begin
t:=j;
max:=s;
end;
end;
if t>now then
begin
now:=t;
sum:=len[now];
end;
len[t]:=;tired[t]:=;
inc(ans,max);
writeln(ans);
end;
end.
然后我们可以发现我们找最大值效率是O(n),那我们就用线段树来维护最大值就行了,效率O(log(n)),则总时间效率为O(nlog(n))。
这里简单说一下思路。建两棵线段树,对于现在走到的点,一棵维护左边的疲劳值最大值,也就是只需要维护推销的疲劳值最大值,一棵维护右边的疲劳值最大值,需要维护推销的疲劳值+走路的疲劳值。
代码如下(3kb):
type
point=record
max,delta,l,r,num:longint;
end;
var
len,tired:array[..]of int64;
i,j:longint;
n,max,ans,now,tt,t1,t2,max2,l,x,y:int64;
a:array[..,..]of point; procedure insert(x,num,delta,t:longint);
begin
if a[x,t].l=a[x,t].r then a[x,t].max:=
else
begin
if num<=(a[x,t].l+a[x,t].r)>> then insert(*x,num,delta,t);
if num>(a[x,t].l+a[x,t].r)>> then insert(*x+,num,delta,t);
if a[*x,t].max>a[*x+,t].max then
begin
a[x,t].max:=a[*x,t].max;
a[x,t].num:=a[*x,t].num;
end
else
begin
a[x,t].max:=a[*x+,t].max;
a[x,t].num:=a[*x+,t].num;
end;
end;
end; procedure build(x,l,r,t:longint);
begin
a[x,t].l:=l;a[x,t].r:=r;
if l=r then
begin
a[x,t].num:=l;
read(a[x,t].max);
a[x,t].delta:=;
if t= then len[l]:=a[x,t].max;
if t= then tired[l]:=a[x,t].max;
end
else
begin
build(*x,l,(l+r)>>,t);
build(*x+,((l+r)>>)+,r,t);
if a[*x,t].max>a[*x+,t].max then
begin
a[x,t].max:=a[*x,t].max;
a[x,t].num:=a[*x,t].num;
end
else
begin
a[x,t].max:=a[*x+,t].max;
a[x,t].num:=a[*x+,t].num;
end;
a[x,t].delta:=;
end;
end; procedure update(x,l,r,delta,t:longint);
begin
if (l<=a[x,t].l)and(a[x,t].r<=r) then
inc(a[x,t].delta,delta)
else
begin
if l<=(a[x,t].l+a[x,t].r)>> then update(*x,l,r,delta,t);
if r>(a[x,t].l+a[x,t].r)>> then update(*x+,l,r,delta,t);
if a[*x,t].max+a[*x,t].delta>a[*x+,t].max+a[*x+,t].delta then
begin
a[x,t].max:=a[*x,t].max+a[*x,t].delta;
a[x,t].num:=a[*x,t].num;
end
else
begin
a[x,t].max:=a[*x+,t].max+a[*x+,t].delta;
a[x,t].num:=a[*x+,t].num;
end;
end;
end; function query(x,l,r,h,t:longint):int64;
var
ret:int64;
begin
ret:=;
if (l<=a[x,t].l)and(a[x,t].r<=r) then
begin
ret:=a[x,t].max+a[x,t].delta;
if ret>=h then
tt:=a[x,t].num;
end
else
begin
inc(a[*x,t].delta,a[x,t].delta);
inc(a[*x+,t].delta,a[x,t].delta);
inc(a[x,t].max,a[x,t].delta);
a[x,t].delta:=;
if l<=(a[x,t].l+a[x,t].r)>> then ret:=query(*x,l,r,h,t);
if r>(a[x,t].l+a[x,t].r)>> then
if ret<query(*x+,l,r,ret,t) then ret:=query(*x+,l,r,ret,t);
end;
query:=ret;
end; begin
readln(n);
build(,,n,);
build(,,n,);
for i:= to n do update(,i,i,len[i]+tired[i],);
ans:=;
max:=;
tt:=;
for i:= to n do
begin
max:=-maxlongint;
if now> then max:=query(,,now,,);
t1:=tt;
max2:=-maxlongint;
if now<n then max2:=query(,now,n,,);
if max<max2 then
begin
max:=max2;
t1:=tt;
end;
if t1>now then
begin
update(,t1,n,-*len[t1],);
insert(,t1,,);
now:=t1;
end;
insert(,t1,,);
inc(ans,max);
writeln(ans);
end;
end.
以下为官方数据情况:
NOIP2015普及组T4推销员(暴力+线段树)的更多相关文章
- 【NOIP2015普及组】 推销员(纪中数据-标准)
题目 [题目描述] 阿明是一名推销员,他奉命到螺丝街推销他们公司的产品.螺丝街是一条死胡同,出口与入口是同一个,街道的一侧是围墙,另一侧是住户.螺丝街一共有 N 家住户,第 i 家住户到入口的距离为 ...
- 【NOIP2015普及组】推销员_详解
题目 题目大意 阿明是一名推销员--螺丝街是一条直线,一端有入口,一共有 \(N(<100,000)\) 家住户,第 \(i\) 家住户到入口的距离为 \(S_i\) 米.由于同一栋房子里可以有 ...
- NOIP2015普及组总结
NOIP2015普及组总结 这次考试总体感觉不错,不过觉得时间有点紧,在最后30分钟才打完. 第一题(金币coin):大大的W!爆搜O(N),一分钟打完: 第二题(扫雷游戏mine):同上: 第三题( ...
- NOIP2004普及组第3题 FBI树
/* 1106: NOIP2004普及组第3题 FBI树 时间限制: 1 Sec 内存限制: 128 MB 提交: 10 解决: 9 [提交] [状态] [讨论版] [命题人:外部导入] 题目描述 我 ...
- [NOIP2015 普及组] 扫雷游戏
[NOIP2015 普及组] 扫雷游戏 难度:入门 题目描述 扫雷游戏是一款十分经典的单机小游戏.在nn行mm列的雷区中有一些格子含有地雷(称之为地雷格),其他格子不含地雷(称之为非地雷格).玩家翻开 ...
- NOIP2015普及组第四题推销员
好久没有写博客了,今天再写一篇.还是先看题: 试题描述 阿明是一名推销员,他奉命到螺丝街推销他们公司的产品.螺丝街是一条死胡同,出口与入口是同一个,街道的一侧是围墙,另一侧是住户.螺丝街一共有 N 家 ...
- hdu 5475 An easy problem(暴力 || 线段树区间单点更新)
http://acm.hdu.edu.cn/showproblem.php?pid=5475 An easy problem Time Limit: 8000/5000 MS (Java/Others ...
- HDU 1394 Minimum Inversion Number(最小逆序数/暴力 线段树 树状数组 归并排序)
题目链接: 传送门 Minimum Inversion Number Time Limit: 1000MS Memory Limit: 32768 K Description The inve ...
- HDU 3974 Assign the task 暴力/线段树
题目链接: 题目 Assign the task Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/O ...
随机推荐
- JS继承方法
1.原型链: 每个构造函数都有一个原型对象,且有一个指针指向该原型对象(prototype),原型对象都包含一个指向构造函数的指针(constructor),而实例都包含一个指向原型对象的内部指针(p ...
- unittest,requests——接口测试脚本及报告
用unittest管理两个利用requests模块,做百度搜索的简单接口测试用例,之后自动输出报告 # encoding=utf-8import requests,unittest,HTMLTestR ...
- 排查GCC 4.4.X版本优化switch-enum的BUG
起因 一次偶然碰到一个诡异的bug,现象是同一份C++代码使用GCC4.4.x版本在开启优化前和优化后的结果不一样,优化后的代码逻辑不正确. 示例代码如下: //main.cpp #include & ...
- Numpy入门笔记第二天
# 数组的组合 import numpy as np arr1 = np.arange(5) arr2 = np.arange(3) print arr1 print arr2 [0 1 2 3 4] ...
- HTML(1)简介
"超"文本标记语言--HTML 文本,是指书面语言的表现形式. 百度百科 说白了,文本就是你能看得到的字,不论是纸上的还是屏幕上的,都是文本.文本就是用来记录信息一种形式. 那么, ...
- #Ubuntu 18.04 安装tensorflow-gpu 1.9
参考 https://tensorflow.google.cn/install/install_linux http://nvidia.com/cuda http://developer.nvidia ...
- 对首师大研究生院的UI分析
我们分析的是首都师范大学研究生院的UI界面 网站链接http://grad.cnu.edu.cn/index.htm 学校设计和石家庄铁道大学的界面类似,都是有一个大的置顶的名字,将学校或者学院的名字 ...
- 《JavaScript设计模式与开发实践》——第3章 闭包和高阶函数
闭包 变量的作用域和生存周期密切相关 高阶函数 函数可以作为参数被传递 函数可以作为返回值输出
- Python语言基础
一.Python简介 Python是跨平台动态语言 特点:优雅.明确.简单 适用:web网站和网络服务:系统工具和脚步:包装其他语言开发的模块 不适用:贴近硬件(首选C):移动开发:IOS/Andro ...
- c# dataGridView排序
一.对阿拉伯数字进行自定义排序: 简单有效方法: 1.该列的sortmode属性为auto...(一般默认) 2.比如首列序号,添加该列数据的时候直接添加int即可.切忌不要用string. obje ...