zoukankan      html  css  js  c++  java
  • 0077-小升初2:废品回收

    题目

    小升初2:废品回收
    难度级别:C; 运行时间限制:1000ms; 运行空间限制:51200KB; 代码长度限制:2000000B
    试题描述
    江北榨油厂每天会产生一批废料,厂方会按照日期批次封装起来等待回收。现在积攒了n天的废料,厂里财务记录下了每天产生废料合同价值。现在回收站委托星空运输公司将这n天的废料分m次回收,要求按照日期的先后顺序回收,回收站每次拨给运输公司的经费是一个定值ans,油厂老板每次允许运走的废料是不允许赊账的,所以运输公司每次运走的废料价值一定不会超过ans,若运输公司想运走第i天的废料,必须运走第i天及之前所有还没有运走的废料。对回收站来说,他的拨款不但要保证运输公司能够恰好m次运回所有的废料,而且拨款越少越好。请你编写程序求出符合回收站要求的每次拨款数额。
     
    输入
    第一行输入两个用空格隔开的正整数n和m
    接下来有n个由空格隔开不超过10000正整数,依次表示每一天产生废料的价值。
    输出
    输出包含一个整数,即回收站每次拨款的数额
    输入示例
    7 5 
    100 400 300 100 500 101 400
    输出示例
    500
    其他说明
    数据范围:1<=m<=n<=100000

    分析

      这题算得上前100题里最难的了。要么搜索,要么二分。如果您是新手的话,可以先跳过。但理解一下,烧脑一下,还是挺好的~

      主要还是二分的思路。没学的就先别看了。

      通过读入时累加最大值和“打擂台”比较最小值确定左右端点,在通过累加当前情况所需钱数判断是否满足条件,并同时根据情况改变端点。最终当左右端点相等时即所有情况判断完成时,输出左右端点中的任意一个即可。

      搜索的代码在这里就不给了,基本和二分的代码一样,搜到了再次调用就行了。

    代码

    #include<bits/stdc++.h>
    using namespace std;
    int a[100005],m,n,sum,mid,l,r,s;
    int main()
    {
    	scanf("%d%d",&n,&m);//本代码使用分治算法解决,若不能理解分治算法的基本思想,请勿抄袭!为自己带来不必要的损失!
    	for(int i=0;i<n;i++)
    	{
    		scanf("%d",&a[i]);
    		r+=a[i];//计算最右端点。
    		if(a[i]>l) l=a[i];//计算最左端点。
    	}
    	while(l<r)//二分条件成立。
    	{
    		mid=(l+r)/2;//计算当前中点。
    		sum=1;
    		s=0;
    		for(int i=0;i<n;i++)//统计总钱数。
    		{
    			s+=a[i];
    			if(s>mid)
    			{
    				sum++;
    				s=a[i];
    			}
    			if(sum>m) break;//如果赊账了,说明该方案不行,结束统计,更改端点。
    		}
    		if(sum>m) l=mid+1;//右移左端点。
    		else r=mid;//左移右端点。
    	}
    	printf("%d",l);//输出l和r中的一个端点就行。
    	return 0;
    }
    

      搜索的代码:

    #include<bits/stdc++.h>//搜索涉及到对函数的调用,看不懂更别抄!!!
    using namespace std;
    int a[100005],n,m,maxn=-1,sum,mid,cnt;//maxn给一个很小的值方便读入和特判。
    void dfs(int l,int r)
    {
    	if(l>=r)//和二分没有本质区别
    	{
    		printf("%d",l);
    		return;
    	}
    	mid=(l+r)/2;
    	cnt=1;
    	sum=0;
    	for(int i=0;i<n;i++)
    	{
    		sum+=a[i];
    		if(sum>mid)
    		{
    			sum=a[i];
    			cnt++;
    		}
    		if(cnt>m) break;
    	}
    	if(cnt>m) dfs(mid+1,r);
    	else dfs(l,mid);
    }
    int main()
    {
    	scanf("%d%d",&n,&m);
    	for(int i=0;i<n;i++)
    	{
    		scanf("%d",&a[i]);
    		sum+=a[i];
    		if(a[i]>maxn) maxn=a[i];
    	}
    	dfs(maxn,sum);
    	return 0;
    }
  • 相关阅读:
    田忌赛马(动态规划)
    关于程序中使用相对路径的小技巧
    ReadyGo新闻管理系统 使用号召帖
    解决:Visual Studio2005打开UTF8网页文件时出现乱码
    10个不为人知 但绝对值得收藏的网站
    幻数
    模板偏特化的威力
    c++ 拷贝
    win7下Visualbox 装Ubuntu10.10, 装eclipsecdt+svn插件全过程
    找完工作的心情
  • 原文地址:https://www.cnblogs.com/DARTH-VADER-EMPIRE/p/9879006.html
Copyright © 2011-2022 走看看