백엔드 개발 C#.Net 튜토리얼 배워야 한다! C++에서 다형성 메커니즘을 구현하기 위한 기본 조건

배워야 한다! C++에서 다형성 메커니즘을 구현하기 위한 기본 조건

Jul 24, 2018 pm 02:11 PM

C++의 다형성 메커니즘을 구현하는 방법 실행 다형성을 구현하는 메커니즘은 기본 클래스의 함수 앞에 virtual 키워드를 추가하고 파생 클래스에서 함수를 다시 작성하는 것입니다. 런타임 기능의 개체입니다.

C++ 다형성의 구현 및 원리

C++의 다형성은 한 문장으로 요약할 수 있습니다. 기본 클래스의 함수 앞에 virtual 키워드를 추가하고, 파생 클래스에서 함수를 다시 작성하면 런타임이 기반 클래스가 됩니다. on 해당 함수를 호출할 객체의 실제 유형입니다. 객체 유형이 파생 클래스인 경우 파생 클래스의 함수를 호출하고, 객체 유형이 기본 클래스인 경우 기본 클래스의 함수를 호출합니다.

1: virtual 키워드로 선언된 함수를 가상 함수라고 합니다. , 가상 함수는 클래스 함수의 멤버여야 합니다.

2: 가상 함수가 있는 클래스에는 가상 테이블이라는 1차원 가상 함수 테이블이 있습니다. 클래스의 개체에는 가상 테이블의 시작 부분을 가리키는 가상 포인터가 있습니다. 가상 테이블은 클래스에 해당하고 가상 테이블 포인터는 개체에 해당합니다.

3: 다형성은 인터페이스의 다중 구현이며 객체 지향의 핵심입니다. 클래스 다형성과 함수 다형성으로 구분됩니다.

4: 다형성은 동적 바인딩과 결합된 가상 함수로 구현됩니다.

5: 순수 가상 함수는 가상 함수에 = 0을 더한 것입니다.

6: 추상 클래스는 하나 이상의 순수 가상을 포함하는 클래스를 나타냅니다. 기능.

순수 가상 함수: virtual void fun()=0, 즉 추상 클래스입니다! 이 함수는 하위 클래스에서 구현되어야 합니다. 즉, 이름이 먼저 있고 내용이 없으며 내용이 파생 클래스에서 구현됩니다.

먼저 예제를 살펴보겠습니다

#include <iostream> #include <stdlib.h>using namespace std; 

class Father
{public:    void Face()
    {        cout << "Father&#39;s face" << endl;
    }    void Say()
    {        cout << "Father say hello" << endl;
    }
};class Son:public Father
{public:     
    void Say()
    {        cout << "Son say hello" << endl;
    }
};int main()
{
    Son son;
    Father *pFather=&son; // 隐式类型转换
    pFather->Say();    return 0;
}
로그인 후 복사

출력 결과는 다음과 같습니다.

Father say hello
로그인 후 복사

main() 함수에서 먼저 Son 클래스의 객체 son을 정의한 다음 Father 클래스를 가리키는 포인터 변수 pFather를 정의합니다. 그런 다음 이것을 사용하십시오. 변수는 pFather->Say()를 호출합니다. 많은 사람들이 이 상황을 C++의 다형성과 혼동하는 경향이 있는 것으로 추정됩니다. son은 실제로 Son 클래스의 객체이므로 Son의 Say를 호출해야 합니다. 클래스를 생성하고 "Son say hello"”를 출력하지만 결과는 그렇지 않습니다.

 컴파일 관점에서:

 컴파일할 때 C++ 컴파일러는 각 클래스가 호출하는 함수(비가상 함수)의 주소를 결정해야 합니다. Son 클래스 객체 son의 주소를 pFather에 할당하면 C++ 컴파일러는 이때 변수 pFather가 Father 객체의 주소를 갖고 있다고 생각합니다. 메인 함수에서 pFather->Say(가 실행될 때) 호출되는 것은 당연히 Father 객체의 Say 함수입니다

메모리 관점에서 보면

배워야 한다! C++에서 다형성 메커니즘을 구현하기 위한 기본 조건

Son 클래스 객체의 메모리 모델은 다음과 같습니다. 위

Son 클래스의 객체를 생성할 때 먼저 Father 클래스의 생성자를 호출하여 생성해야 합니다. Father 클래스의 객체를 호출한 다음 Son 클래스의 생성자를 호출하여 생성의 자체 부분을 완료합니다. , 이로써 완전한 Son 클래스 객체를 분리합니다. Son 클래스 객체를 Father 타입으로 변환할 때, 그 객체는 위 그림에서 "Father 객체가 점유하고 있는 메모리"인 원래 객체의 전체 메모리 모델 중 상위 절반으로 간주됩니다. 유형 변환 사용 객체 포인터가 해당 메소드를 호출할 때 당연히 해당 메소드가 위치한 메모리에서 메소드를 호출하므로 "Father Say hello"를 출력하는 것이 논리적입니다.

많은 사람들이 생각하는 것처럼 위 코드에서 pFather는 실제로 Son 클래스의 객체를 가리킨다는 것을 알고 있습니다. 따라서 이 결과를 얻으려면 다음을 사용해야 합니다. 가상 기능.

 이전 출력 결과는 컴파일 시 객체가 호출한 함수의 주소를 컴파일러가 이미 결정했기 때문입니다. 이 문제를 해결하려면 런타임 바인딩을 사용해야 합니다. 컴파일러가 런타임 바인딩을 사용할 때 실행되는 함수의 유형을 결정합니다. 컴파일러가 런타임에 바인딩을 사용하도록 하려면 기본 클래스에서 함수를 선언할 때 virtual 키워드를 사용해야 합니다. 기본 클래스에서 virtual로 설정되면 함수는 모든 파생 클래스에서 virtual이 되며 명시적으로 virtual로 선언할 필요가 없습니다.

 코드를 약간 변경하여 실행 결과를 살펴보세요

#include <iostream> #include <stdlib.h>using namespace std; 

class Father
{public:    void Face()
    {        cout << "Father&#39;s face" << endl;
    }    virtual void Say()
    {        cout << "Father say hello" << endl;
    }
};class Son:public Father
{public:     
    void Say()
    {        cout << "Son say hello" << endl;
    }
};int main()
{
    Son son;
    Father *pFather=&son; // 隐式类型转换
    pFather->Say();    return 0;
}
로그인 후 복사

실행 결과:

Son say hello
로그인 후 복사

결과가 "Son say hello"인 것으로 확인되었습니다. 즉, 개체 유형에 따라 올바른 함수가 호출됩니다. , 가상일 때 뒤에서 무슨 일이 일어나는지 Say()를 선언하면 됩니다.

컴파일러는 컴파일 중에 Father 클래스에 가상 함수가 있음을 발견합니다. 이때 컴파일러는 가상 함수를 포함하는 각 클래스에 대해 가상 테이블(즉, vtable)을 생성합니다. 이 배열에는 각 가상 함수의 주소가

배워야 한다! C++에서 다형성 메커니즘을 구현하기 위한 기본 조건

에 저장됩니다.

那么如何定位虚表呢?编译器另外还为每个对象提供了一个虚表指针(即vptr),这个指针指向了对象所属类的虚表,在程序运行时,根据对象的类型去初始化vptr,从而让vptr正确的指向了所属类的虚表,从而在调用虚函数的时候,能够找到正确的函数,对于第二段代码程序,由于pFather实际指向的对象类型是Son,因此vptr指向的Son类的vtable,当调用pFather->Son()时,根据虚表中的函数地址找到的就是Son类的Say()函数.

  正是由于每个对象调用的虚函数都是通过虚表指针来索引的,也就决定了虚表指针的正确初始化是非常重要的,换句话说,在虚表指针没有正确初始化之前,我们不能够去调用虚函数,那么虚表指针是在什么时候,或者什么地方初始化呢?

  答案是在构造函数中进行虚表的创建和虚表指针的初始化,在构造子类对象时,要先调用父类的构造函数,此时编译器只“看到了”父类,并不知道后面是否还有继承者,它初始化父类对象的虚表指针,该虚表指针指向父类的虚表,当执行子类的构造函数时,子类对象的虚表指针被初始化,指向自身的虚表。

  总结(基类有虚函数的):

  1:每一个类都有虚表

  2:虚表可以继承,如果子类没有重写虚函数,那么子类虚表中仍然会有该函数的地址,只不过这个地址指向的是基类的虚函数实现,如果基类有3个虚函数,那么基类的虚表中就有三项(虚函数地址),派生类也会虚表,至少有三项,如果重写了相应的虚函数,那么虚表中的地址就会改变,指向自身的虚函数实现,如果派生类有自己的虚函数,那么虚表中就会添加该项。

  3:派生类的虚表中虚地址的排列顺序和基类的虚表中虚函数地址排列顺序相同。

  这就是c++中的多态性,当c++编译器在编译的时候,发现Father类的Say()函数是虚函数,这个时候c++就会采用晚绑定技术,也就是编译时并不确定具体调用的函数,而是在运行时,依据对象的类型来确认调用的是哪一个函数,这种能力就叫做c++的多态性,我们没有在Say()函数前加virtual关键字时,c++编译器就确定了哪个函数被调用,这叫做早期绑定。

  c++的多态性就是通过晚绑定技术来实现的。

  c++的多态性用一句话概括就是:在基类的函数前加上virtual关键字,在派生类中重写该函数,运行时将会根据对象的实际类型来调用相应的函数,如果对象类型是派生类,就调用派生类的函数,如果对象类型是基类,就调用基类的函数。

  虚函数是在基类中定义的,目的是不确定它的派生类的具体行为,例如:

  定义一个基类:class Animal //动物,它的函数为breathe()

  再定义一个类class Fish //鱼。它的函数也为breathe()

  再定义一个类class Sheep //羊,它的函数也为breathe()

将Fish,Sheep定义成Animal的派生类,然而Fish与Sheep的breathe不一样,一个是在水中通过水来呼吸,一个是直接呼吸,所以基类不能确定该如何定义breathe,所以在基类中只定义了一个virtual breathe,它是一个空的虚函数,具体的函数在子类中分别定义,程序一般运行时,找到类,如果它有基类,再找到它的基类,最后运行的是基类中的函数,这时,它在基类中找到的是virtual标识的函数,它就会再回到子类中找同名函数,派生类也叫子类,基类也叫父类,这就是虚函数的产生,和类的多态性的体现。

  这里的多态性是指类的多态性。

  函数的多态性是指一个函数被定义成多个不同参数的函数。当你调用这个函数时,就会调用不同的同名函数。

一般情况下(不涉及虚函数),当我们用一个指针/引用调用一个函数的时候,被调用的函数是取决于这个指针/引用的类型。

当设计到多态性的时候,采用了虚函数和动态绑定,此时的调用就不会在编译时候确定而是在运行时确定。不在单独考虑指针/引用的类型而是看指针/引用的对象的类型来判断函数的调用,根据对象中虚指针指向的虚表中的函数的地址来确定调用哪个函数

现在我们看一个体现c++多态性的例子,看看输出结果:

#include <iostream> #include <stdlib.h>using namespace std; 

class CA 
{ 
public: 
    void f() 
    { 
        cout << "CA f()" << endl; 
    } 
    virtual void ff() 
    { 
        cout << "CA ff()" << endl; 
        f(); 
    } 
}; 

class CB : public CA 
{ 
public : 
    virtual void f() 
    { 
        cout << "CB f()" << endl; 
    } 
    void ff() 
    { 
        cout << "CB ff()" << endl; 
        f(); 
        CA::ff(); 
    } 
}; 
class CC : public CB 
{ 
public: 
    virtual void f() 
    { 
        cout << "C f()" << endl; 
    } 
}; 

int main() 
{ 
    CB b; 
    CA *ap = &b; 
    CC c; 
    CB &br = c; 
    CB *bp = &c; 

    ap->f(); 
    cout << endl;

    b.f(); 
    cout << endl;

    br.f(); 
    cout << endl;

    bp->f(); 
    cout << endl;

    ap->ff(); 
    cout << endl;

    bp->ff(); 
    cout << endl;    return 0; 
}
로그인 후 복사

输出结果:

CA f()CB f()C f()C f()CB ff()CB f()CA ff()CA f()CB ff()C f()CA ff()CA f()
로그인 후 복사

 相关推荐:

【c#教程】C# 多态性       

C# 多态性

C++ 비디오 튜토리얼_무료 C++ 튜토리얼 온라인 학습

위 내용은 배워야 한다! C++에서 다형성 메커니즘을 구현하기 위한 기본 조건의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

C 언어로 다양한 기호를 사용하는 방법 C 언어로 다양한 기호를 사용하는 방법 Apr 03, 2025 pm 04:48 PM

C 언어 커버 산술, 할당, 조건, 논리, 비트 연산자 등의 기호의 사용 방법은 기본 수학 연산에 사용되며, 할당 연산자는 할당 및 추가, 곱하기, 분할 할당에 사용되며, 곱하기 및 분할 할당에 사용되며, 조건에 따라 조건 운영자가 사용되며, 비트 오퍼레이터에 사용되며, 스페셜 오퍼레이터는 비트 수준의 운영에 사용됩니다. 포인터, 파일 종료 마커 및 비수통 값.

C 현에서 숯의 역할은 무엇입니까? C 현에서 숯의 역할은 무엇입니까? Apr 03, 2025 pm 03:15 PM

C에서 숯 유형은 문자열에 사용됩니다. 1. 단일 문자를 저장하십시오. 2. 배열을 사용하여 문자열을 나타내고 널 터미네이터로 끝납니다. 3. 문자열 작동 함수를 통해 작동합니다. 4. 키보드에서 문자열을 읽거나 출력하십시오.

멀티 스레딩과 비동기 C#의 차이 멀티 스레딩과 비동기 C#의 차이 Apr 03, 2025 pm 02:57 PM

멀티 스레딩과 비동기식의 차이점은 멀티 스레딩이 동시에 여러 스레드를 실행하는 반면, 현재 스레드를 차단하지 않고 비동기식으로 작업을 수행한다는 것입니다. 멀티 스레딩은 컴퓨팅 집약적 인 작업에 사용되며 비동기식은 사용자 상호 작용에 사용됩니다. 멀티 스레딩의 장점은 컴퓨팅 성능을 향상시키는 것이지만 비동기의 장점은 UI 스레드를 차단하지 않는 것입니다. 멀티 스레딩 또는 비동기식을 선택하는 것은 작업의 특성에 따라 다릅니다. 계산 집약적 작업은 멀티 스레딩을 사용하고 외부 리소스와 상호 작용하고 UI 응답 성을 비동기식으로 유지 해야하는 작업을 사용합니다.

C 언어로 Char Array를 사용하는 방법 C 언어로 Char Array를 사용하는 방법 Apr 03, 2025 pm 03:24 PM

char 어레이는 문자 시퀀스를 C 언어로 저장하고 char array_name [size]로 선언됩니다. 액세스 요소는 첨자 연산자를 통해 전달되며 요소는 문자열의 끝점을 나타내는 널 터미네이터 '\ 0'으로 끝납니다. C 언어는 strlen (), strcpy (), strcat () 및 strcmp ()와 같은 다양한 문자열 조작 함수를 제공합니다.

C 언어로 특수 문자를 처리하는 방법 C 언어로 특수 문자를 처리하는 방법 Apr 03, 2025 pm 03:18 PM

C 언어에서 특수 문자는 다음과 같은 탈출 시퀀스를 통해 처리됩니다. \ n 라인 브레이크를 나타냅니다. \ t는 탭 문자를 의미합니다. char c = '\ n'과 같은 특수 문자를 나타 내기 위해 탈출 시퀀스 또는 문자 상수를 사용하십시오. 백 슬래시는 두 번 탈출해야합니다. 다른 플랫폼과 컴파일러마다 다른 탈출 시퀀스가있을 수 있습니다. 문서를 참조하십시오.

C 언어에서 Char와 WCHAR_T의 차이 C 언어에서 Char와 WCHAR_T의 차이 Apr 03, 2025 pm 03:09 PM

C 언어에서 char와 wchar_t의 주요 차이점은 문자 인코딩입니다. char ascii를 사용하거나 ascii를 확장하고, wchar_t는 유니 코드를 사용합니다. Char는 1-2 바이트를 차지하고 WCHAR_T는 2-4 바이트를 차지합니다. Char는 영어 텍스트에 적합하며 WCHAR_T는 다국어 텍스트에 적합합니다. Char_t는 널리 지원되며, 컴파일러 및 운영 체제가 유니 코드를 지원하는지 여부에 따라 다릅니다. Char는 문자 범위가 제한되며 WCHAR_T는 더 큰 문자 범위를 가지며 특수 함수는 산술 작업에 사용됩니다.

C 언어로 Char를 변환하는 방법 C 언어로 Char를 변환하는 방법 Apr 03, 2025 pm 03:21 PM

C 언어에서 숯 유형 변환은 다른 유형으로 직접 변환 할 수 있습니다. 캐스팅 : 캐스팅 캐릭터 사용. 자동 유형 변환 : 한 유형의 데이터가 다른 유형의 값을 수용 할 수 있으면 컴파일러가 자동으로 변환됩니다.

C 언어 합계의 기능은 무엇입니까? C 언어 합계의 기능은 무엇입니까? Apr 03, 2025 pm 02:21 PM

C 언어에는 내장 합계 기능이 없으므로 직접 작성해야합니다. 합계는 배열 및 축적 요소를 가로 질러 달성 할 수 있습니다. 루프 버전 : 루프 및 배열 길이를 사용하여 계산됩니다. 포인터 버전 : 포인터를 사용하여 배열 요소를 가리키며 효율적인 합계는 자체 증가 포인터를 통해 달성됩니다. 동적으로 배열 버전을 할당 : 배열을 동적으로 할당하고 메모리를 직접 관리하여 메모리 누출을 방지하기 위해 할당 된 메모리가 해제되도록합니다.

See all articles