【Single Num II】cpp
题目:
Given an array of integers, every element appears three times except for one. Find that single one.
Note:
Your algorithm should have a linear runtime complexity. Could you implement it without using extra memory?
代码:
class Solution {
public:
int singleNumber(vector<int>& nums) {
int result = ;
const int W = sizeof(int)*;
int *count = new int[W]();
for (size_t i = ; i < nums.size(); ++i)
{
for (size_t j = ; j < W; ++j)
{
count[j] += (nums[i] >> j) & ;
}
}
for (size_t i = ; i < W; ++i)
{
if ( count[i]%!= )
{
result += ( << i);
}
}
delete []count;
return result;
}
};
Tips:
1. 算法原理:由于都是int型的,只需要记录每个bit上1出现的次数;如果1出现的次数不是3的倍数,则将该位置置为1,并赋值到result中。
这道题主要是熟悉下位运算的各种操作,有时巧用位运算,可以大大提升代码效率。
================================================================
学习了另一种解法,代码如下:
class Solution {
public:
int singleNumber(vector<int>& nums) {
int one=, two=, three=;
for (size_t i = ; i < nums.size(); ++i)
{
two |= (one & nums[i]);
one ^= nums[i];
three = two & one;
one &= ~three;
two &= ~three;
}
return one;
}
};
参考链接:https://leetcode.com/discuss/857/constant-space-solution
简单说就是,用二进制模拟三进制运算。
循环体中的五条语句分别做了如下的事情:
1. 算上nums[i]之后,1出现了两次的bit位数有哪些
(包括两种情况:a.原来就是2次的,nums[i]在该bit上是0;原来是1次的,nums[i]在该bit上是1)。
2. 算上nums[i]之后,1出现了一次的bit位数有哪些。
3. 算上nums[i]之后,1出现了三次的bit位数有哪些
4和5. 满3进位,清空。
这里可能会有一个疑问:1出现次数为一和为二的能否算重了?
假设nums[i]在某一个bit上是0:
a. 算two的时候,这一位不会影响原来two这一bit的取值。
b. 算one的时候,相当于该bit与0异或,保持不变。
假设nums[i]在某一个bit上是1:
a. 算two的时候,如果one在该bit上也是1,则two这一bit取1,如果one在这一位上是0,则two这一bit上不变。
b. 算one的时候,相当于该bit与1异或,取反。如果one原来是1,则进位,one置为0;如果one原来是0,则变为1。
对比颜色相同的两端文字,可以知道是不会算重复的。
想到这里还有一个疑问:two = two | (one & nums[i]),如果某一bit上two原来就是1,并且还有进位了,那这?
这种case是不可能出现的:因为one two three初始都是0,假如连着三个nums[i]在某个bit上都为1,则three在该bit上就为1了。后面两条语句,就保证了two和one被清空了,因此永远不会出现two原来是1并且有进位的情况。
======================================================
第二次过这道题,想了一段时间。大体思路已经想出来了,但是位运算操作(“左移多少位再跟1进行与操作”这些的技巧)参考了书上的code。代码还是一次AC了。
class Solution {
public:
int singleNumber(vector<int>& nums) {
int int_bits = sizeof(int)*;
vector<int> count(int_bits,);
// count each '1' in each bit
for ( int i=; i<nums.size(); ++i )
{
for ( int j=; j<count.size(); ++j )
{
count[j] += (nums[i] >> j) & ;
count[j] = count[j] % ;
}
}
// extract single number
int single = ;
for ( int i=; i<count.size(); ++i )
{
single = single + ( count[i] << i );
}
return single;
}
};
还是这种思路好记一些,二进制代替三进制运算的那个并没有印象了。
【Single Num II】cpp的更多相关文章
- 【Word Break II】cpp
题目: Given a string s and a dictionary of words dict, add spaces in s to construct a sentence where e ...
- 【Unique Paths II】cpp
题目: Follow up for "Unique Paths": Now consider if some obstacles are added to the grids. H ...
- 【Path Sum II】cpp
题目: Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals the give ...
- 【Spiral Matrix II】cpp
题目: Given an integer n, generate a square matrix filled with elements from 1 to n2 in spiral order. ...
- 【palindrome partitioning II】cpp
题目: Given a string s, partition s such that every substring of the partition is a palindrome. Return ...
- 【Jump Game II 】cpp
题目: Given an array of non-negative integers, you are initially positioned at the first index of the ...
- 【Combination Sum II 】cpp
题目: Given a collection of candidate numbers (C) and a target number (T), find all unique combination ...
- 【Word Ladder II】cpp
题目: Given two words (start and end), and a dictionary, find all shortest transformation sequence(s) ...
- 【N-Quens II】cpp
题目: Follow up for N-Queens problem. Now, instead outputting board configurations, return the total n ...
随机推荐
- MySQL使用一张表的一列更新另一张表的一列
使用MySQL中,在一张表etl_table_field_info上新增了一个字段tgt_table_en_name,该字段的值想从表etl_table_property_info的tgt_table ...
- HDU 2955 Robberies抢劫案(01背包,变形)
题意:要抢劫,但是抢每个银行都有被抓的概率,问在低于规定的被抓概率情况下最多能抢到多少钱. 输入:第一行为T,表示共T个测试例子.每个例子的第一行给出一个浮点数P,是规定被抓的概率上限.第一行还有一个 ...
- 自行解决12306页面显示异常的问题(长城宽带下WWW。12306无法正常使用)
前二天突然发现家里所用的长城宽带的www.12306.cn无法正常显示,点击余票查询或者车票预订均打不开,加载时间非常长,现象好似CSS等资源文件未载入成功(如图所示)更换chrome.firefox ...
- 64位系统中为VS2012添加OpenGL工具包
之前一直都是按照网上教程进行的添加,以前使用的系统是32位的,所以一直都没有问题.最近换了64位系统,要使用到OpenGL,于是就又进行了原来的工作,但进行测试时,老是失败: 但是在目录:" ...
- 全面了解linux情况常用命令
查看linux服务器CPU详细情况1. 显示CPU个数命令 # cat /proc/cpuinfo | grep "physical id" | sort | uniq | wc ...
- 【BZOJ4196】[NOI2015] 软件包管理器(树链剖分)
点此看题面 大致题意: 有\(n\)个软件包,它们的依赖关系形成一棵树.现在,问你安装或卸载一个软件包,会影响多少个软件包的安装状态. 树链剖分 这道题应该是 树链剖分 算法比较入门的题目吧. 对于安 ...
- 2017.12.11 String 类中常用的方法
1.编写程序将 "jdk" 全部变为大写,并输出到屏幕,截取子串"DK" 并输出到屏幕 package demo; import java.util.Scann ...
- java打包打包
http://blog.sina.com.cn/s/blog_6b9dcc870101k8xq.html 上面说的最后一种方法,不太对. 下面这个可以 Try the fat-jar extensio ...
- WebViewJavaScriptBridge的原理解析
理解WebViewJavaScriptBridge原理 前提条件都是需要bridge在OC实例化,然后二者的互调才可以进行下去 _bridge = [WebViewJavascriptBridge b ...
- JS - Object.create(prototype)方法
用Object.create(prototype)方法创建一个对象,这个对象的原型将指向这个传入的prototype参数