클래스 생성자에 람다함수 전달시 에러가 뜨네요
글쓴이: hendrix55 / 작성시간: 목, 2023/07/27 - 8:36오후
typedef double Fct(double); struct Function { Function(Fct f, double, double, Point, int, double, double); ... } int main() { for (int n=0; n<50; ++n) { ... Function e {[n](double x){ return expe(x,n); }, ...}; }
안중요한 부분은 ...로 생략햇는데요
Function 생성자에 [](double x){ return cos(x); } 이렇게 하면 괜찮은데 람다 개시자에 n 넣으니까 에러가 뜨는데 왜이런거죠
람다를 따로 공부한게 아니라서 책에 람다를 잠깐 설명하면서 코드에 넣어놓은걸 실습하고 잇는데 에러가 떠서 해결이 안되네요..
Forums:
어떤 언어인지 안적었는데 C++인거 같으니
어떤 언어인지 안적었는데 C++인거 같으니
저 역시 안해봐서 인터넷 참고 했으니 틀릴 수도 있습니다
https://learn.microsoft.com/ko-kr/cpp/cpp/lambda-expressions-in-cpp?view=msvc-170
대괄호 내용이 잘 못 된거 같네요
[&, n]
일 듯
그리고 질문할 때는 에러 메시지도 같이 붙여주세요
--=--=--=--=--=--=--=--=--=--=--=--=--=--=--=--
N년째 초보입니다
설명하기 쉽지 않은 이야기
설명하기 쉽지 않은 이야기
1. capture가 없는 lambda는 전통적인 C/C++ 함수로 볼 수 있고, 그렇게 쓸 수 있습니다.
lambda 표현식에 capture list가 비어 있을 경우, C++는 해당 lambda 표현식의 type에 예컨대 아래와 같은 암시적 변환 연산자를 작성해주지요.
그래서 이 경우 lambda 표현식을 전통적인 함수 타입, 즉
typedef double Fct(double);
과 같은 타입으로 받을 수 있는 것입니다.2. 하지만 원래 lambda는 전통적인 함수가 아니에요.
원래 lambda 표현식의 타입은 함수 호출 연산자(
operator()
)가 오버라이드 된 객체 타입입니다.C++11 이전에 STL을 많이 사용하셨다면 함수자(Functor)라는 이름으로 익숙하게 많이 만들어 보셨을 거에요. 함수처럼 생기고 함수처럼 쓰이지만 함수가 아닙니다.
대표적인 차이점으로 객체는 멤버 변수(필드)를 가질 수가 있고, 그래서 capture를 할 수 있는 거죠.
capture를 단 하나라도 하게 되면 그 객체는 더 이상 함수처럼 쓸 수가 없고, 그래서 함수로 변환할 수 없어서 에러가 나는 겁니다.
3. 해결하려면?
뭐 대충 이런 게 떠오르네요.
(1) 함수 매개변수를 받는 부분을 템플릿화 합니다.
STL 컨테이너/알고리즘들이 이런 식으로 되어 있는데, 템플릿은 굉장히 강력하고 유연한 기능이라서 경험이 적다면 원하는 대로 사용하기가 조금 까다롭습니다.
(2)
std::function
을 씁니다.이것도 어떻게 보면 템플릿을 쓰는 거긴 한데 골치아픈 부분을 C++ library에 다 위임해 버리는 거죠.
(3) Fct를
operator()
가 순수 가상 함수인 인터페이스 클래스로 만듭니다.사용자가 그걸 상속받아서 구현하게 하는 거죠. main 함수는 Fct &으로 객체를 받으면 됩니다.
아예 함수처럼 보이기를 포기하고 객체의 규칙을 따르는 거죠.
대신 이런 방법을 쓰면 더 이상 전통적인 함수 포인터를 받을 수는 없습니다. 물론 따로 오버로딩하면 되긴 하죠.
댓글 달기