【NOIP2015普及组】 推销员(纪中数据-标准)

博客介绍了NOIP2015普及组的推销员问题,阿明需要在螺丝街按顺序向住户推销产品。内容涉及如何利用线段树优化算法解决该问题,降低时间复杂度,以通过更大规模的数据测试。
摘要由CSDN通过智能技术生成

【题目描述】
阿明是一名推销员,他奉命到螺丝街推销他们公司的产品。螺丝街是一条死胡同,出口与入口是同一个,街道的一侧是围墙,另一侧是住户。螺丝街一共有 N 家住户,第 i 家住户到入口的距离为 Si 米。由于同一栋房子里可以有多家住户,所以可能有多家住户与入口的距离相等。阿明会从入口进入,依次向螺丝街的 X 家住户推销产品,然后再原路走出去。
阿明每走 1 米就会积累 1 点疲劳值,向第 i 家住户推销产品会积累 Ai 点疲劳值。阿明是工作狂,他想知道,对于不同的 X,在不走多余的路的前提下,他最多可以积累多少点疲劳值。

【输入】
第一行有一个正整数 N,表示螺丝街住户的数量。
接下来的一行有 N 个正整数,其中第 i 个整数 Si 表示第 i 家住户到入口的距离。数据保证S1≤S2≤…≤Sn<108。 接下来的一行有 N 个正整数,其中第 i 个整数 Ai 表示向第 i 户住户推销产品会积累的疲劳值。数据保证 Ai<103。
【输出】

输出 N 行,每行一个正整数,第 i 行整数表示当 X=i 时,阿明最多积累的疲劳值。
【样例输入】

Sample Input1:
5
1 2 3 4 5
1 2 3 4 5

Sample Input2:
5
1 2 2 4 5
5 4 3 4 1

Sample Output1:
15
19
22
24
25

Sample Output2:
12
17
21
24
27

【数据范围限制】
对于 20%的数据,1≤N≤20;
对于 40%的数据,1≤N≤100;
对于 60%的数据,1≤N≤1000;
对于 100%的数据,1≤N≤100000。

【题解】 这题乍一看很难,打暴力很明显是不能过的,只能拿到60分。有一位大佬就苦思冥想,想着怎样拿满分。最后,得出结论:这道题要用线段树!其实我们是可以用一些巧妙的方法对暴力程序进行优化,使它的时间复杂度大大的降低。
我们先来讲讲60分的暴力做法吧! 由于60%的数据中,N是小于等于1000的,所以我们可以轻易地用N方的方法骗到60分~只需打双重循环即可(用 i 枚举 x,用 j 枚举选择哪些住户),因为我们可以套样例发现一些神奇的规律:
每一行的答案都是上一行的答案加上max( A[i]+阿明要多走的距离)

#include<cstdio>
using namespace std;
#define maxint64 9223372036854775807
#define maxlongint 2147483647
int dis[100010],a[100010];//dis数组就是s数组
bool bz[100010];//bz数组用来标记到过哪些住户
int main()
{
   
	freopen("salesman.in","r",stdin);
	freopen("salesman.out","w",stdout);
	int n,i,j,distance=0;
	long long max,ans,maxx,t;
	scanf("%d",&n);
	for(i=1;i<=n;i++) scanf("%d",&dis[i]);
	for(i=1;i<=n;i++) scanf("%d",&a[i]);
	max=0,maxx=0;
	
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值