“This is the second day of my participation in the Gwen Challenge in November. See details: The Last Gwen Challenge in 2021”
Topic:
Refer to Offer II 004. A number that appears only once
Given an array of integers, nums, each element appears exactly three times except for one element. Find and return the element that appears only once.
Example 1:
Input: nums = [2,2,3,2Copy the code
Example 2:
Input: nums = [0,1,0,1,0,1,100] output: 100Copy the code
Tip:
1 <= nums.length <= 3 * 104
-231 <= nums[i] <= 231 - 1
nums
In, except for an element only occursAt a timeOutside, every other element appears exactlyThree times
Advanced: Your algorithm should have linear time complexity. Can you do it without using extra space?
Method 1:
- with
Map
Records the number of occurrences of each element - traverse
Map
findvalue
A value of1
The elements of the
Method 1:
var singleNumber = function (nums) {
let map = new Map(a);// Count the number of occurrences of each element
for (let i = 0; i < nums.length; i++) {
map.set(nums[i], (map.get(nums[i]) || 0) + 1);
}
// Find the element with value 1 in our map
for (let [key] of map) {
if (map.get(key) === 1) {
returnkey; }}};Copy the code
Method 2 Thinking
- Use binary, so that each position has only 0 and 1 values in 32 bits
- If I go over the sum of each position, I get 3 divisible by 3, and I get 1 divisible by 3
- Define a variable
result
, records the result of each position divisible by 30 | | 1
- Starting with the last digit, the current position of the updated result is solved in each round
- At the end of the loop, the result is the value that we’re looking for once
Method 2 implementation
var singleNumber = function (nums) {
let result = 0;
const n = nums.length;
// Binary is 32 bits, judge the value of each bit
for (let i = 0; i < 32; i++) {
// Sum the values of each digit
let total = 0;
for (let j = 0; j < n; j++) {
// (nums[j] >> I) &1 => get the current value
total += (nums[j] >> i) & 1;
}
// If the current bit is not divisible by 3, the element that occurs once has a value in the current bit
if (total % 3! = =0) {
// Push the current bit to the result
result += (total % 3) << i; }}return result;
};
Copy the code
conclusion
Method 1 is the easiest way to do it, and of course the next step in this problem is to use constant space. Like when the array is really long our map space will overflow. So this is where we use bit operations to solve the problem. There are several knowledge points in the middle bit operation:
-
Floor (nums[j] / math.pow (2, I)). It’s basically the current value divided by 2 to the I, rounded down.
-
(nums[j] >> I) the left part of &1 has been explained in the previous step, this step calls the left part of n, n & 1 can find whether n is odd or even, n & 1 = 1 when n is odd, n & 1 = 0 when n is even.
-
Result += (total % 3) << I We iterate from the last digit, which is equivalent to unshift one digit at a time in binary to the original array. For example, 10 => 110 => 1110; Except for this one, we’re going to return base 10, so we’ll just use +=; That’s 2 plus 4 plus 8 plus 16.
If you understand it, you can click on it and see you in the next topic. If there is no accident after the article will be in this form, there are good suggestions welcome to comment area.