보편 참조는 참조 정보에 대해 부호화가 된다. 왼값의 경우에는 왼값 참조로 연역되고 오른값의 경우에는 비참조 형식으로 연역된다. Widget widgetFactory(); // 오른값을 돌려주는 함수 Widget w; func(w); func(widgetFactory()); 보편 참조를 사용하는 func 함수는 인수에 따라 연역되는 형식이 달라진다. func 함수에 왼값을 넘겨주면 다음과 같이 된다. void func(Widget& && param); // 최종 결과 void func(Widget& param); Widget& && 에서 Widget& 로 변경되는데 그 이유는 바로 참조 죽약 때문이다. 참조에 대한 참조를 사용하게 되면 참조 축약이 일어나는데 다음의 경우의 수가 있다. 왼값에 대한 왼값 ..
항목 26에서 나온 문제에 대하여 해결 방법을 알아보자. 2019/12/01 - [C++/Effective Modern C++] - [Effective Modern C++] 항목 26. 보편 참조에 대한 중복적재를 피하라 불러오는 중입니다... 중복적재를 포기한다 항목 26에서 사용한 logAndAdd 함수를 중복적재를 사용하다보니 문제가 발생하였다. 문제가 발생하는 함수에 대하여 함수 명을 logAnddAddNameIdx 와 같이 변경하면 문제를 해결할 수 있다. 그렇지만 생성자의 경우에는 함수명을 변경할 수 없기 때문에 해결할 수 없는 경우도 있다. const T& 매개변수를 사용한다. 보편 참조 매개변수 대신에 const에 대한 왼값 참조 매개변수를 사용할 수 있다. 이는 항목 26에서 사용한 방법..
이전 항목에서 우리는 보편 참조에 대한 중복적재의 문제점에 대해 이야기하였고 문제를 해결하기도 하였다. 다음과 같이 말이다. template void logAndAdd(T&& name) { auto now = std:;chrono::system_clock::now(); log(now, "logAndAdd"); names.emplace(std:;forward(name)); } std::string petName("Darla"); logAndAdd(petName); logAndAdd(std::string("Persephone")); logAndAdd("Patty Dog"); 위의 코드는 어떤 상황에서든 함수 하나로 우리가 원하대로 수행해준다. 그런데 여기서 오버로딩 함수 하나를 더 추가한다고 해보자. voi..
오른값 참조는 이동할 수 있는 객체에만 묶인다. 다시 말하면 어떤 매개변수가 오른값 참조라면, 객체를 이동할 수 있다는 의미이다. 오른값 참조로 캐스팅해주는 객체가 바로 std::move 함수이다. 반면에 보편 참조는 오른값 참조일수도 있지만 아닐수도 있다. 오른값일때만 오른값으로 캐스팅해주어야하는데 이 역할을 하는 함수가 std::forward 이다. 오른값 참조일때도 std::forward 를 사용할 수 있지만, 실수의 여지가 생기며 관용구에서 벗어나기 때문에 사용하지 않는것이 좋다. 그런데 보편 참조에 std::move 를 사용하는 것은 매우 위험하다. 코드를 살펴보자. class Widget { public: template void setName(T&& newName) // 보편 참조 { nam..
오른값 참조를 사용할 때, 우리는 &&를 뒤에 붙여 오른값 참조임을 명시한다. 하지만 보편 참조의 경우에도 &&를 붙이기 때문에 구분해서 사용해야만 한다. 예시를 살펴보자. void f(Widget&& param); // 오른값 참조 Widget&& var1 = Widget(); // 오른값 참조 auto&& var2 = var1; // 오른값 참조 아님 template void f(std::vector&& param); // 오른값 참조 template void f(T&& param); // 오른값 참조 아님 언뜻보기에는 오른값 참조처럼 보이지만 보편 참조인 경우가 상당히 많다. 보편 참조의 의미는 오른값 참조이거나 왼값 참조 중 하나라는 의미이다. 보편 참조는 일반적으로 두 가지 문맥에서 나타난다. ..
C++11 에 등장한 move semantic 에 중요한 함수 두가지가 있다. 바로 std::move 와 std::forward 함수이다. std::forward 는 std::move 함수를 이해하면 쉽게 이해할 수 있으므로 먼저 std::move 에 대해서 알아보자. std::move 함수의 이름만 보면 std::move 가 이동을 수행해줄 것처럼 보이지만 사실 타입 캐스팅이 전부이다. 그런데도 이 함수의 이름이 move 인 이유는 오른값으로 캐스팅을 하기 때문이다. std::move 를 구현하면 다음과 같이 구현할 수 있다. template decltype(auto) move(T&& param) { using ReturnType = remove_reference_t&&; return static_ca..
Pimpl 이란 "pointer to implementation" 을 의미한다. 코드를 먼저 살펴보자. 기존코드 class Widget { public: Widget(); ... private: std::string name; std:;vector data; Gadget g1, g2, g3; }; Pimpl 적용 코드 class Widget { public: Widget(); ~Widget(); ... private: struct Impl; Impl *pImpl; }; Pimpl 적용 코드를 보면 std::string, std::vector, Gadget 을 선언하지 않고 있기 때문에 #include 할 필요가 없어졌다. 덕분에 컴파일 속도가 빨라지게 되었다. 이렇게 선언만 하고 정의를 하지 않는 형식을..
선호 이유 간결한 코드 설명을 하기 전에 코드를 먼저 살펴보자. auto spw1(std::make_shared()); std::shared_ptr spw2(new Widget()); 먼저 코드의 길이부터 약간이지만 더 짧아진 것을 확인할 수 있다. 또한 타입 선언을 한 번만 작성하는 곳을 볼 수 있는데, 이는 소프트웨어 공학의 핵심 교의 중 하나인 "코드 중복을 피하라"를 잘 지키는 것이다. 소스 코드의 중복이 많으면 컴파일 시간이 늘어나며 일관성이 없는 코드로 진화하기 일쑤이다. 코드의 비일관성은 버그로 이어지는 경우가 많다. 예외 안정성 이 역시 코드를 먼저 살펴보자. void processWidget(std::shared_ptr spw, int priority); int computePriorit..
- Total
- Today
- Yesterday
- 다이소
- Modern
- 보편 참조
- MOVE
- 포인터
- 발아시기
- std::forward
- detach
- CPP
- Effective
- Future
- Join
- Effective Modern C++
- auto
- C++
- std::move
- Perfect
- async
- Overloading
- 람다
- C++14
- Forwarding
- C++11
- 보편참조
- Unreal
- forward
- const
- thread
- Override
- C
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |