洛谷 P1582 倒水
题目描述
一天,CC买了N个容量可以认为是无限大的瓶子,开始时每个瓶子里有1升水。接着~~CC发现瓶子实在太多了,于是他决定保留不超过K个瓶子。每次他选择两个当前含水量相同的瓶子,把一个瓶子的水全部倒进另一个里,然后把空瓶丢弃。(不能丢弃有水的瓶子)
显然在某些情况下CC无法达到目标,比如N=3,K=1。此时CC会重新买一些新的瓶子(新瓶子容量无限,开始时有1升水),以到达目标。
现在CC想知道,最少需要买多少新瓶子才能达到目标呢?
输入输出格式
输入格式:
一行两个正整数, N,K(1\le N\le 2\times 10^9,K\le 10001≤N≤2×109,K≤1000)。
输出格式:
一个非负整数,表示最少需要买多少新瓶子。
输入输出样例
3 1
1
13 2
3
1000000 5
15808
解题思路:
首先明确一点,每个瓶中的水量都是2的幂,这个不难证明。 其次,想要瓶子更少,则要尽可能把瓶子合并,这是什么意思呢? 举个例子,输入N=13,K=2,先不考虑购买新瓶子和K,给出13个瓶子的两种合并方案,4 4 4 1和8 4 1。不废话,后者显然更好。 其实也不难证明上面这条的最优性质,总之,我们总是希望尽可能把多的瓶子合并。 实际算法不难,首先把瓶子先进行合并,最后总会得到一个无法再合并的结果,比如上面的8 4 1,但是这时候我们仍有三个瓶子,而数据要求我们最多剩下2个瓶子,所以我们第二步就是对最后两个瓶子进行合并,这时候直接4-1=3,即所求需要购买的瓶子数,于是最后两个瓶子可以合并为一个蓄水量为8的瓶子。
算法设计也比较简单,我这里用的是递归来实现。
func1(n,r): 返回将n个瓶子存入数组f之后,所使用的数组长度,r传入1。 我们在这个函数中找到小于n的最大的2的幂y,这个数字填充数组当前位置,然后再递归调用func1(n-y,r+1),返回其返回值。 边界条件为n==0,此时直接返回r-1。
func2(n,r): 合并数组f中下标为r~n的瓶子,通常r<=n。 两个边界条件:1.n<r+1,直接返回0,因为n绝对小于r,不需要合并。2.n==r+1,说明要合并的瓶子是相邻的两个,直接将他们合并,然后返回就好了。 如果需要合并且合并的不是相邻两个瓶子,那么我们可以递归地调用func2(n,r+1),这样会把编号为r+1到n的瓶子合并,于是我们就可以直接再把编号为r和r+1的瓶子合并,注意要计算结果。
AC代码:
#include<cstdio>
#define ll long long
using namespace std;
ll n,k,f[];
int process1(ll n1,ll r) {
if(!n1) return r - ;
ll y = ;
while(true) {
if(y * > n1) break;
y = y + y;
}
f[r] = y;
return process1(n1-y,r+);
}
int process2(ll n2,ll r) {
if(n2 < r + ) return ;
if(n2 == r + ) {
ll y = f[r] - f[n2];
f[r] *= ;
return y;
}
ll u = process2(n2,r+);
ll e = f[r] - f[r+];
f[r] *= ;
return u + e;
}
int main()
{
scanf("%lld%lld",&n,&k);
int t = process1(n,);
int ans = process2(t,k);
printf("%d",ans);
return ;
}
洛谷 P1582 倒水的更多相关文章
- 洛谷 P1582 倒水 解题报告
P1582 倒水 题目描述 一天,CC买了N个容量可以认为是无限大的瓶子,开始时每个瓶子里有1升水.接着~~CC发现瓶子实在太多了,于是他决定保留不超过K个瓶子.每次他选择两个当前含水量相同的瓶子,把 ...
- 洛谷P1582 倒水
P1582 倒水 题目描述 一天,CC买了N个容量可以认为是无限大的瓶子,开始时每个瓶子里有1升水.接着~~CC发现瓶子实在太多了,于是他决定保留不超过K个瓶子.每次他选择两个当前含水量相同的瓶子,把 ...
- 洛谷P1582 倒水 二进制 lowbit __builtin_popcount
P1582 倒水:https://www.luogu.org/problemnew/show/P1582 题意: 给定n瓶装有1升的水瓶,每次可以把两瓶装水量相同的水和成一瓶,问最少还要增加几瓶装有1 ...
- 洛谷 - P1582 - 倒水 - 位运算
https://www.luogu.org/problemnew/show/P1582 要求用最少的瓶子,那肯定不能有两个一样的瓶子,否则合并更优. 枚举其二进制位,每次加上lowbit,将最后一个1 ...
- 洛谷P1582 倒水 二进制的相关应用
https://www.luogu.org/problem/P1582 #include<bits/stdc++.h> using namespace std; long long N,K ...
- 洛谷P1582 倒水题解
题目 分析 这个题并不难,只是需要仔细思考我们首先可以很轻松的把这个题给疏通一下题意. 1:首先我们最后每个瓶子中装的水一定是一个$2^x$,因为每次都是$2$倍的加,这个应该很好理解. 2:我们要明 ...
- 洛谷 P1582 倒水 (二进制)
这道题实际上是考二进制 很容易看出杯子水量一定是2的i次方 所以n杯水最后剩下的水一定是n用二进制表示中1的个数 所以就枚举n来求什么时候1的个数小于k 那么这里有个优化,不然会超时 因为每次加的目的 ...
- Java实现 洛谷 P1582 倒水
import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; import ...
- 洛谷P1582——倒水(进制,数学)
https://www.luogu.org/problem/show?pid=1582 题目描述 一天,CC买了N个容量可以认为是无限大的瓶子,开始时每个瓶子里有1升水.接着~~CC发现瓶子实在太多了 ...
随机推荐
- POJ 2096 找bug 期望dp
题目大意: 一个人受雇于某公司要找出某个软件的bugs和subcomponents,这个软件一共有n个bugs和s个subcomponents,每次他都能同时随机发现1个bug和1个subcompon ...
- idea使用之maven本地索引更新
idea使用起来的确挺顺手的,但遇到一些问题时,直接百度很难有答案,只能自己慢慢摸索了. 今天在倒腾idea仓库索引的时候,就出事情啦. 先说idea更新本地仓库索引吧,打开settings--> ...
- 应用CLR的线程池
大家都知道这个线程的建立和销毁都需要很大的性能开销,当有比较多且不同的任务需要完成时,可以考虑使用线程池来管理这些线程.在以windows NT为内核的操作系统上每个进程都包含一个线程池,在线程池中存 ...
- 基于图片识别服务的IOS图片识别程序
由于TensorFlow提供的IOS版Demo相对于Android版识别率不高,所以开发了通过识别服务进行图片识别的IOS版程序. 该程序基于图片识别服务(http://www.cnblogs.com ...
- ArcGIS for Android离线数据编辑实现原理
来自:http://blog.csdn.net/arcgis_mobile/article/details/7565877 ArcGIS for Android中现已经提供了离线缓存图片的加载功能,极 ...
- 工作总结 EF GroupBy() Select() Select() 中 Count() 分组 求总
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- [Android Studio] 取消引用库打包出现异常-- provided dependencies can only be jars
Warning: Project App: provided dependencies can only be jars. com.android.support:appcompat-v7:22.2. ...
- 8.跟我学solr---UpdateRequestProcessor具体解释
简单介绍 java web开发的同学应该非常熟悉,在开发中常常会使用filter来处理请求中的一些切面需求. solr也提供类似的一种链式结构的handler来满足在加入数据索引请求的时候.通过切片的 ...
- 浅析分布式数据库中间件DDM
前言 DDM是什么?这是华为云Paas推出的分布式数据库中间件,DDM(Distributed Database Middleware)是一个实现了Mysql协议栈的服务器,前端用户可以把它看做一个数 ...
- Visual Studio静态编译
1.Visual Studio静态编译设置: [Project]-[prj_name Properties],按例如以下框设置 2.why配置成静态编译? 假设动态编译,在没有安装Visual Stu ...