> 백엔드 개발 > C++ > Null 포인터에 대한 비가상 메서드 호출이 때때로 C의 충돌을 피하는 이유는 무엇입니까?

Null 포인터에 대한 비가상 메서드 호출이 때때로 C의 충돌을 피하는 이유는 무엇입니까?

DDD
풀어 주다: 2024-12-14 09:29:12
원래의
837명이 탐색했습니다.

Why Do Non-Virtual Method Calls on Null Pointers Sometimes Avoid Crashes in C  ?

가상이 아닌 메서드 호출과 Null 포인터: 역설

널 포인터를 통해 클래스 멤버에 액세스하면 일반적으로 충돌이 발생합니다. 그러나 C에서는 특정 비가상 메서드가 널 포인터에서도 작동하는 것처럼 보입니다. 이 동작은 몇 가지 질문을 제기합니다. 어떻게 발생하며 객체는 어디에 할당됩니까?

비가상 메서드 호출 이해

C에서 null에 대해 비가상 메서드가 호출되는 경우 포인터를 사용하면 컴파일러는 해당 메서드와 관련된 함수에 대한 직접 호출을 생성합니다. 숨겨진 매개변수(객체에 대한 포인터)를 함수에 전달하여 이를 수행합니다.

제공된 예에서:

class Foo {
  void say_hi();
};

Foo* foo = nullptr;
foo->say_hi();
로그인 후 복사

컴파일러는 이를 다음과 같이 변환합니다.

void Foo_say_hi(Foo* this);

Foo_say_hi(foo);
로그인 후 복사

say_hi 메소드는 객체의 멤버를 참조하지 않으므로 null 포인터를 역참조하지 않으므로 오류를 방지합니다. error.

정의되지 않은 동작 및 컴파일러 최적화

공식적으로 null 포인터에서 메서드를 호출하는 것은 정의되지 않은 동작입니다. 그러나 컴파일러는 개체가 null이 아니라고 가정하여 코드를 최적화할 수 있습니다. 이는 예기치 않은 동작으로 이어질 수 있으므로 위험합니다.

예제의 경우 컴파일러는 충돌을 피하기 위해 비가상 메서드 호출을 최적화합니다. 그러나 이 동작이 보장되지는 않는다는 점에 유의하는 것이 중요합니다. 널 포인터에서 가상이 아닌 메소드를 호출하는 것은 지정되지 않은 결과를 초래할 수 있으므로 피해야 합니다.

객체 할당

예제에서 foo 포인터가 참조하는 객체는 주요 기능. Foo* 유형의 지역 변수가 스택에 생성되고 여기에 할당된 값은 널 포인터입니다. 이는 객체 자체가 메모리 어디에도 존재하지 않는다는 뜻입니다.

위 내용은 Null 포인터에 대한 비가상 메서드 호출이 때때로 C의 충돌을 피하는 이유는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿