给定一个整数N,找出一个比N大且最接近N,但二进制权值与该整数相同 的数
1,问题描述
给定一个整数N,该整数的二进制权值定义如下:将该整数N转化成二进制表示法,其中 1 的个数即为它的二进制权值。
比如:十进制数1717 的二进制表示为:0000 0110 1011 0101 故它的二进制权值为7(二进制表示中有7个1)
现在要求一个比N大,且最靠近N的数,且这个数的二进制权值与N相同。(这里不考虑Integer.MAX_VALUE 和负数情形。)
对于有符号的32位整数而言:它们的补码如下:
Integer.MAX_VALUE= 0111 1111 1111 1111 1111 1111 1111 1111 (2^32-1)
Integer.MIN_VALUE= 1000 0000 0000 0000 0000 0000 0000 0000 (-2^32)
0 = 0000 0000 0000 0000 0000 0000 0000 0000
-1= 1111 1111 1111 1111 1111 1111 1111 1111
(负数的补码是在原码的基础上,符号位不变,其余位取反,末位加1)参考:原码, 反码, 补码 详解
2,问题分析
思路①
先求出N的二进制权值,然后从N+1开始递增,依次判断这个数的二进制权值是否与N相同,直到找到一个相同的二进制权值的数。
而求解二进制权值的算法可以用移位操作来实现。可参考:JAVA中常用的二进制位操作
//求解正数的二进制表示法中的 1 的位数
private static int countBit(int num){
int count = 0;
for(; num > 0; count++)
{
num &= (num - 1);
}
return count;
}
思路①这种方式,当N很大时,效率会很慢。
那有没有更好的思路呢?
其实我们的目的是找到一个数,只要这个数的二进制权值与N相同,且该数大于N且最接近N即可。
那么,可以先将N用二进制表示出来。然后,从低位开始寻找N的二进制中出现 1 之后,第一次出现0的位,假设是第 i 位,那么将第 i 位置为1,得到一个新的数M,此时 M 的二进制中 1 的个数比N多一个。再把M的二进制中的 第 i-1 位的 1 设置为0 ,就得到了大于N且最接近N的二进制权值一样的数。
示例如下:
N= 0010 1001 011 1111
将第5位置为0,得到了M(最右边的位为第0位)
M= 0010 1001 011 1111
由于是从低位开始寻找第一次出现0的位。故第5位的右边全是1,再将M的 第 i-1 位(第四位)设置为0,得到了H
H= 0010 1001 0110 1111
H所对应的十进制数,就是题目中要寻找的数。
再比如:
N= 0010 1001 011 1100
M= 0010 1001 011 1100
H= 0010 1001 011 1100
再比如:
N= 0000 1000
M= 0001 1000
H= 0001 0000
3,代码实现:
思路①实现:
import java.util.Scanner;
public class Main{
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
while(sc.hasNextLong())
{
int num = sc.nextInt();
long start = System.currentTimeMillis();
int weight = countBit(num);
int k = num + 1;
while(countBit(k) != weight)
{
k++;
}
long end = System.currentTimeMillis();
System.out.println("res:" + k + " time: " + (end - start));
}
sc.close();
}
private static int countBit(int num){
int count = 0;
for(; num > 0; count++)
{
num &= (num - 1);
}
return count;
}
}
②思路②实现:
import java.util.Scanner;
public class Larger_Near_Than_N {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
while(sc.hasNextInt())
{
int number = sc.nextInt();
long start = System.currentTimeMillis();
int result = findNearThanN(number);
long end = System.currentTimeMillis();
System.out.println("res:" + result + " time: " + (end - start));
}
sc.close();
}
private static int findNearThanN(int number){
int result = -1;
int first_indexOf_1 = getFirst_1(number);
if(first_indexOf_1 != -1)//找到了一个 二进制位 1
{
//如果找到了一个二进制位1, indexOf_0不可能为0
int indexOf_0 = getFirst_0(number, first_indexOf_1);
int tmp = setBit_1(number, indexOf_0);
result = setBit_0(tmp, indexOf_0 - 1);
}
return result;
}
//第 i位为1 返回true,为0 返回false
private static boolean getBit(int number, int i){
return ((number & (1 << i)) != 0);
}
//从右到左(低位开始)查找number的二进制位 1 的位置
private static int getFirst_1(int number){
int index = -1;
for(int i = 0; i <= 31; i++)
if(getBit(number, i))
{
index = i;
break;
}
return index;//返回二进制位 1 在 number 中的位置
}
//从 start+1 位置开始,查找 number的二进制中,第一个出现的0的位置
private static int getFirst_0(int number, int start){
int index = -1;
for(int i = start + 1; i <= 31; i++)
{
if(!getBit(number, i))
{
index = i;
break;
}
}
return index;
}
//将 number 的二进制表示法中的第 i 位设置为 1
private static int setBit_1(int number, int i){
return (number | (1 << i));
}
//将 number 的二进制表示法中的第 i 位设置为 0
private static int setBit_0(int number, int i){
int mask = ~(1 << i);
return (number & mask);
}
}
给定一个整数N,找出一个比N大且最接近N,但二进制权值与该整数相同 的数的更多相关文章
- C语言:找出一个大于给定整数m且紧随m的素数,-求出能整除x且不是偶数的数的个数,
//函数fun功能:找出一个大于给定整数m且紧随m的素数,并作为函数值返回. #include <stdlib.h> #include <conio.h> #include & ...
- 平面上给定n条线段,找出一个点,使这个点到这n条线段的距离和最小。
题目:平面上给定n条线段,找出一个点,使这个点到这n条线段的距离和最小. 源码如下: #include <iostream> #include <string.h> #incl ...
- 给定一个英文字符串,请编写一个PHP函数找出这个字符串中首先出现三次的那个英文字符(需要区分大小写),并返回
给定一个英文字符串,请编写一个PHP函数找出这个字符串中首先出现三次的那个英文字符(需要区分大小写),并返回 //统计字符串中出现的字符的出现次数 public function strNum(){ ...
- Class 找出一个整形数组中的元素的最大值
目的:找出一个整形数组中的元素的最大值 以下,我们用类和对象的方法来做. #include<iostream> using namespace std; class Array_m ...
- Android系统,动态找出一个包下所有的类
最近在写一个android应用,由于针对不同机型功能很不同,为了隔离变化,希望将各项功能插件化,通过编译开关来控制不同版本的功能,而不是在代码中通过逻辑来判断. 我想了一个办法,用表驱动的方法,结合插 ...
- c语言题目:找出一个二维数组的“鞍点”,即该位置上的元素在该行上最大,在该列上最小。也可能没有鞍点
//题目:找出一个二维数组的“鞍点”,即该位置上的元素在该行上最大,在该列上最小.也可能没有鞍点. // #include "stdio.h" #include <stdli ...
- 笔试题&面试题:找出一个数组中第m小的值并输出
题目:找出一个数组中第m小的值并输出. 代码: #include <stdio.h> int findm_min(int a[], int n, int m) //n代表数组长度,m代表找 ...
- C语言程序,找出一个二维数组的鞍点。
什么是鞍点????? 鞍点就是在一个二维数组中,某一个数在该行中最大,然而其在该列中又是最小的数,这样的数称为鞍点. 昨天突然在书上看到这样的一道题,就自己尝试着写了一个找出一个二维数组中的鞍点. 好 ...
- [PY3]——找出一个序列中出现次数最多的元素/collections.Counter 类的用法
问题 怎样找出一个序列中出现次数最多的元素呢? 解决方案 collections.Counter 类就是专门为这类问题而设计的, 它甚至有一个有用的 most_common() 方法直接给了你答案 c ...
随机推荐
- mysql 数据类型,字符集
数据类型 1,数值类型2,字符串类型3,日期和时间4,ENUM和SET5,几何数据类型 数据类型选项 unsigned 无负值 zerofill 数值显示有影响,会前置0来填充不 ...
- Android 手机卫士16--手机杀毒
1.指定动画一直旋转 rotateAnimation.setRepeatCount(RotateAnimation.INFINITE); android:repeatCount 重复的次数,默认为0, ...
- SharpGL学习笔记(十三) 光源例子:环绕二次曲面球体的光源
这是根据徐明亮<OpenGL游戏编程>书上光灯一节的一个例子改编的. 从这个例子可以学习到二次曲面的参数设置,程序中提供了两个画球的函数,一个是用三角形画出来的,一个是二次曲面构成的. 你 ...
- Android studio 快捷添加构造方法以及set与get
第一种方式 快捷键: Alt + lnsert (笔记本可能没有后面的按键) 按快捷键会出现下面这个页面: 第二种方式:点开后是跳出上面那个选择框
- setTimeout实现动画的黄金优化法则
1.使用递归思想实现setTimeout的轮询动画:在每一次执行方法的时候都重新的设置一个定时器,然后在指定时间内重新的执行当前的方法 问题:每一次设置的定时器,虽然不执行了,但是还存在呢,浪费性能 ...
- Sharepoint学习笔记—习题系列--70-573习题解析 -(Q115-Q117)
Question 115You create a timer job.You need to debug the timer job.To which process should you attac ...
- dict和set
#dict和set #dict #Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map #使用键-值(key-value)存储,具有极快的查找速度. #字 ...
- iOS设计模式之组合模式
组合模式(Composite) 基本理解 整体和部分可以一直对待. 组合模式:将对象组合成树形结构以表示"部分--整体"的层次结构.组合模式使得用户对单个对象和组合独享的使用具有一 ...
- OC中NSDictionary和NSSet简单操作
/** * 字典 存放键值对类型的数据 存放数据是无序的 */ // 字典在控制台输出是用{}包括起来的 // NSDictionary 不可变字典 // 1.创建对象 // 初始化方法 NSDic ...
- iOS实现(超级猜图)源码
//首先建立模型文件 QLLQuestion.hheQLLQuestion.m文件 #import <Foundation/Foundation.h> @interface QLLQues ...