思路

  • 可以把任意一个数转化为2a+2b+2c+...+2n

    • 例如137的二进制为10001001,这就等效于27+23+2^0
    • 以上结果如何通过程序循环处理呢?需要把数字n分解为上述公式,对指数(a,b,...n)依次进行递归
    • 要对整个结果进行递归生成字符串组后一次性输出比较麻烦,但若是递归输出就会很简单。

算法流程

  1. 将数字n的幂次方组合信息计算出来,存放在数组中
  2. 输出每一个加数项的底数和空格,指数通过递归方式输出
cout<<"2(";
mici(p);
cout<<")";

第1步的处理方法有多种,可以使用数组,也可以使用按位运算,详见例程

例程1

#include<iostream>
using namespace std;
void power(int n){
int i;
int plusFlag=1;
for(i=15; i>=0; i--){
if(n&(1<<i)){ //按位运算
if(plusFlag==1) plusFlag=0; //只有一次不输出“+”
else printf("+");
if(i==0) printf("2(0)"); //临界点
else if(i==1) printf("2"); //临界点
else if(i==2) printf("2(2)"); //临界点
else{
printf("2(");
power(i); //递归
printf(")");
}
}
}
}
int main(){
int n;
cin>>n;
power(n);
return 0;
}

该程序使用了按位运算和移位运算:

移位运算

左移位运算“a<<b”,可以将a的二进制位左移b位,右边补0,左边移出的丢弃
右移位运算“a>>b”,可以将a的二进制位右移b位,左边补0,右边移动的丢弃

例1:137在4字节表示的情况下为00000000 00000000 00000000 10001001

00000000 00000000 00000000 10001001 >>4 得到:

00000000 00000000 00000000 00001000

00000000 00000000 00000000 10001001 <<4 得到:

00000000 00000000 00001000 10010000

同时可以看到,左移1位在左边没有有效位(1)溢出的情况下,相当于乘2,右移1位相当于除2(只保留整数部分)

按位运算

1. 0&0-->0		0&1-->0		1&0-->0		1&1-->1
2. 0|0-->0 0|1-->1 1|0-->1 1|1-->1
3. ^0-->1 ^1-->0

运算规则见例子:

10101010 & 01010101 ---> 00000000	//170 & 85 ---> 0
10101010 | 01010101 ---> 11111111 //170 | 85 ---> 127
^10101010 ---> 01010101 //^170 ---> 85
^11111111 ---> 00000000 //^127 ---> 0

通过按位、移位运算我们可以对数字的比特位进行操作,但功能不仅限于此,在很多题目中,善用按位和移位运算可以帮助我们轻松解决问题,要善于灵活应用,本题就是一个例子。按位和移位运算都是CPU的基本指令,所以这些运算的效率是比较高的。

自反赋值运算符

和其它运算符一样,按位和移位也有自反赋值运算符,方便使用:

a<<=b	//a=a<<b;
a>>=b //a=a>>b;
a&=b //a=a&b;
a|=b //a=a|b;

例程2

#include<iostream>
using namespace std;
void power(int n){
int jishu[16]={0}; //存放级数信息(每个幂次的指数)
int count=0; //有几个加法项 (幂次)
int i=0; //i用于计算指数
while(n){
if(n%2) jishu[count++]=i;
n/=2;
i++; //每做一次除法,指数增加1
}
//这里的思路是:共出现了count次1,第i个1出现的位置是jishu[i],
//所以jishu[i]代表的是第i个1的级数,即2^jishu[i]
for(int i=count-1; i>=0; i--){
if(jishu[i]==0) printf("2(0)"); //临界值
else if(jishu[i]==1) printf("2"); //临界值
else if(jishu[i]==2) printf("2(2)"); //临界值
else{
printf("2(");
power(jishu[i]);
printf(")");
}
if(i!=0) printf("+"); //最后一次不输出“+”
}
}
int main(){
int n;
cin>>n;
power(n);
return 0;
}

NOI-OJ 2.2 ID:8758 2的幂次方表示的更多相关文章

  1. NOI2.2 8758:2的幂次方表示

    描述任何一个正整数都可以用2的幂次方表示.例如: 137=27+23+20 同时约定方次用括号来表示,即ab可表示为a(b).由此可知,137可表示为: 2(7)+2(3)+2(0) 进一步:7=22 ...

  2. 【noi 2.2_8758】2的幂次方表示(递归)

    题意:将正整数N用2的幂次方表示(彻底分解至2(0),2). 解法:将层次间和每层的操作理清楚,母问题分成子问题就简单了.但说得容易,操作没那么容易,我就打得挺纠结的......下面附上2个代码,都借 ...

  3. [NOI OJ]6044:鸣人和佐助

    6044:鸣人和佐助 查看 提交 统计 提问 总时间限制:  1000ms 内存限制:  65536kB 描述 佐助被大蛇丸诱骗走了,鸣人在多少时间内能追上他呢? 已知一张地图(以二维矩阵的形式表示) ...

  4. 九度OJ 1095:2的幂次方 (递归)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:913 解决:626 题目描述: Every positive number can be presented by the exponent ...

  5. 递归--练习9--noi8758 2的幂次方表示

    递归--练习9--noi8758 2的幂次方表示 一.心得 找准子问题就好 二.题目 8758:2的幂次方表示 总时间限制:  1000ms 内存限制:  65536kB 描述 任何一个正整数都可以用 ...

  6. SWUST OJ NBA Finals(0649)

    NBA Finals(0649) Time limit(ms): 1000 Memory limit(kb): 65535 Submission: 404 Accepted: 128   Descri ...

  7. 【noi 2.6_9283】&【poj 3088】Push Botton Lock(DP--排列组合 Stirling数)

    题意:N个编号为1~N的数,选任意个数分入任意个盒子内(盒子互不相同)的不同排列组合数. 解法:综合排列组合 Stirling(斯特林)数的知识进行DP.C[i][j]表示组合,从i个数中选j个数的方 ...

  8. 【noi 2.6_9290】&【poj 2680】Computer Transformation(DP+高精度+重载运算符)

    题意:给一个初始值1,每步操作将1替换为01,将0替换为10.问N步操作后有多少对连续的0. 解法:f[i]表示第i步后的答案.可以直接打表发现规律--奇数步后,f[i]=f[i-1]*2-1;偶数步 ...

  9. 【noi 2.6_2000】&【poj 2127】 最长公共子上升序列 (DP+打印路径)

    由于noi OJ上没有Special Judge,所以我是没有在这上面AC的.但是在POJ上A了. 题意如标题. 解法:f[i][j]表示a串前i个和b串前j个且包含b[j]的最长公共上升子序列长度 ...

随机推荐

  1. js获取时间相关函数

    js获取时间函数 var myDate = new Date; var year = myDate.getFullYear();//获取当前年 var yue = myDate.getMonth()+ ...

  2. RH2288V3服务器 硬件安装以及更换硬件

    滑道类型:L型滑道.可伸缩滑道.报轨 L型滑道:只适用于华为机柜. 准备工具:防静电带或者防静电手套.螺丝刀.浮动螺母安装条. 安装服务器步骤: 1.安装L型滑道   2.安装浮动螺母  3.安装服务 ...

  3. python中创建虚拟环境

    # virtualenv 虚拟环境安装 pip install virtualenv # 创建虚拟环境        virtualenv [虚拟环境名称] # 进入虚拟环境 windows : 进入 ...

  4. Docker的使用初探(一):常用指令说明

    目录 Docker的使用初探(一):常用指令说明 为什么要用Docker Docker的安装与简单使用 国内镜像加速 常用指令 Docker的使用初探(一):常用指令说明 前几个星期实践的了,再不记录 ...

  5. 在 Xshell 中 使用 hbase shell 进入后 无法删除

    在 Xshell 中 使用 hbase shell 进入后 无法删除 问题: 在hbase shell下,误输入的指令不能使用backspace和delete删除,使用过的人都知道,这是有多坑,有多苦 ...

  6. ZooKeeper学习总结 第一篇:ZooKeeper快速入门

    1. 概述 Zookeeper是Hadoop的一个子项目,它是分布式系统中的协调系统,可提供的服务主要有:配置服务.名字服务.分布式同步.组服务等. 它有如下的一些特点: 简单 Zookeeper的核 ...

  7. 文本分类实战(三)—— charCNN模型

    1 大纲概述 文本分类这个系列将会有十篇左右,包括基于word2vec预训练的文本分类,与及基于最新的预训练模型(ELMo,BERT等)的文本分类.总共有以下系列: word2vec预训练词向量 te ...

  8. Maven基础入门与核心知识

    Apache Maven是一个软件项目管理和综合工具.基于项目对象模型(POM)的概念,Maven可以从一个中心资料片管理项目构建,报告和文件. Maven是一个项目管理和综合工具.Maven提供了开 ...

  9. 2-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案安全篇(监听Wi-Fi和APP的数据)

    1-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案安全篇(来看一下怎么样监听网络数据,监听电脑上位机软件的数据) 因为那个软件只能监听咱自己电脑上的数据,所以咱就用电 ...

  10. 手把手教你发布一个Python包

    本文主题如下: 编写一个包(Python 源代码),但不是本文的重点. 编译包,观察编译后的文件. 发布包,发布的包可以有多种类型. 如何在 Pypi 中查看已发布的包 注意: 本文编写的包在 Pyt ...