>백엔드 개발 >C++ >C++ 구문 오류: 멤버 함수 포인터가 비멤버 함수를 가리킬 수 없습니다. 어떻게 처리합니까?

C++ 구문 오류: 멤버 함수 포인터가 비멤버 함수를 가리킬 수 없습니다. 어떻게 처리합니까?

王林
王林원래의
2023-08-22 16:43:471544검색

C++ 구문 오류: 멤버 함수 포인터가 비멤버 함수를 가리킬 수 없습니다. 어떻게 처리합니까?

C++ 프로그래밍에서 멤버 함수 포인터는 클래스의 멤버 함수에 대한 포인터입니다. 멤버 함수 포인터를 사용하면 런타임에 호출할 멤버 함수를 동적으로 선택할 수 있으며 이는 매우 유용한 기술입니다. 그러나 때때로 멤버 함수 포인터가 멤버가 아닌 함수를 가리킬 수 없는 문제가 발생합니다. 이를 어떻게 처리해야 할까요?

먼저 멤버 함수 포인터가 멤버가 아닌 함수를 가리킬 수 없는 이유를 이해해야 합니다. 이는 비멤버 함수의 유형이 멤버 함수의 유형과 다르기 때문입니다. 멤버 함수는 클래스의 멤버 변수 및 함수에 액세스하려면 이 포인터를 암시적으로 전달해야 합니다. 따라서 멤버 함수 포인터는 멤버 함수의 주소와 클래스 포인터를 저장하는 반면, 비멤버 함수는 이 포인터가 없어 클래스의 멤버 변수와 함수에 접근할 수 없기 때문에 멤버 함수 포인터가 포인터를 가리킬 수 없다. 비회원 기능.

몇 가지 트릭을 사용하면 이 문제를 해결할 수 있습니다. 다음은 몇 가지 해결 방법입니다.

  1. 펑터 또는 람다 표현식 사용

펑터는 함수 호출 연산자를 구현하는 클래스입니다. 클래스의 객체를 함수처럼 호출할 수 있도록 함수 호출 연산자를 오버로드할 수 있습니다. 따라서 멤버 함수 포인터가 functor 객체를 가리킬 수 있도록 functor를 사용하여 멤버 함수의 기능을 구현할 수 있습니다. 람다 표현식도 동일한 기능을 달성할 수 있습니다.

다음은 functor 사용 예입니다.

#include <iostream>

class MyClass {
public:
    void foo() {
        std::cout << "Hello from foo!";
    }
};

class FunctionObject {
public:
    void operator()() {
        obj.foo();
    }
    MyClass obj;
};

int main() {
    FunctionObject f;
    f();
    return 0;
}

위 코드에서는 MyClass 객체와 Operator() 함수가 포함된 FunctionObject 클래스를 정의합니다. functor 객체가 호출되면 연산자라고 합니다. () 함수를 사용하여 MyClass의 foo 함수를 호출합니다.

FunctionObject 객체의 주소를 멤버 함수 포인터에 전달하여 foo 함수를 호출할 수 있습니다. 예:

void (MyClass::*func_ptr)() = &MyClass::foo;
FunctionObject f;
MyClass* p = &(f.obj);
(p->*func_ptr)();

위 코드에서는 MyClass의 foo 함수를 가리키는 멤버 함수 포인터 func_ptr을 정의합니다. 그런 다음 FunctionObject 객체 f를 생성하고 해당 객체의 obj 멤버 주소를 MyClass 포인터 p에 할당했습니다. 마지막으로 멤버 액세스 연산자(p->*)와 멤버 함수 포인터 func_ptr을 사용하여 foo 함수를 호출합니다.

  1. 전역 함수와 void* 포인터 사용

또 다른 해결책은 전역 함수와 void* 포인터를 사용하는 것입니다. void 포인터를 매개변수로 받아들이고 이를 클래스에 대한 포인터로 변환하고 멤버 함수를 호출하는 전역 함수를 정의할 수 있습니다. 그런 다음 이 전역 함수의 주소를 멤버 함수 포인터에 할당하여 클래스의 멤버 함수를 호출할 수 있습니다.

다음은 전역 함수와 void* 포인터를 사용하는 예입니다.

#include <iostream>

class MyClass {
public:
    void foo() {
        std::cout << "Hello from foo!";
    }
};

void invoke_function(void* obj_ptr, void (*func_ptr)()) {
    MyClass* p = static_cast<MyClass*>(obj_ptr);
    (p->*func_ptr)();
}

int main() {
    void (*func_ptr)() = &MyClass::foo;
    MyClass obj;
    invoke_function(&obj, func_ptr);
    return 0;
}

위 코드에서는 void

포인터와 함수 포인터를 매개 변수로 받아들이는 전역 함수 호출_함수를 정의합니다. Invoke_function 함수에서 void 포인터 obj_ptr을 MyClass 포인터 p로 변환하고 멤버 액세스 연산자(p->*)와 함수 포인터 func_ptr을 사용하여 foo 함수를 호출합니다. 메인 함수에서는 MyClass의 foo 함수를 가리키는 멤버 함수 포인터 func_ptr을 정의합니다. 그런 다음 MyClass 객체 obj를 생성하고 해당 주소를 Invoke_function 함수에 전달한 다음 func_ptr을 Invoke_function 함수에 전달합니다.

요약

멤버 함수 포인터는 런타임에 호출할 멤버 함수를 동적으로 선택할 수 있는 강력한 도구입니다. 그러나 멤버 함수 포인터를 사용할 때 멤버가 아닌 함수를 가리켜야 하는 경우 functor 또는 Lambda 표현식을 사용하거나 전역 함수 및 void* 포인터를 사용할 수 있습니다. 이러한 기술은 멤버 함수 포인터가 멤버가 아닌 함수를 가리킬 수 없는 문제를 해결하는 데 도움이 될 수 있습니다.

위 내용은 C++ 구문 오류: 멤버 함수 포인터가 비멤버 함수를 가리킬 수 없습니다. 어떻게 처리합니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.