最近在看位运算的知识,十分感叹于位运算的博大精深,正好leetcode有 Bit Manipulation 的专题,正好拿来练练手。

Subsets


给出一个由不同的数字组成的数组,枚举它的子数组(子集)。

这道题我之前用递归解过,而且效率还不错(beat 83.33%),解法如下不加详述了:

/**
 * @param {number[]} nums
 * @return {number[][]}
 */

var ans, res, len;

function dfs(index, nums) {
  ans.push(res.concat());

  for (var i = index; i < len; i++) {
    res.push(nums[i]);
    dfs(i + 1, nums);
    res.pop();
  }
}

var subsets = function(nums) {
  nums.sort(function(a, b) {
    return a - b;
  });

  ans = [],
  res = [],
  len = nums.length;

  dfs(0, nums);

  return ans;
};

如果用位运算解,这是一道经典的枚举子集

比如我们有数组[1, 2, 3],还是要用到标志位的概念,全集为1 1 1,全集表示都取,它的子集有1 0 11 1 01 0 0等等,我们通过枚举子集,然后再通过子集获取原来数组的元素即可。

比如子集 1 0 1,我们可以依次获取最右边的1,然后根据获取的1的大小判断数组元素的index位置从而获取数组,之后把该1置为0即可。

/**
* @param {number[]} nums
* @return {number[][]}
*/
var subsets = function(nums) {
  nums.sort(function(a, b) {
    return a - b;
  });

  var all = (1 << nums.length) - 1
    , ans = [];

  // 枚举子集
  for (i = all; i; i = (i - 1) & all) {
    var tmp = i
      , res = [];

    while (tmp) {
      var rightMostOne = tmp & (-tmp)
        , index = Math.log(rightMostOne) / Math.log(2);

      res.push(nums[index]);
      tmp = tmp & (tmp - 1);
    }

    ans.push(res);
  }

  // 子集的枚举没有0,所以要special insert
  ans.push([]);
  return ans;
};

Power of Two


判断一个数是不是2的幂。如果一个数是2的幂,那么该数与上该数-1为0。我们以8举例:

1 0 0 0
0 1 1 1

很明显,上面两数做与运算的结果是0。但是有个特殊的情况是,0 & (0 - 1) === 0,所以我们还得判断该数为正。

/**
 * @param {number} n
 * @return {boolean}
 */
var isPowerOfTwo = function(n) {
  return (!(n & (n - 1)) && n > 0);
};

这里加强下,如果知道某数是2的幂,求解这个指数值。即Math.pow(2, x) = n ,求x的值。

也很简单,用个log的换底公式(其实没有涉及位运算):

return Math.log(n) / Math.IN2;

Missing Number


给出一个0~n组成的数组[0, 1, 2, 3 ... n],从中随即去掉一个数字,给你新的数组,求解被去掉的数字。比如给你[0, 1, 3],返回2

这题涉及^运算的性质:

// if
a ^ b = c;

// then
a ^ c = b;
b ^ c = a;

解法也就呼之欲出了。还是假设数组[0, 1, 3],我们可以知道n为3(等于数组长度),从而可以计算出0 ^ 1 ^ 2 ^ 3的值,我们把它赋值给c;然后我们计算所给数组的元素的异或值,赋值给a,假设被舍弃的元素为b,我们可以得到如下等式:

a ^ b = c;

根据前边所讲:

b = a ^ c;

完整代码:

/**
 * @param {number} n
 * @return {boolean}
 */
/**
 * @param {number[]} nums
 * @return {number}
 */
var missingNumber = function(nums) {
  var a = nums.reduce(function(pre, item) {
    return pre ^ item;
  });

  var b = nums.reduce(function(pre, item, index) {
    return pre ^ index;
  }, nums.length);

  return a ^ b;
};

leetcode - 位运算题目汇总(上)的更多相关文章

  1. leetcode - 位运算题目汇总(下)

    接上文leetcode - 位运算题目汇总(上),继续来切leetcode中Bit Manipulation下的题目. Bitwise AND of Numbers Range 给出一个范围,[m, ...

  2. BitMap - leetcode [位运算]

    136. Single Number 因为A XOR A = 0,且XOR运算是可交换的,于是,对于实例{2,1,4,5,2,4,1}就会有这样的结果: (2^1^4^5^2^4^1) => ( ...

  3. All LeetCode Questions List 题目汇总

    All LeetCode Questions List(Part of Answers, still updating) 题目汇总及部分答案(持续更新中) Leetcode problems clas ...

  4. 78 leetCode 位运算解法

    按照自己的理解题目,数组内所有的组合:假如[1,2,3,4]看成1111到0000里面的排列组合,取位运算. vector<vector > subsets(vector&nums ...

  5. leetcode.位运算.136只出现一次的元素-Java

    1. 具体题目 给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次.找出那个只出现了一次的元素. 说明:你的算法应该具有线性时间复杂度. 你可以不使用额外空间来实现吗? 示例 1 ...

  6. leetcode top 100 题目汇总

    首先表达我对leetcode网站的感谢,与高校的OJ系统相比,leetcode上面的题目更贴近工作的需要,而且支持的语言广泛.对于一些比较困难的题目,可以从讨论区中学习别人的思路,这一点很方便. 经过 ...

  7. leetcode 位运算异或

    1. 只出现一次的数字(136) 异或的性质总结: 相异为1,相同为0: a ^ a = 0; 0 ^ a = a; 如果 a ^ b = c 成立,那么a ^ c = b 与 b ^ c = a 均 ...

  8. JavaScript 位运算总结&拾遗

    最近补充了一些位运算的知识,深感位运算的博大精深,此文作为这个系列的总结篇,在此回顾下所学的位运算知识和应用,同时也补充下前文中没有提到的一些位运算知识. 把一个数变为大于等于该数的最小的2的幂 一个 ...

  9. 位运算总结&拾遗

    JavaScript 位运算总结&拾遗 最近补充了一些位运算的知识,深感位运算的博大精深,此文作为这个系列的总结篇,在此回顾下所学的位运算知识和应用,同时也补充下前文中没有提到的一些位运算知识 ...

随机推荐

  1. [Linux 性能检测工具]VMSTAT

    VMSTAT NAME:          Vmstat: 报告虚拟内存统计 语法 :        vmstat [-a] [-n] [-t] [-S unit] [delay [ count]] ...

  2. TSQL--临时表和表变量

    1. 临时表适用数据量较大的情况,因为临时表可以建立索引 2. 表变量适用于数据较小的情况,表变量只能在定义时创建约束(PRIMARY KEY/UNIQUE)从而间接建立索引 3. 临时表是事务性的, ...

  3. apache https配置

    1.  确认是否安装ssl模块 是否有mod_ssl.so文件 2.  生成证书和密钥 linux下 步骤1:生成密钥 命令:openssl genrsa 1024 > server.key 说 ...

  4. 继续说一下2016里面的json功能(1)

    首先先来测试数据,数据是使用之前的,就 不要在意这些细节了啊~ 借用上一篇的测试数据 ),Chinese int ,Math int) ,),(,),(,),(,null); 然后我们使用这个表里面生 ...

  5. openstack-kilo--issue(十一)Failed connect to 169.254.169.254:80; No route to host

    # curl http://169.254.169.254/latest/user-data curl: () Failed connect to ; No route to host 解决方案: c ...

  6. 烂泥:高负载均衡学习haproxy之TCP应用

    本文由ilanniweb提供友情赞助,首发于烂泥行天下 在前几篇文章中,我们介绍了haproxy的配置参数,而且配置例子都是http协议(7层应用)的. 这篇文章,开始介绍haproxy的4层TCP应 ...

  7. iOS UIAlertView添加输入框

    这玩意有时不用就忘,还是记录一下吧 添加: UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"新建文件夹" mes ...

  8. 正则表达式(/[^0-9]/g,'')中的"/g"是什么意思?

    解答“正则表达式(/[^0-9]/g,'')中的"/g"是什么意思?”这个问题,也为了能够便于大家对正则表达式有一个更为综合和深刻的认识,我将一些关键点和容易犯糊涂的地方再系统总结 ...

  9. Android ScrollView中的组件设置android:layout_height="fill_parent"不起作用的解决办法

    例子,在ScrollView下加入的组件,无论如何也不能自动扩展到屏幕高度. 布局文件. [html] <?xml version="1.0" encoding=" ...

  10. java设计模式之命令模式

    学校中.生活中.社会中总是会存在一定的阶层,虽然我们很多人都不可认可阶层的存在.命令这一词也就在阶层中诞生.家长命令孩子,老师命令学生,领导命令小娄娄.这些都在我们的生活存在的东西,相信这一个模式学习 ...