농담곰담곰이의곰담농

[인프런] 비트 마스킹

by 브이담곰
📘 10주완성 C++ 코딩테스트 강의를 수강한 후 정리한 글입니다.

※ 비트 연산자 활용법

1. idx번째 비트 끄기

 

ex)

S = 10010 (18) 에서 1번째 비트를 끄려면,

 

먼저 00010을 만든다. ( 1 << 1 )

 

그 다음 해당 비트를 뒤집는다.11101

 

여기서, S = 10010과 & 연산을 하면,

10010
&
11101 =
10000 이 된다.

 

 

따라서,

idx번째 비트 끄기 : S &= - (1 << idx )

 

 

코드

더보기
#include <bits/stdc++.h>
using namespace std;
int main(){
int S = 18;
int idx = 1;
S &= - ( S << idx );
cout << S << '\n'; //16
return 0;
}

 

 

2. idx번째 비트 XOR 연산 

: 해당 비트가 0이면 1, 1이면 0으로 만든다.

 

S = 10010 (18)

 

ex 1) 1번째 비트 0으로 만들기.

 

먼저 1 << 1 을 통해 00010 을 만든다.

 

10010 
^
00010 =
10000

 

ex 2) 0번째 비트 1로 만들기.

 

1 << 0 = 00001

 

10010
^
00001 = 
10001

 

 

따라서,

idx번째 비트 XOR 연산 : S ^= ( 1 << idx )

 

 

코드

더보기
#include <bits/stdc++.h>
using namespace std;
int main(){
int S = 10;
int idx = 1;
S ^= ( 1 << idx );
cout << S << '\n'; //16
return 0;
}

 

3. 최하위 켜져있는 비트 찾기

 

10010 : 최하위 켜져있는 비트 = 1번째 비트

10000:  최하위 켜져있는 비트 = 4번째 비트

→ 오른쪽에서부터 탐색하여 1을 찾아서 해당 인덱스를 찾으면 된다.

 

ex)

S = 10010

 

~ S = -  (S + 1) 이기 때문에,

01101 + 1 =  - S

01110 = - S

 

01110
&
10010=
00010

 

 

최하위 켜져있는 비트 찾기:  idx = (S & -S)

 

 

코드

더보기
#include <bits/stdc++>
using namespace std;
int main(){
int S = 18;
int idx = ( S & -S );
cout << idx << '\n'; //2
return 0;
}

 

 

4. 크기가 n인 집합의 모든 비트를 켜기

배열 크기가 4인 집합을  [1,1,1,1]이 아닌, 15로 "표현" 할 수 있다.

 

크기가 4라면

16(1<< 6) 이 되고

(1<<4) - 1 = 15(1111) 가 된다.

 

따라서,

크기가 n인 집합의 모든 비트를 켜기 : ( 1 << n ) - 1

 

 

코드

더보기
#include <bits/stdc++.h>
using namespace std;
int main(){
int n = 4;
cout << ( 1 << n ) - 1 << '\n' ; //15
return 0;
}

 

5. idx번째 비트를 켜기

 

ex)

S = 10010(18) 에서, 0번째 비트를 키면

10011(19)가 된다.

 

1 << 0 = 00001 과 S를 OR 연산을 하면

10010
|
00001 =
10011

 

따라서,

idx번째 비트를 켜기 : S |= ( 1 << idx )

 

코드

더보기
#include <bits/stdc++>
using namespace std;
int main(){
int S = 18;
int idx = 0;
S |= ( 1 << idx );
cout << S << '\n' ; //19
return 0;
}

 

 

6. idx번째 비트가 켜져있는지 확인하기

 

ex)

S = 10010에서 3번째 비트가 켜져있는지 확인하려면

10010
&
00100 = 
00000

결과 값이 0이므로 False가 된다.

 

따라서,

idx번째 비트가 켜져있는지 확인하기 : if(S & ( 1 << idx))

 

 

코드

더보기
#include <bits/stdc++.h>
using namespace std;
int main(){
int S = 10;
int idx = 0;
if( S & ( 1 << idx ))
cout << "해당 idx :" << idx << "가 켜져있습니다.\n";
else
cout << "해당 idx :" << idx << "가 꺼져있습니다.\n";
return 0;
}

 

비트마스킹 : 경우의 수, 매개변수

 

어떤 특정원소를 찾을 때의 시간 복잡도.

- 배열의 어떤 요소를 찾을 때 선형적인 시간 : O(N)

- sorted array에서 이분탐색으로 찾을 때 : O(logN)

- 해싱테이블 : O(1)

 

vector<bool>, bitset, set<int>로 표현해서 무언가를 찾는 연산을 하지만, 비트 연산을 쓰면 가볍고 빠르게 할 수 있다.

불리언배열의 역할을 하는 "하나의 숫자"를 만들어서 "비트 연산자"를 통해 탐색, 수정 등의 작업을 하는 것을 비트 마스킹이라고 한다.

 

비트마스킹을 이용한 경우의 수

ex) { 딸기, 사과, 포도 배 }의 모든 경우의 수는?

각각의 요소는 2개의 상태값(포함, 불포함) 을 갖기 때문에 16가지의 경우의 수가 있다.

 

#include <bits/stdc++.h>
using namespace std;
const int n = 4;
int main()
{
string a[n] = {"사과", "딸기", "포도", "배"};
for(int i = 0; i < ( 1 << n ); i++)
{
string ret = "";
for(int j = 0; j < n; j++){
if(i & ( 1<<j)){
ret += (a[j] + " ");
}
}
return 0;
}

 

블로그의 프로필 사진

블로그의 정보

농담곰담곰이의곰담농

브이담곰

활동하기