传送门

•题意

  给出一个整数 n ,问能否将 n 分解成 k 个数之和,且这 k 个数必须是 2 的幂。

  如果可以,输出"YES",并打印出任意一组解,反之输出"NO";

•题解

  预备知识补充:如何求出数 num 最少需要多少个 2的幂之和?

  例如 :

    num = 3 = 20+21至少需要两个

    num = 4 = 22 至少需要一个

    num = 17 = 24+20 至少需要两个

  根据贪心的思想 :

    令 2x ≤ num,求出最大的 x ,那么此时num可以表示为 num = 2x+num1 ( num1 = num-2x );

    num1接着重复上述过程,求出  ≤num1 的最近的2x1,num1 = 2x1+num2 ( num2 = num1-2x1 );

    那么num最少的2的幂之和就为 : 2x+2x1+2x2+.......;

  如何求出x,x1,x2,......呢?

        2x : ≤ num 的距num最近的2的幂

        2x1 : ≤ num1 的距num1最近的2的幂

        2x2 : ≤ num2 的距num2最近的2的幂

        2x3 : ≤ num3 的距num3最近的2的幂

  易得 :

    (1) : num / 2x = oddNum , num / 2x1 = oddNum , num / 2x2 = oddNum ,......

    (2) : num / a = evenNum , num / b = evenNum , num / c = evenNum ,........

  (1)证明 :

    num / 2x = 1;

    num1 / 2x1 = 1 → (num-2x) / 2x1 = 1 → num / 2x1 - 2x / 2x1 = num / 2x1 - 2x-x1 = 1 → num / 2x1 = 1 + 2x-x1 = oddNum ( 奇+偶 );

    num2 / 2x2 = 1 → (num-2x-2x1) / 2x2 = 1 → num / 2x1 - 2x / 2x2- 2x1 / 2x2 = 1 → num / 2x1 = 1 + 2x-x2+2x1-x2 = oddNum ( 奇+偶+偶 );

    .................

  (2)证明 :

    num / a = (num1+2x) / a = num1 / a + 2x / a = 0+偶 = evenNum;( 2x1 ≤ num1 < a )

    ..................

  所以说:

 for i: to k
if(num/(^i)为奇数)
那么2^i就为num最少需要的2的幂之和的成员之一

  并且,$2^i$ 等价于 两个 $2^{i-1}$,所以,可以通过 $2^i$ 转化为 $2^{i-1}$ 开填充;

•Code

 #include<iostream>
#include<cstdio>
using namespace std; int n,k;
int e[];//e[i] : num需要e[i]个2^i void Solve()
{
int curK=;
for(int i=;(<<i) <= n;++i)
if(n>>i&)
{
e[i]=;
curK++;
}
//最少需要curK个2的幂
if(k < curK || k > n)
{
printf("NO\n");
return ;
}
printf("YES\n");
for(int i=;~i;--i)
{
if(!e[i])
continue;
if(curK == k)
break;
int x=min(e[i],k-curK);
e[i] -= x;//减少x个2^i
e[i-] += *x;//增加2*x个2^(i-1)
curK += x;//比之前多了x个
}
for(int i=;i <= ;++i)
for(int j=;j < e[i];++j)
printf("%d ",<<i);
}
int main()
{
scanf("%d%d",&n,&k);
Solve();
return ;
}

•感悟  

  其实,在比赛时,并没有做出这道题,不过也有点小想法,还不成熟;

  赛后看排名,无意间看到了hdu大神Claris的排名,然后,看了一下Claris的提交代码,哇,真简洁,

  是我目前无法达到的。

  大约花费了一个多小时的时间才理解了%%%%%%%%%%%%


分割线2019.5.23

重新温习了一下这道题;

假设 n 最少由 k 个2的幂组成:

n = 2x1 + 2x2 +.......+ 2xk

那么 n / 2xi 为奇数;

今天重新想了一下这个,没有像之前那么繁琐的推公式,一想就想到;

如果 n / 2xi 为偶数,那么 2xi 可以变为 2xi+1 使得组成 n 这个幂值更大,那么,肯定比2xi所需的2的幂少,与假设矛盾;


再次分割2019.10.24

  二进制思想;

  十进制数 n 对应的二进制的第 i 位如果为 1,那么 $2^i$ 就是二进制转十进制 n 的组成部分;

  那么,也即是说,n 对应的二进制有多少个 1,n 就至少需要多少个 2 的幂之和;

  如果这些不够 k 个,那么就通过一个 $2^i$ 可以转化为两个 $2^{i-1}$ 的形式来增加幂之和的个数;

Codeforces Round #529 (Div. 3) C. Powers Of Two(数学????)的更多相关文章

  1. Codeforces Round #529 (Div. 3) C. Powers Of Two

    http://codeforces.com/contest/1095/problem/C 题意:给n找出k个2的幂,加起来正好等于n.例如 9,4:9 = 1 + 2 + 2 + 4 思路:首先任何数 ...

  2. Codeforces Round #529 (Div. 3) C. Powers Of Two (二进制)

    题意:给你一个数\(n\),问是否能有\(k\)个\(2\)次方的数构成,若满足,输出一种合法的情况. 题解:从高到低枚举二进制的每一位,求出\(n\)的二进制的\(1\)的位置放进优先队列中,因为\ ...

  3. # Codeforces Round #529(Div.3)个人题解

    Codeforces Round #529(Div.3)个人题解 前言: 闲来无事补了前天的cf,想着最近刷题有点点怠惰,就直接一场cf一场cf的刷算了,以后的题解也都会以每场的形式写出来 A. Re ...

  4. Codeforces Round #529 (Div. 3) E. Almost Regular Bracket Sequence (思维)

    Codeforces Round #529 (Div. 3) 题目传送门 题意: 给你由左右括号组成的字符串,问你有多少处括号翻转过来是合法的序列 思路: 这么考虑: 如果是左括号 1)整个序列左括号 ...

  5. Codeforces Round #368 (Div. 2) C. Pythagorean Triples(数学)

    Pythagorean Triples 题目链接: http://codeforces.com/contest/707/problem/C Description Katya studies in a ...

  6. Codeforces Round #622 (Div. 2) B. Different Rules(数学)

    Codeforces Round #622 (Div. 2) B. Different Rules 题意: 你在参加一个比赛,最终按两场分赛的排名之和排名,每场分赛中不存在名次并列,给出参赛人数 n ...

  7. Codeforces Round #284 (Div. 2)A B C 模拟 数学

    A. Watching a movie time limit per test 1 second memory limit per test 256 megabytes input standard ...

  8. CodeForces Round #529 Div.3

    http://codeforces.com/contest/1095 A. Repeating Cipher #include <bits/stdc++.h> using namespac ...

  9. Codeforces Round #529 (Div. 3) 题解

    生病康复中,心情很不好,下午回苏州. 刷了一套题散散心,Div 3,全部是 1 A,感觉比以前慢了好多好多啊. 这几天也整理了一下自己要做的事情,工作上要努力... ... 晚上还是要认认真真背英语的 ...

随机推荐

  1. StringBuilder与String有哪些区别?

    System.String具备不可修改性,在程序中这样的特性容易产生性能上的问题.针对这个问题.NET提供的StringBuilder类可以解决类似的问题. String 和 StringBuilde ...

  2. 三、kubernetes环境搭建(实践)

    一.目前近况 docker 版本 K8S支持 18.06的 二.安装docker #1.配置仓库 sudo yum install -y yum-utils device-mapper-persist ...

  3. if 结构语句

    if 条件: print()#不只是能输入print

  4. 简单介绍一下在CentOS上安装Docker。

    简单介绍一下在CentOS上安装Docker. 前置条件: 64-bit 系统 kernel 3.10+ 1.检查内核版本,返回的值大于3.10即可. $ uname -r 2.使用 sudo 或 r ...

  5. python模块psutil的使用

    介绍 psutil是一个跨平台库(http://code.google.com/p/psutil/),能够轻松实现获取系统运行的进程和系统利用率(包括CPU.内存.磁盘.网络等)信息.它主要应用于系统 ...

  6. UESTC1013-我的魔法栈-模拟/排列组合

    有一个串,有黑色和白色两种元素.一次操作可以把最上面的白色元素变成黑色,同时把这个元素上面的所有元素变成白色. 给你一个30以内的串,计算变成全黑时,元素变化的总和. 我用的方法比较笨,打表处理了1- ...

  7. 洛谷p1091合唱队形题解

    题目 合唱队形首先要满足的是从1这个位置到中间任意的位置为单增的,从中间任意的位置到最后是单减的,且长度最长.这样才能满足出列的同学最少. 如果要满足这个条件那么我们可以先预处理出每个点的从前找的最长 ...

  8. 洛谷P1434滑雪题解及记忆化搜索的基本步骤

    题目 滑雪是一道dp及记忆化搜索的经典题目. 所谓记忆化搜索便是在搜索的过程中边记录边搜索的一个算法. 当下次搜到这里时,便直接使用. 而且记忆化搜索一定要满足无后效性,为什么呢,因为如果不满足无后效 ...

  9. Matplotlib学习---用matplotlib画面积图(area chart)

    这里利用Nathan Yau所著的<鲜活的数据:数据可视化指南>一书中的数据,学习画图. 数据地址:http://book.flowingdata.com/ch05/data/us-pop ...

  10. mac centos linux 安装PHP扩展 INTL(国际化) ———— error: 'ext/standard/php_smart_str.h'

    PHP简单源码安装扩展 五个步骤: 详细说明下: cd /fujieace/php7.0/ext/intl:#进入INTL扩展目录? 在编译扩展时候需要phpize准备环境,准备程序需要获取这个目录的 ...