[LeetCode]singleNumber
题目:singleNumber
Given an array of integers, every element appears twice except for one. Find that single one.
Note:
Your algorithm should have a linear runtime complexity. Could you implement it without using extra memory?
给一个数组里面除了唯一的一个数只出现一次,其他每个数都重复了一次。
要求:线性的时间复杂度,和O(1)的空间复杂度
思路:
如果不要求线性的时间复杂度,则可以考虑用排序的方法,成熟稳定,又有很好的通用性。
但是,如果要保证线性的时间复杂度,则可以考虑使用hash_map来找单个出现的数字。
通常思路里面效率的瓶颈在于如何确定当前元素是前面已经出现过的元素,此时可以通过hash_map来确保在O(1)的时间复杂度里确定当前元素是否出现过。
我用删除的方法来降低冲突率,同时,方便最后找到最终的singleNumber。
/**
*Given an array of integers, every element appears twice except for one. Find that single one.
*Note:
*Your algorithm should have a linear runtime complexity. Could you implement it without using extra memory?
**/
/**空间复杂度O(n),时间复杂度O(n)但是依赖map的性能**/
int LeetCode::singleNumberMap(vector<int>& nums){
hash_map<int,int> hmp;
auto it = nums.cbegin();
while (it != nums.cend()){
auto temp = hmp.find(*it);
if (temp != hmp.cend()){//找得到当前元素,说明它在前面出现过
//(*temp).second = 2;
hmp.erase(temp);//删除重复元素
}
else{
hmp.insert(make_pair(*it,));//插入首次出现的元素
}
++it;
}
if (hmp.size() == ){//如果最终存在singleNumber
auto ret = hmp.begin();
return (*ret).first;
}
return -;//不存在,则返回-1
}
思路:
使用异或的方法;
因为元素是数字,且数组保证只重复两次;可以考虑使用异或的方法,来直接求出单一数字。
这个方法比上面的更高效,且空间复杂度O(1),而上面的空间复杂度O(n)。
/**
空间复杂度O(1),时间复杂度O(n)
使用异或来求单个数字
**/
int LeetCode::singleNumberXOR(vector<int>& nums){
int sum = ;
auto it = nums.cbegin();
while (it != nums.cend()){
sum = sum ^ (*it);//异或
++it;
}
return sum;
}
题目:singleNumberII
Given an array of integers, every element appears three times except for one, which appears exactly once. Find that single one.
Note:
Your algorithm should have a linear runtime complexity. Could you implement it without using extra memory?
给一个数组里面除了唯一的一个数只出现一次,其他每个数都重复了两次(即出现3次)。
要求:线性的时间复杂度,和O(1)的空间复杂度
思路:
使用hash_map同样可以求出来,思路跟第一个是一样的。
但是,如何用异或更高效的找到singleNumber呢?
首先还是对每个元素异或,保留奇数次出现的元素(once);
同时,增加一个变量accumulate |= (*it) & once;
上面的算式能够保留出现2次以上的元素;(it表示当前元素)
例如:once中出现过奇数次则表示保留了该数字,则必然出现2次以上,于是上面表达式的结果确实是保留了当前元素;
once出现了偶数次,表示未保留该数字,则取决于accumulate中是否保留该数字,没有则是0。
因此,上面表达式确实能够保留出现2次以上的元素。
然后出现3次则表示,上面两个值都是保留当前元素的状态,于是在异或一次就消除了出现3次的元素。
int LeetCode::singleNumber(vector<int>& nums){
//once标记出现奇数次的数字,accumulate标记出现2次以上的数字
int once = ,accumulate = ;
auto it = nums.cbegin();
while (it != nums.cend()){
//once中出现过奇数次则表示保留了该数字,则必然出现2次以上;
//once出现了偶数次,表示未保留该数字,则取决于accumulate中是否保留该数字,没有则是0
accumulate |= (*it) & once;// 只要第二次或者以上出现,就为1
once ^= (*it);// 出现奇数次保留,偶数次抛弃
int t = once & accumulate;// 第三次的时候one和accumulation都保留了该位的值
once &= ~t; // 清零出现三次的该位的值
accumulate &= ~t; //once = (once ^ (*it)) & ~accumulate;
//accumulate = (accumulate ^ (*it)) & ~once;
++it;
}
return once;
}
上面一连串的求解,还可以缩减成下面两个表达式:
once = (once ^ (*it)) & ~accumulate;
accumulate = (accumulate ^ (*it)) & ~once;
题目:Single Number III
Given an array of numbers nums
, in which exactly two elements appear only once and all the other elements appear exactly twice. Find the two elements that appear only once.
For example:
Given nums = [1, 2, 1, 3, 2, 5]
, return [3, 5]
.
Note:
- The order of the result is not important. So in the above example,
[5, 3]
is also correct. - Your algorithm should run in linear runtime complexity. Could you implement it using only constant space complexity?
给一个数组里面除了两个数只出现一次,其他每个数都出现了两次。
要求:线性的时间复杂度,和O(1)的空间复杂度。
思路:
设数组为A[],其中两个只出现一次的数字是A,B
根据第一个问题的思路,如果只有一个数字可以直接将所有数字异或得到。因此,该问题如果异或所有数字得到的值是A^B。
那么,如果能将A,B区分开就可以了;异或表示两个数字中不相同的位是1,即如果A^B的某一位是1,表示A和B中对应的该为分别为0和1,或1和0。
反之,只要将该位为1的所有数从新异或一遍,就能的到A或B中的一个数字,于是就区分开了。
vector<int> LeetCode::singleNumber3(vector<int>& nums){
int xorall = , first1 = ;//第一个1
vector<int>result(, );
for (auto i : nums){//求A^B
xorall ^= i;
}
first1 = xorall & (~(xorall - ));//求A^B中第一个1的值,例如A = 4,B = 8,A^B = 12,first1 = (0100 ^ 1000) & (0100) = 4
for (auto i : nums){
if (first1 & i)result[] ^= i;//所有A^B第一个1相同位置为1的元素异或
else result[] ^= i;;
}
return result;
}
[LeetCode]singleNumber的更多相关文章
- LeetCode——single-number系列
LeetCode--single-number系列 Question 1 Given an array of integers, every element appears twice except ...
- leetcode — single-number
/** * Source : https://oj.leetcode.com/problems/single-number/ * * * Given an array of integers, eve ...
- Leetcode SingleNumber I & II & III 136/137/260
SingleNumber I: 题目链接:https://leetcode-cn.com/problems/single-number/ 题意: 给定一个非空整数数组,除了某个元素只出现一次以外,其余 ...
- Leetcode 136 137 260 SingleNumber I II III
Leetccode 136 SingleNumber I Given an array of integers, every element appears twice except for one. ...
- LeetCode 2: single-number II
Given an array of integers, every element appears three times except for one. Find that single one. ...
- LeetCode 1: single-number
Given an array of integers, every element appears twice except for one. Find that single one. soluti ...
- single-number leetcode C++
Given an array of integers, every element appears twice except for one. Find that single one. Note: ...
- [LeetCode] Single Number III 单独的数字之三
Given an array of numbers nums, in which exactly two elements appear only once and all the other ele ...
- [LeetCode] Single Number II 单独的数字之二
Given an array of integers, every element appears three times except for one. Find that single one. ...
随机推荐
- 使用Counter进行计数统计
使用Counter进行计数统计 想必大家对计数统计都不陌生吧!,简单的说就是统计某一项出现的次数.实际应用中很多需求都需要用到这个模型,如检测样本中某一值出现的次数.日志分析某一消息出现的频率分析文件 ...
- Spring学习之旅(十二)--持久化框架
对于本职工作来说 JDBC 就可以很好的完成,但是当我们对持久化的需求变得更复杂时,如: 延迟加载 预先抓取 级联 JDBC 就不能满足了,我们需要使用 ORM框架 来实现这些需求. Spring 对 ...
- Python多进程的Join和daemon(守护)的用法
join和daemon 下面仅以多进程为例: 知识点一: 当一个进程启动之后,会默认产生一个主线程,因为线程是程序执行流的最小单元,当设置多线程时,主线程会创建多个子线程,在python中,默认情况下 ...
- 你知道JavaScript这六种错误类型吗?
前言 今日话题,了解JavaScript的错误处理机制. 一.ReferenceError 引用一个不存在的变量时发生的错误.将一个值分配给无法分配的对象,比如对函数的运行结果或者函数赋值. 举栗子 ...
- CSS 之Grid 网格知识梳理2
继上篇的CSS 之Grid下半部分 14.将单元格划分到一个区域,使用grid-template-areas属性: ag: grid-template-areas: "header h ...
- Angular Material 的设计之美
前言 Angular Material 作为 Angular 的官方组件库,无论是设计交互还是易用性都有着极高的质量.正如官方所说其目的就是构建基于 Angular 和 Typescript 的高质量 ...
- 美团张志桐:美团 HTTP 服务治理实践
2019 年 7 月 6 日,OpenResty 社区联合又拍云,举办 OpenResty × Open Talk 全国巡回沙龙·上海站,美团基础架构部技术专家张志桐在活动上做了<美团 HTTP ...
- Educational Codeforces Round 48 D Vasya And The Matrix
EDU #48 D 题意:给定一个矩阵,已知每一行和每一列上数字的异或和,问矩阵上的数字是多少,不存在则输出NO. 思路:构造题,可以考虑只填最后一行,和最后一列,其中(n,m)要特判一下.其他格子给 ...
- 模板汇总——ST(暂)
int Log[N]; struct ST { ], a[N]; void init(int n) { ]=-); i < N; i++) Log[i] = Log[i - ] + ((i &a ...
- CodeForces Round#480 div3 第2场
这次div3比上次多一道, 也加了半小时, 说区分不出1600以上的水平.(我也不清楚). A. Remove Duplicates 题意:给你一个数组,删除这个数组中相同的元素, 并且保留右边的元素 ...