区分c++中的声明与定义

 更新时间:2020年08月30日 16:46:45   作者:Dabelv  
这篇文章主要介绍了如何区分c++中的声明与定义,帮助大家更好的理解和学习c++,感兴趣的朋友可以了解下

C++编码过程中,我们经常谈及“定义”和“声明”,二者是编程过程中的基本概念。我们需要使用一个变量、类型(类、结构体、枚举、共用体)或者函数时,我们需要提前定义和声明。定义和声明的过程,就像我们向图书馆借阅书籍一般,需要先完成书籍的印刷,即创造出书籍,这是一个定义的过程,有了书籍,我们需要到图书馆完成借阅的登记手续,这是申明的过程。完成了申明,我们有了使用书籍的权限,就可以尽情的畅游在知识的海洋。如果说书籍是自己委托印刷厂印刷的,那么你无需向他人借阅,即无需声明,可以直接使用书籍。一本书籍只需要印刷一次,但是可以被多人多次借阅,也就是说定义只需要一次,但是声明可以有多次。这里的书籍指代的是“定义”和“声明”作用的对象,即变量、类型和函数。C/C++中,使用一个变量、类型或者函数必须先在使用前完成定义和声明。

定义和声明是容易混淆的概念,但通过上面的类比说明,可以看出:
(1)“定义”和“声明”的本质区别是声明可以出现多次,而定义只能出现一次;
(2)把声明的东西放在头文件,把定义的东西放在源文件(.c或者.cpp文件);
(3)类型的定义应该放在头文件,因为类型不具有外部连接性,不同源文件拥有相同的类型定义不会报编译错误,但头文件不能重复包含。

1.变量的定义与声明

定义变量,指明变量所属类型、名称、分配内存空间与初始化其初始值,如

int a=1;
int a(1);

如果不显示初始化,则按照编译器默认进行初始化。

声明变量,指明变量所属类型与变量名称,如:

extern int a;

有一点需要注意,对于局部变量以及全局静态变量是不能通过extern进行前置申明的,即不能在定义之前通过声明来引用,因为局部变量的作用域是当前代码块,全局静态变量的作用域是当前源文件,都不是全局作用域,所以不能通过extern进行前置声明。全局变量允许在定义之前通过前置声明进行引用。参见如下代码片段。

#include <stdio.h>

extern int a;
extern static int b; //报错
int main()
{
 extern int c; //报错
 printf("a=%d,b=%d,c=%d",a,b,c);
 int c=2;
}
int a=0;
static int b=1;

2.类型的定义与声明

定义类型,指明类型的名称和内容。

struct test{int a;}

或者给已经存在的类型起个别名。

typedef int int32;

注意,类型的作用域是源文件,即类型不具有外部连接性质,因此可以在不同源文件定义相同名称的类型,比如定义同名的类是不会报重定义错误的,这也说明了类型的定义应该放在头文件中,但在同一个源文件定义相同名称的类型则编译器会报重定义错误。

声明类型,只给出类型的名称。

class A;

类型被声明之后,可以用于声明其他的标识符,但不能利用它来定义对象,也不能使用类型的成员。

class A;  //先声明类
void show(A& a); //申明函数
class A  //后定义
{
public:
 int a;
 char b;
};

以上代码没有错误。但如下代码将报错:

class A; //先声明
int main(int argc,char* argv[])
{
 A classA;
 classA.a=5;
 return 0;
}

class A //后定义
{
public:
 int a;
 char b;
};

该程序无法通过编译,原因是类A的所有成员的有效范围均是从定义类的地方开始,故使用未知的构造函数初始化其成员变量是错误的,会报使用未定义的 class “A”错误。

使用类型之前给出其定义式即可,即给出类型的名称和内容,或者利用typedef给一个类型起个别称。虽然说是定义类型,但还是将类型的定义放在头文件,即使被不同的源文件包含,因为类型不具有外部连接特性,所以不会报重定义错误的,这和变量和函数不同,不能将变量和函数的定义放在头文件!切记!这里所说的外部连接与内部连接的区别在于链接器接的时候是否将当前目标文件中的定义与其它目标文件对比,并报告是否有重定义错误。内连接不会进行对比,故不同源文件中定义相同名称的类型是不会报错的。

3.函数的定义与声明

定义函数,指明函数返回类型、函数名称、函数参数和函数体。

int test(char a,int b)
{
 return a+b
}

声明函数,指明函数返回类型,函数名称和函数参数。

int test(char a,int b);
//或者无需给出形参名称,只需要类型即可
int test(char,int);

从上面可以看出,函数定义与函数声明的区别是主要有两点:
(1)函数定义需要给出函数体,即函数的具体实现,函数声明不需要;
(2)函数定义必须给出形参名称,声明可以只给出形参类型。

以上就是区分c++中的声明与定义的详细内容,更多关于c++ 声明与定义的资料请关注脚本之家其它相关文章!

相关文章

  • C字符串操作函数的实现详细解析

    C字符串操作函数的实现详细解析

    以下是对C语言中字符串操作函数的实现进行了详细的分析介绍,需要的朋友可以过来参考下
    2013-08-08
  • C++ qt 使用jsoncpp json 读写操作

    C++ qt 使用jsoncpp json 读写操作

    JsonCpp是一个基于C++语言的开源库,用于C++程序的Json数据的读写操作,本文重点给大家介绍C++ qt 使用jsoncpp json 读写操作,感兴趣的朋友跟随小编一起看看吧
    2021-11-11
  • Qt读写CSV文件的三种方式及优劣对比

    Qt读写CSV文件的三种方式及优劣对比

    最近的要用到CSV格式的数据,所以这篇文章讲述一下QT读取CSV文件数据,下面这篇文章主要给大家介绍了关于Qt读写CSV文件的三种方式及优劣对比的相关资料,需要的朋友可以参考下
    2023-11-11
  • C++ 多线程之互斥量(mutex)详解

    C++ 多线程之互斥量(mutex)详解

    这篇文章主要为大家详细介绍了C++多线程之互斥量(mutex),文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-02-02
  • C++ STL中五个常用算法使用教程及实例讲解

    C++ STL中五个常用算法使用教程及实例讲解

    本文主要介绍了C++ STL算法中常见的五个算法的使用教程并附上了案例详解,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-11-11
  • C语言中的setlinebuf()、utmpname()、rewind函数使用

    C语言中的setlinebuf()、utmpname()、rewind函数使用

    这篇文章主要介绍了C语言中的setlinebuf()、utmpname()、rewind函数使用,是C语言中操作文件的一些基本函数,需要的朋友可以参考下
    2015-08-08
  • c++中八大排序算法

    c++中八大排序算法

    本篇文章主要介绍了八大排序算法,详细的介绍了八个算法思想,实现代码,稳定性,时间复杂度等,具有一定的参考价值,有需要的可以了解一下。
    2016-11-11
  • 有关C++头文件的包含顺序研究

    有关C++头文件的包含顺序研究

    下面小编就为大家带来一篇有关C++头文件的包含顺序研究。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • C++深入讲解类与对象之OOP面向对象编程与封装

    C++深入讲解类与对象之OOP面向对象编程与封装

    学习过C语言的小伙伴知道:C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题,接下来让我们详细的了解
    2022-05-05
  • C++超详细讲解函数参数的默认值

    C++超详细讲解函数参数的默认值

    在C++中,定义函数时可以给形参指定一个默认的值,这样调用函数时如果没有给这个形参赋值(没有对应的实参),那么就使用这个默认的值。也就是说,调用函数时可以省略有默认值的参数
    2022-05-05

最新评论