> 백엔드 개발 > C++ > 본문

C에서 전역 변수 선언을 교환하면 예기치 않은 동작이 발생하는 이유는 무엇입니까?

Susan Sarandon
풀어 주다: 2024-10-30 06:38:03
원래의
791명이 탐색했습니다.

 Why Does Swapping Global Variable Declarations in C   Lead to Unexpected Behavior?

C의 전역 초기화 순서 : 종속성 무시

C에서는 번역 단위 내 전역 변수의 초기화 순서가 잘 정의되어 있습니다. 그러나 전역 변수 간의 종속성을 무시하면 놀라운 동작이 발생할 수 있습니다.

다음 코드를 고려하세요.

<code class="cpp">struct Foo;

extern Foo globalFoo;

struct Foo {
    Foo() { printf("Foo::Foo()\n"); }
    void add() { printf("Foo::add()\n"); }
    static int addToGlobal() {
        printf("Foo::addToGlobal() START\n");
        globalFoo.add();
        printf("Foo::addToGlobal() END\n");
        return 0;
    }
};

Foo globalFoo;
int dummy = Foo::addToGlobal();

int main() {
    printf("main()\n");
    return 0;
}</code>
로그인 후 복사

GCC 4.4.3으로 컴파일할 때 예상되는 출력은 다음과 같습니다.

Foo::Foo()
Foo::addToGlobal() START
Foo::add()
Foo::addToGlobal() END
main()
로그인 후 복사

이는 정적 메서드 Foo::addToGlobal()을 호출하기 전에 전역 변수 globalFoo가 초기화되기 때문입니다. 그러나 globalFoo와 더미 선언의 순서를 바꾸면 출력은 다음과 같습니다.

Foo::addToGlobal() START
Foo::add()
Foo::addToGlobal() END
Foo::Foo()
main()
로그인 후 복사

Foo의 인스턴스 메서드가 구성되지 않은 인스턴스에서 호출되는 것으로 보입니다. 이는 전역 초기화 순서가 종속성을 무시하기 때문입니다.

dummy를 초기화하기 전에 Foo의 생성자가 호출되도록 하려면 동일한 번역 단위에서 dummy보다 먼저 globalFoo가 정의되어 있는지 확인해야 합니다. 또는 동적 초기화 전에 null로 초기화되는 전역 인스턴스에 대한 정적 포인터를 사용할 수 있습니다. 그런 다음 addToGlobal 메소드는 포인터가 null인지 확인하고 필요한 경우 전역 Foo를 생성할 수 있습니다.

위 내용은 C에서 전역 변수 선언을 교환하면 예기치 않은 동작이 발생하는 이유는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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