-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy path05_unique_number_II.cpp
144 lines (112 loc) · 3.58 KB
/
05_unique_number_II.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
/*
Problem Name: Unique Numbers - II
We are given an array containg n numbers.
All the numbers are present twice except for two numbers which are present only once.
Find the unique numbers in linear time without using any extra space.
(Hint - Use Bitwise)
Input Format: First line contains the number n.
Second line contains n space separated number.
Constraints: n < 10^5
Output Format: Output a single line containing the unique numbers separated by space
Sample Input: 4
3 1 2 1
Sample Output: 2 3
Explanation: Smaller number comes before larger number
*/
#include <iostream>
using namespace std;
// function to find unique numbers
void getUniqueNumbers(int arr[], int size)
{
int unique_xor_val = 0; // to store xor value of both unique numbers
int first_num = 0;
int second_num = 0;
// finding the xor of both unique numbers
for(int i=0; i<=size-1; i++)
{
unique_xor_val = unique_xor_val^arr[i];
}
// finding the position of first set bit in unique_xor_val
int temp = unique_xor_val;
int pos = 0;
while((temp&1) != 1)
{
pos++;
temp = temp >> 1;
}
// find those numbers which contains set bit at position 'pos'
int mask = 1<<pos;
for(int i=0; i<=size-1; i++)
{
if(arr[i]&mask)
{
first_num = first_num^arr[i];
}
}
second_num = unique_xor_val ^ first_num;
cout << "Unique Numbers are: ";
cout << min(first_num, second_num) << " " << max(first_num, second_num);
}
// function to drive code
int main()
{
int total_num;
cout << "Enter total numbers: ";
cin >> total_num;
int arr[total_num];
cout << "Enter numbers: ";
for(int i=0; i<=total_num-1; i++)
{
cin >> arr[i];
}
// finding the unique numbers
getUniqueNumbers(arr, total_num);
cout << endl;
return 0;
}
/*
OUTPUT:
Case 1:
Enter total numbers: 4
Enter numbers: 3 1 2 1
Unique Numbers are: 2 3
Case 2:
Enter total numbers: 8
Enter numbers: 5 1 2 1 2 3 5 7
Unique Numbers are: 3 7
APPROACH:
Once again, we need to use XOR to solve this problem. But this time, we need to do it in two passes:
In the first pass, we XOR all elements in the array, and get the XOR of the two numbers we need to find.
Note that since the two numbers are distinct, so there must be a set bit (that is, the bit with
value '1') in the XOR result. Find out an arbitrary set bit (for example, the rightmost set bit).
In the second pass, we divide all numbers into two groups, one with the aforementioned bit set,
another with the aforementinoed bit unset. Two different numbers we need to find must fall into
the two distrinct groups. XOR numbers in each group, we can find a number in either group.
Code: (JAVA)
public int[] singleNumber(int[] nums)
{
// Pass 1 :
// Get the XOR of the two numbers we need to find
int diff = 0;
for (int num : nums)
{
diff ^= num;
}
// Get its last set bit
diff &= -diff;
// Pass 2 :
int[] rets = {0, 0}; // this array stores the two numbers we will return
for (int num : nums)
{
if ((num & diff) == 0) // the bit is not set
{
rets[0] ^= num;
}
else // the bit is set
{
rets[1] ^= num;
}
}
return rets;
}
*/