Welcome to OStack Knowledge Sharing Community for programmer and developer-Open, Learning and Share
Welcome To Ask or Share your Answers For Others

Categories

0 votes
347 views
in Technique[技术] by (71.8m points)

c++链接错误,始终想不通

------------------- 文件 1.h -----------------------------------

#pragma once
class A
{
public:
    static int x;

public:
    void func();
};

int A::x = 1;

-------------------- 文件 1.cpp -----------------------

#include "1.h"

void A::func()
{
    ;
}

-------------------- 文件 main.cpp -----------------------

#include<iostream>
#include"1.h"

int main()
{
    A a;

    return 0;
}

一起运行报错:
图片描述

int A::x = 1;是应该放在cpp里,但是放在.h里的话,#pragma once防止头文件被二次编译,怎么还会报错呢,我的理解是如果去掉#pragma once它才会报这样的错,但是事实却是这样,我不知道这是为什么?或者我对#pragma once的理解有误?(PS:不要讨论#ifndef...#define...#endif#pragma once的区别,重点不是这个)

我的想法是二次编译(网上博文看来的)这个说法是不是错的,#pragma once到底是为了防止头文件被二次啥啥呢?


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome To Ask or Share your Answers For Others

1 Answer

0 votes
by (71.8m points)

是因为你的int A::x分别在1.cpp和main.cpp被定义了两次,链接器不知道哪个才是正确的。

你可以把int A::x = 1; 放到1.cpp中去。

pragma once的作用并不是防止多次编译,而是多次包含,不管你写不写pragma once,每个包含1.h的cpp文件都会拷贝一份1.h到它的编译单元中进行编译,pragma once的作用是在多次(直接或间接)包含同一个头文件时不重复包含。

a.h

#pragma once
int a = 0;

main.cpp

#include<iostream>
#include "a.h"
#include "a.h"
#include "a.h"
#include "a.h"
#include "a.h"
#include "a.h"
using namespace std;

int main()
{
    cout << a << endl;
    return 0;
}

main.cpp可以正常编译而不发生int a 的重定义,因为a.h只被包含了一次。


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome to OStack Knowledge Sharing Community for programmer and developer-Open, Learning and Share
Click Here to Ask a Question

...