알고리즘

#include <algorithm>로 사용할 수 있습니다.

표준 라이브러리에서 제공하는 다양한 알고리즘 함수를 포함하고 있습니다.
정렬, 탐색, 변환, 수치 연산 등의 기능을 제공합니다.

반복적인 알고리즘을 직접 구현할 필요 없이 표준화된 함수를 사용해 코드의 가독성과 효율성을 높일 수 있습니다.
코딩 테스트나 알고리즘 문제를 풀다 보면 모든 코드를 구현하기에 시간이 부족할 수 있습니다.
해당 헤더에서 제공하는 기능들을 사용해서 시간을 효율적으로 사용하는 것이 좋습니다.

next_permutation & prev_permutation

next_permutation함수와 prev_permutation함수는 순열 생성 함수입니다.
해당 함수를 사용하기 위해서는 원본 데이터가 정렬된 상태여야 합니다.

한 번 호출될 때 $O(n)$, 모든 순열을 생성하려면 $O(n!)$의 시간 복잡도를 가집니다.

next_permutation함수는 주어진 범위를 사전순(lexicographical)으로 다음 순열로 바꿔줍니다.
prev_permutation함수는 주어진 범위를 사전순으로 이전 순열로 바꿔줍니다.

bool std::next_permutation(BidirectionalIterator first, BidirectionalIterator last);
bool std::prev_permutation(BidirectionalIterator first, BidirectionalIterator last);

first는 순열의 시작 반복자, last는 순열의 끝 반복자입니다.
next_permutation함수는 다음 순열이 존재한다면 true, 마지막 순열이었다면 false를 반환하고, 배열을 처음 순열(오름차순)로 변경합니다.
prev_permutation함수는 이전 순열이 존재한다면 true, 첫 번째 순열이었다면 false를 반환하고, 배열의 마지막 순열(내림차순)로 변경합니다.

예시 #1

#include <iostream>
#include <algorithm>
#include <vector>

using namespace std;

int main()
{
    vector<int> vec = {1, 2, 3};

    do
    {
        for (int x : vec)
        {
            cout << x << " ";
        }
        cout << endl;
    } while (next_permutation(vec.begin(), vec.end())); // 다음 순열이 존재하는 한 계속 탐색
}
1 2 3
1 3 2
2 1 3
2 3 1
3 1 2
3 2 1

예시 #2
위의 순열을 거꾸로 출력한다면 다음과 같습니다.

#include <iostream>
#include <algorithm>
#include <vector>

using namespace std;

int main()
{
	vector<int> vec = { 1, 2, 3 };

	sort(vec.begin(), vec.end(), greater<int>());	// {3, 2, 1}로 시작

	do
	{
		for (int e : vec)
		{
			cout << e << " ";
		}

		cout << endl;
	} while (prev_permutation(vec.begin(), vec.end()));
}

출력은 다음과 같습니다.

3 2 1
3 1 2
2 3 1
2 1 3
1 3 2
1 2 3

예시 #3
배열 {1, 2, 3, 4}에서 길이가 3인 순열을 모두 출력합니다.
단, 순열의 첫 번째 원소는 항상 1이어야 합니다.

#include <iostream>
#include <algorithm>
#include <vector>

using namespace std;

int main()
{
	vector<int> vec = { 1, 2, 3, 4 };

	do
	{
		for (int e : vec)
		{
			cout << e << " ";
		}

		cout << endl;
	} while (next_permutation(vec.begin() + 1, vec.end()));
}

출력은 다음과 같습니다.

1 2 3 4
1 2 4 3
1 3 2 4
1 3 4 2
1 4 2 3
1 4 3 2

nth_element

nth_element함수는 주어진 범위에서 n번째 원소를 찾고, 이를 기준으로 왼쪽은 n번째보다 작은 원소들, 오른쪽은 n번째보다 큰 원소들로 부분 정렬을 수행합니다.

평균 적으로 $O(n)$, 최악의 경우 $O(n^2)$의 시간 복잡도를 가집니다.

void std::nth_element(RandomIt first, RandomIt nth, RandomIt last);
void std::nth_element(RandomIt first, RandomIt nth, RandomIt last, Compare comp);

first는 정렬 범위의 시작 반복자, last는 정렬 범위의 끝 반복자입니다.
nthnth번째 위치로 0-based index입니다.
comp는 비교 함수로 기본값은 오름차순(<)입니다.

#include <iostream>
#include <algorithm>
#include <vector>

using namespace std;

int main()
{
	vector<int> vec = { 15, 3, 9, 8, 5, 2, 10, 7, 6 };
	vector<int> coutVec;

	for (int e : vec)
	{
        // 짝수만
		if (e % 2 == 0)
		{
			coutVec.push_back(e);
		}
	}

	nth_element(coutVec.begin(), coutVec.begin() + 2, coutVec.end());

	cout << coutVec[2] << endl; // 8
}

unique

unique함수는 연속된 중복 요소를 제거한 끝 위치를 반환하는 함수입니다.
중복되지 않은 요소는 컨테이너의 앞부분으로 이동시키고, 중복된 요소는 삭제하지 않고 뒤쪽에 위치하게 합니다.
중복을 제거한 후 실제 컨테이너의 크기를 변경하려면 erase함수를 사용해주어야 합니다.

해당 함수를 사용하기 위해서는 원본 데이터가 정렬된 상태여야 합니다.

template <class ForwardIt>
ForwardIt unique(ForwardIt first, ForwardIt last);
template <class ForwardIt, class BinaryPred>
ForwardIt unique(ForwardIt first, ForwardIt last, BinaryPred p);

firstlast는 연산을 수행할 범위의 시작과 끝을 나타내는 반복자입니다.
ForwardIt는 중복 제거 후 유효한 범위의 끝(iterator)을 반환합니다.
BinaryPredtrue를 반환하면 두 요소가 중복된 것으로 간주됩니다.

예시 #1

#include <algorithm>
#include <vector>

int main()
{
    std::vector<int> v = { 1, 4, 4, 2, 2, 2, 5, 1 };
    std::sort(v.begin(), v.end()); // 정렬 {1, 1, 2, 2, 2, 4, 4, 5}

    auto newEnd = std::unique(v.begin(), v.end()); // {1, 2, 4, 5, ?, ?, ?, ?}
    v.erase(newEnd, v.end()); // 중복되는 구간 삭제
}

lower_bound & upper_bound

lower_bound함수는 특정 값 이상의 값이 처음 나오는 위치를 찾습니다.
upper_bound함수는 특정 값 초과의 값이 처음 나오는 위치를 찾습니다.

이진 탐색을 사용합니다.

시간 복잡도는 $O(log n)$을 가집니다.

template <class ForwardIt, class T>
ForwardIt lower_bound(ForwardIt first, ForwardIt last, const T& value);
template <class ForwardIt, class T>
ForwardIt upper_bound(ForwardIt first, ForwardIt last, const T& value);

firstlast는 연산을 수행할 범위의 시작과 끝을 나타내는 반복자입니다.
value는 비교하려는 값입니다.
반환값인 ForwardIt는 특정 값 이상 혹은 초과의 값이 처음 나오는 위치를 찾지만, 만약 찾지 못하는 경우 last를 반환합니다.

예시 #1

#include <iostream>
#include <algorithm>
#include <vector>

int main()
{
    std::vector<int> v = { 1, 2, 4, 4, 5, 7, 8 };

    auto lb = std::lower_bound(v.begin(), v.end(), 4);
    auto ub = std::upper_bound(v.begin(), v.end(), 4);

    std::cout << lb - v.begin() << std::endl; // 2
    std::cout << ub - v.begin() << std::endl; // 4
}

예시 #2
특정 값을 가진 원소의 개수를 구하는 방법입니다.

#include <iostream>
#include <vector>
#include <algorithm>

int main()
{
    std::vector<int> vec = { 1, 3, 3, 3, 5, 7, 9 };

    int x = 3;
    int count = std::upper_bound(vec.begin(), vec.end(), x) - std::lower_bound(vec.begin(), vec.end(), x);

    std::cout << count << std::endl; // 3
}

transform

transform함수는 특정 구간 내의 원소들을 원하는 대로 변환시켜줍니다.

// 단항
template <class InputIt, class OutputIt, class UnaryOp>
OutputIt transform(InputIt first1, InputIt last1, OutputIt d_first, UnaryOp unary_op);

first1last1은 연산을 수행할 범위의 시작과 끝을 나타내는 반복자입니다.
d_first은 연산 결과를 저장할 컨테이너의 시작 반복자입니다.
unary_op는 단일 인자를 받는 함수 또는 람다식입니다.

// 이항
template <class InputIt1, class InputIt2, class OutputIt, class BinaryOp>
OutputIt transform(InputIt1 first1, InputIt1 last1, InputIt2 first2, OutputIt d_first, BinaryOp binary_op);

first1last1은 연산을 수행할 범위의 시작과 끝을 나타내는 반복자입니다.
first2first1과 함께 연산될 입력입니다.
d_first은 연산 결과를 저장할 컨테이너의 시작 반복자입니다.
binary_op는 두 개의 인자를 받는 함수 또는 람다식입니다.

예시 #1
단항 연산을 하는 방법입니다.

#include <iostream>
#include <vector>
#include <algorithm>
#include <cmath> // pow 함수

using namespace std;

int main()
{
    vector<int> v1 = { 1, 2, 3, 4, 5 };
    vector<int> v2(5); // v1과 같은 크기

    transform(v1.begin(), v1.end(), v2.begin(), [](int x)
    {
        return pow(x, 2); // 각 원소를 제곱
    });

    for(int x : v2) cout << x << " "; // 1 4 9 16 25
}

예시 #2
이항 연산을 하는 방법입니다.

#include <iostream>
#include <vector>
#include <algorithm>
#include <numeric> // plus<int>

using namespace std;

int main() {
    vector<int> v1 = { 1, 2, 3, 4, 5 };
    vector<int> v2 = { 6, 7, 8, 9, 10 };
    vector<int> v3(5);

    transform(v1.begin(), v1.end(), v2.begin(), v3.begin(), plus<int>());

    for (int x : v3) cout << x << " "; // 7 9 11 13 15
}

min_element & max_element & minmax_element

주어진 범위에서 최소값, 최대값 또는 최소값과 최대값의 위치를 반환하는 함수입니다.

// min_element & max_element
template <class ForwardIt>
ForwardIt min_element(ForwardIt first, ForwardIt last);
template <class ForwardIt, class Compare>
ForwardIt max_element(ForwardIt first, ForwardIt last, Compare comp);

firstlast는 연산을 수행할 범위의 시작과 끝을 나타내는 반복자입니다.
UnaryPred는 비교 함수로, 람다 함수 또한 사용될 수 있습니다.
반환값인 ForwardIt는 최소값 혹은 최대값을 가지는 iterator이며, 만약 찾지 못하는 경우 end에 해당하는 iterator를 반환합니다.

예시 #1

#include <iostream>
#include <string>
#include <vector>
#include <algorithm>

using namespace std;

int main()
{
    vector<int> v = {4, 1, 6, 3, 8, 2};

    // max_element 최댓값의 위치를 찾아줍니다
    auto max_it = max_element(v.begin(), v.end());
    cout << *max_it << endl; // 8
    
    // min_element 최솟값의 위치를 찾아줍니다
    auto min_it = min_element(v.begin(), v.end());
    cout << *min_it << endl; // 1
    
    // minmax_element 최솟값과 최댓값의 위치를 찾아줍니다.
    auto [min_iter, max_iter] = minmax_element(v.begin(), v.end());
    cout << *max_iter << endl; // 8
    cout << *min_iter << endl; // 1
}

count & count_if

count함수는 특정 값을 가진 원소의 개수를 반환합니다.
count_if함수는 특정 조건을 만족하는 원소의 개수를 반환합니다.

template <class InputIt, class T>
typename std::iterator_traits<InputIt>::difference_type 
count(InputIt first, InputIt last, const T& value);

template <class InputIt, class UnaryPred>
typename std::iterator_traits<InputIt>::difference_type 
count_if(InputIt first, InputIt last, UnaryPred p);

firstlast는 연산을 수행할 범위의 시작과 끝을 나타내는 반복자입니다.
value는 개수를 세고자하는 값입니다.
UnaryPred는 비교 함수로, 람다 함수 또한 사용될 수 있습니다.
std::iterator_traits<InputIt>::difference_type는 반환 자료형으로 일반적으로 int 혹은 long타입입니다.

예시 #1

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

int main()
{
    vector<int> v = { 1, 1, 2, 2, 3, 3, 4 };

    // 값이 2인 원소 개수
    int cnt = count(v.begin(), v.end(), 2);

    // 짝수 원소 개수
    int even_cnt = count_if(v.begin(), v.end(), [](int x) { return x % 2 == 0; });

    cout << cnt << endl;    // 2
    cout << even_cnt << endl;   // 3
}

Date:     Updated:

카테고리:

태그:

Cpp 카테고리 내 다른 글 보러가기

댓글남기기