c++复杂类型指针变量的声明
- 格式:doc
- 大小:51.50 KB
- 文档页数:7
C语⾔中变量的声明和定义变量声明和变量定义变量定义:⽤于为变量分配存储空间,还可为变量指定初始值。
程序中,变量有且仅有⼀个定义。
变量声明:⽤于向程序表明变量的类型和名字。
定义也是声明,extern声明不是定义定义也是声明:当定义变量时我们声明了它的类型和名字。
extern声明不是定义:通过使⽤extern关键字声明变量名⽽不定义它。
[注意]变量在使⽤前就要被定义或者声明。
在⼀个程序中,变量只能定义⼀次,却可以声明多次。
定义分配存储空间,⽽声明不会。
C++程序通常由许多⽂件组成,为了让多个⽂件访问相同的变量,C++区分了声明和定义。
变量的定义(definition)⽤于为变量分配存储空间,还可以为变量指定初始值。
在程序中,变量有且仅有⼀个定义。
声明(declaration)⽤于向程序表明变量的类型和名字。
定义也是声明:当定义变量的时候我们声明了它的类型和名字。
可以通过使⽤extern声明变量名⽽不定义它。
不定义变量的声明包括对象名、对象类型和对象类型前的关键字extern。
extern声明不是定义,也不分配存储空间。
事实上它只是说明变量定义在程序的其他地⽅。
程序中变量可以声明多次,但只能定义⼀次。
只有当声明也是定义时,声明才可以有初始化式,因为只有定义才分配存储空间。
初始化式必须要有存储空间来进⾏初始化。
如果声明有初始化式,那么它可被当作是定义,即使声明标记为extern。
任何在多⽂件中使⽤的变量都需要有与定义分离的声明。
在这种情况下,⼀个⽂件含有变量的定义,使⽤该变量的其他⽂件则包含该变量的声明(⽽不是定义)。
如何清晰的区分变量声明和定义extern通知编译器变量在其他地⽅被定义1.extern告诉编译器变量在其他地⽅定义了。
例如:extern int i;//声明,不是定义int i;//声明,也是定义,未初始化带有初始化式的声明必定式定义2.如果声明有初始化式,就被当作定义,即使前⾯加了extern。
c类的定义和声明(原创版)目录1.C 语言的基本数据类型2.C 语言的变量和常量3.C 语言的运算符4.C 语言的控制语句5.C 语言的函数6.C 语言的数组和字符串7.C 语言的文件操作正文C 语言是一种高级计算机编程语言,广泛应用于操作系统、嵌入式系统、游戏开发等领域。
C 语言的定义和声明是编写程序的基础,下面我们将详细介绍 C 语言的基本概念。
1.C 语言的基本数据类型包括整型、浮点型、字符型和空类型等。
整型通常表示整数,浮点型表示小数,字符型表示单个字符。
2.在 C 语言中,变量和常量是用来存储数据的。
变量是可变的,常量是不可变的。
在声明变量时,需要指定变量的数据类型,例如:int a; double b; char c; 常量可以用 const 关键字来声明,例如:const int MAX_VALUE = 100;3.C 语言提供了丰富的运算符,包括算术运算符、关系运算符、逻辑运算符、赋值运算符和位运算符等。
运算符的优先级和结合性也需要掌握,例如:a + b 和 a * b 的优先级不同,需要用小括号来明确运算顺序。
4.C 语言的控制语句包括条件语句(if-else)、循环语句(for、while、do-while)和跳转语句(break、continue、goto)。
掌握这些控制语句,可以编写出更加复杂的程序。
5.函数是 C 语言中重要的组织代码的方式,可以实现代码的模块化和重用。
函数的声明和调用需要使用函数原型,例如:int add(int a, intb); int main() { int x = add(10, 20); }6.数组和字符串是 C 语言中常用的数据结构。
数组是一段连续的内存空间,可以存储多个相同类型的数据。
字符串是一段字符数组,通常用字符串常量或字符数组来表示。
7.文件操作是 C 语言中重要的功能之一,可以用来读写数据。
C 语言提供了 fopen、fread、fwrite 等函数来实现文件操作。
C语言中指针变量作为函数参数详解C语言中指针变量作为函数参数详解在C语言中,函数的参数不仅可以是整数、小数、字符等具体的数据,还可以是指向它们的指针。
用指针变量作函数参数可以将函数外部的地址传递到函数内部,使得在函数内部可以操作函数外部的数据,并且这些数据不会随着函数的结束而被销毁。
像数组、字符串、动态分配的内存等都是一系列数据的集合,没有办法通过一个参数全部传入函数内部,只能传递它们的指针,在函数内部通过指针来影响这些数据集合。
有的时候,对于整数、小数、字符等基本类型数据的操作也必须要借助指针,一个典型的例子就是交换两个变量的值。
有些初学者可能会使用下面的方法来交换两个变量的值:#includevoid swap(int a, int b){ int temp; //临时变量 temp = a; a = b; b = temp;}int main(){ int a = 66, b = 99; swap(a, b); printf("a = %d, b = %dn", a, b); return 0;}运行结果:a = 66,b = 99从结果可以看出,a、b 的值并没有发生改变,交换失败。
这是因为 swap() 函数内部的 a、b 和 main() 函数内部的 a、b 是不同的变量,占用不同的内存,它们除了名字一样,没有其他任何关系,swap() 交换的是它内部 a、b 的值,不会影响它外部(main() 内部)a、b 的值。
改用指针变量作参数后就很容易解决上面的问题:#includevoid swap(int *p1, int *p2){ int temp; //临时变量temp = *p1; *p1 = *p2; *p2 = temp;}int main(){ int a = 66, b = 99; swap(&a, &b); printf("a = %d, b = %dn", a, b); return 0;} 运行结果:a = 99,b = 66调用 swap() 函数时,将变量 a、b 的地址分别赋值给 p1、p2,这样 *p1、*p2 代表的就是变量 a、b 本身,交换 *p1、*p2 的值也就是交换 a、b 的值。
关于C++的变量和类的声明和定义什么是变量?变量或者叫对象,是⼀个有具名的、可以供程序操作的存储空间。
这⾥具名是指变量是有名字的,可供操作是指能进⾏加减乘除或者输⼊输出等操作,存储空间则是指有⼀块属于它的内存空间。
为了便于说明,标题和后⾯的内容将对象分为两种。
对于内置的,⽂章称作为变量,对于⾃定义的,称之为对象。
第⼀个问题:在C++中,什么是定义,什么是声明?①定义:⽤于为变量分配存储空间,还可为变量指定初始值。
在程序中,变量有且仅有⼀个定义。
②声明:⽤于向程序表明变量的类型和名字。
在程序中,变量可以有多个声明。
定义也是声明:当定义变量时我们声明了它的类型和名字。
上⾯的这些内容很容易理解,很多情况下,定义就是声明,由于C++⽀持分离式编译,因此⼀个C++程序允许别分成许多块,由这些块共同组成完整的程序。
既然程序可以分成多块,那么如果要在所有块之间共⽤⼀个变量,那就要能够在这些块之间共享代码。
为了⽀持这种分离式编译机制,C++中就将声明和定义区分开来了。
第⼆个问题:定义也是声明,那么如果仅仅声明?回答这个问题之前,需要了解什么是作⽤域,这⾥假定你已知晓C++的作⽤域知识。
在需要声明⼀个变量之前,先想⼀想为什么需要声明变量?因为程序需要使⽤定义在别处(通常是别的⽂件中)的变量。
这⾥使⽤别处的变量暗含了⼀个意义,这个变量应当是个全局变量,因为当前作⽤域找不到,所以需要去别的地⽅找,⽽别的地⽅定义的变量应当全局变量,能够被我看到(这⾥只考虑全局变量,其他情况暂不考虑)。
既然变量已经在别的地⽅定义过了,⽽且变量有且仅有⼀个定义,那么我不能再次定义⽽只能声明了,如何声明?声明的⽅式是使⽤extern关键词,形如extern int i,这⾥有了关键词extern的修饰,因此是⼀个声明,⽽不是定义。
从这个声明中,得知了变量的类型和名字,但是没有分配内存。
假如给上⾯声明的变量⼀个值,那还是不是声明呢?回答是:不是。
《C语言基础教案》word版教案章节:一、C语言概述1. C语言的历史和发展2. C语言的特点和应用范围3. C语言的编译过程二、C语言基础语法1. 数据类型整型、浮点型、字符型变量和常量的声明和使用2. 运算符和表达式算术运算符关系运算符逻辑运算符赋值运算符条件运算符逗号运算符3. 控制语句条件语句循环语句跳转语句三、函数和数组1. 函数的定义和声明2. 函数的参数传递3. 函数的返回值4. 数组的声明和使用5. 字符串的操作四、指针和引用1. 指针的概念和声明2. 指针的运算3. 指针与数组4. 指针与函数5. 指针与动态内存分配五、结构体和文件操作1. 结构体的定义和声明2. 结构体的使用3. 文件的概念和打开4. 文件的读写操作5. 文件的关闭和错误处理六、顺序结构与分支结构1. 顺序结构的实现2. 分支结构的概念与实现3. 条件语句的嵌套4. 逻辑表达式与布尔类型七、循环结构1. 循环结构的概念与类型2. for循环的实现与应用3. while循环的实现与应用4. do-while循环的实现与应用5. 循环控制语句:break与continue八、数组与字符串1. 一维数组的声明、初始化与操作2. 二维数组的概念与操作3. 字符串的基本操作4. 字符串数组的应用5. 排序算法与数组的应用九、指针与内存管理1. 指针的基本概念与运算2. 指针与数组的关系3. 指针与函数的调用4. 指针与动态内存分配5. 内存管理:malloc、calloc、realloc与free十、结构体与联合体1. 结构体的概念与使用2. 结构体数组的操作3. 结构体指针的应用4. 联合体的概念与使用5. 枚举类型的声明与使用十一、函数的高级应用1. 递归函数的概念与实现2. 函数指针的应用3. 函数调用的方式与参数传递4. 全局变量与局部变量的作用域5. 静态局部变量的使用十二、指针与数组1. 指针与数组的关系2. 指针数组的概念与使用3. 数组指针的概念与使用4. 指针函数的概念与使用5. 函数指针数组的应用十三、文件操作1. 文件的概念与文件指针2. 文件的打开与关闭3. 文件的读写操作4. 文件的定位与操作5. 文件权限与错误处理十四、标准库函数1. 标准输入输出函数2. 字符串处理函数3. 数学函数4. 日期与时间函数5. 随机数函数十五、C语言编程实例1. 计算器程序的设计与实现2. 文本编辑器的基本功能实现3. 排序算法的实现与应用4. 树状数组与动态规划算法5. 简单的游戏设计与实现重点和难点解析本文档为《C语言基础教案》的全篇内容,主要涵盖了C语言的概述、基础语法、函数和数组、指针和引用、结构体和文件操作等基础知识,以及高级应用、函数的高级应用、文件操作、标准库函数和编程实例等进阶内容。
在C语言中,指针是一种特殊的数据类型,它存储了一个变量的内存地址。
指针的声明需要使用`*`符号。
例如,如果我们想要声明一个指向整数的指针,我们可以这样写:
```c
int *ptr;
```
这里,`ptr`是一个指向整数的指针。
注意,`*`符号表示这是一个指针,而不是指针所指向的数据类型。
你也可以声明指向其他类型的指针,比如指向字符的指针:
```c
char *str;
```
在这个例子中,`str`是一个指向字符的指针。
如果你想要声明一个指向指针的指针,你可以这样写:
```c
int **ptr_ptr;
```
这里,`ptr_ptr`是一个指向指向整数的指针的指针。
总的来说,C语言中指针的声明格式是:
```c
数据类型*指针名;
```
其中,`数据类型`是你想要指针指向的数据类型,`指针名`是你给指针起的名字。
C语言所有复杂的指针声明,都是由各种声明嵌套构成的。
如何解读复杂指针声明呢?右左法则是一个既著名又常用的方法。
不过,右左法则其实并不是C标准里面的内容,它是从C 标准的声明规定中归纳出来的方法。
C标准的声明规则,是用来解决如何创建声明的,而右左法则是用来解决如何辩识一个声明的,两者可以说是相反的。
右左法则的英文原文是这样说的:The right-left rule: Start reading the declaration from the innermost paren theses, go right, and then go left. When you encounter parentheses, the direction should be reversed. Once everything in the parentheses has b een parsed, jump out of it. Continue till the whole declaration has be en parsed.这段英文的翻译如下:右左法则:首先从最里面的圆括号看起,然后往右看,再往左看。
每当遇到圆括号时,就应该掉转阅读方向。
一旦解析完圆括号里面所有的东西,就跳出圆括号。
重复这个过程直到整个声明解析完毕。
笔者要对这个法则进行一个小小的修正,应该是从未定义的标识符开始阅读,而不是从括号读起,之所以是未定义的标识符,是因为一个声明里面可能有多个标识符,但未定义的标识符只会有一个。
现在通过一些例子来讨论右左法则的应用,先从最简单的开始,逐步加深:int (*func)(int *p);首先找到那个未定义的标识符,就是func,它的外面有一对圆括号,而且左边是一个*号,这说明func是一个指针,然后跳出这个圆括号,先看右边,也是一个圆括号,这说明(*func)是一个函数,而func是一个指向这类函数的指针,就是一个函数指针,这类函数具有int*类型的形参,返回值类型是int。
c语言指针的用法c语言是一种高级编程语言,它可以直接操作内存中的数据。
指针是c语言中一种特殊的变量,它可以存储另一个变量的地址,也就是内存中的位置。
通过指针,我们可以间接地访问或修改内存中的数据,从而实现更高效和灵活的编程。
本文将介绍c语言指针的基本概念、定义和初始化、运算和应用,以及一些常见的错误和注意事项。
希望本文能够帮助你掌握c语言指针的用法,提高你的编程水平。
指针的基本概念指针是一种数据类型,它可以存储一个地址值,也就是内存中某个位置的编号。
每个变量在内存中都有一个唯一的地址,我们可以用指针来记录这个地址,然后通过这个地址来访问或修改变量的值。
例如,假设有一个整型变量a,它的值为10,它在内存中的地址为1000(为了简化,我们假设地址是十进制数)。
我们可以定义一个指向整型的指针p,并把a的地址赋给p,如下所示:int a =10; // 定义一个整型变量a,赋值为10int*p; // 定义一个指向整型的指针pp =&a; // 把a的地址赋给p这里,&a表示取a的地址,也就是1000。
p = &a表示把1000赋给p,也就是让p指向a。
从图中可以看出,p和a是两个不同的变量,它们占用不同的内存空间。
p存储了a的地址,也就是1000。
我们可以通过p 来间接地访问或修改a的值。
指针的定义和初始化指针是一种数据类型,它需要在使用前进行定义和初始化。
定义指针时,需要指定它所指向的变量的类型。
初始化指针时,需要给它赋一个有效的地址值。
定义指针的一般格式为:type *pointer_name;其中,type表示指针所指向的变量的类型,如int、char、float等;pointer_name表示指针的名称,如p、q、ptr等;*表示这是一个指针类型。
例如:int*p; // 定义一个指向整型的指针pchar*q; // 定义一个指向字符型的指针qfloat*ptr; // 定义一个指向浮点型的指针ptr注意,在定义多个指针时,每个指针前都要加*号,不能省略。
C语言变量名命名规则一、程序风格:1、严格采用阶梯层次组织程序代码:各层次缩进的分格采用VC的缺省风格,即每层次缩进为4格,括号位于下一行。
要求相匹配的大括号在同一列,对继行则要求再缩进4格。
例如:2、提示信息字符串的位置在程序中需要给出的提示字符串,为了支持多种语言的开发,除了一些给调试用的临时信息外,其他所有的提示信息必须定义在资源中。
3、对变量的定义,尽量位于函数的开始位置。
二、命名规则:1、变量名的命名规则①、变量的命名规则要求用“匈牙利法则”。
即开头字母用变量的类型,其余部分用变量的英文意思或其英文意思的缩写,尽量避免用中文的拼音,要求单词的第一个字母应大写。
即:变量名=变量类型+变量的英文意思(或缩写)对非通用的变量,在定义时加入注释说明,变量定义尽量可能放在函数的开始处。
见下表:对未给出的变量类型要求提出并给出命名建议给技术委员会。
②、指针变量命名的基本原则为:对一重指针变量的基本原则为:“p”+变量类型前缀+命名如一个float*型应该表示为pfStat对多重指针变量的基本规则为:二重指针:“pp”+变量类型前缀+命名三重指针:“ppp”+变量类型前缀+命名......③、全局变量用g_开头,如一个全局的长型变量定义为g_lFailCount,即:变量名=g_+变量类型+变量的英文意思(或缩写)④、静态变量用s_开头,如一个静态的指针变量定义为s_plPerv_Inst,即:变量名=s_+变量类型+变量的英文意思(或缩写)⑤、成员变量用m_开头,如一个长型成员变量定义为m_lCount;即:变量名=m_+变量类型+变量的英文意思(或缩写)⑥、对枚举类型(enum)中的变量,要求用枚举变量或其缩写做前缀。
并且要求用大写。
如:enum cmEMDAYS{EMDAYS_MONDAY;EMDAYS_TUESDAY;……};⑦、对struct、union、class变量的命名要求定义的类型用大写。
c语言的各种声明
int f;
//声明一个整形变量
int *f;
//声明一个指针变量
int *f,g;
//声明一个指针变量f和一个整形变量g
int f();
//声明一个函数,返回整形数值
int (*f)();
//声明一个函数指针,f是指针变量,返回int型int *f();
//声明一个指针函数,f是函数,该函数返回int行指针
int *(*f)();
//声明一个函数的指针,也可以说是一个只想函数地址变量,f为指针变量,只想这个函数,返回int型指针
int f[];
//声明一个数组,
int *f[];
//声明一个数组指针,f为1个数组,数组里的数据存储的都是执行int型数据的指针。
int (*f[])();
//声明一个函数指针数组,f是一个数组,数组里存储的数据都是指向一个函数的指针,这个指针指向的函数返回int型数据
int *(*f[])();
//声明一个函数指针数组,f是一个数组,数组里存储的数据都是指向一个函数的指针,这个指向的函数返回int型指针数据(返回指向int数据的地址)struct A
{
char a1;
int a2;
};
//声明一个没有名字的结构体
struct A f;
//声明一个结构变量f
typedef B
{
int b1;
char b2;
}b;
//声明一个命为b的结构b f;
//声明一个结构变量f。
C语言中的结构体指针与typedef一、结构体指针的概念结构体是C语言中一种复合类型,它由若干数据成员组成,我们可以通过定义结构体变量的方式来使用结构体类型。
当我们需要在函数间传递结构体变量或在函数中动态创建结构体变量时,就需要使用结构体指针来进行操作。
二、结构体指针的定义和使用1. 定义结构体指针在C语言中,我们可以通过在结构体类型名称前加上"*"来定义结构体指针。
如果我们有一个名为Student的结构体类型,我们可以定义一个指向Student类型的指针变量ptr_stu如下所示:```struct Student {char name[20];int age;};struct Student *ptr_stu;```2. 结构体指针的初始化和使用我们可以通过使用取位置区域符""将结构体变量的位置区域赋给结构体指针,然后可以通过指针来访问结构体的成员变量。
假设我们有一个名为stu的结构体变量:```struct Student stu = {"John", 20};struct Student *ptr_stu = stu;printf("Name: s\n", ptr_stu->name);printf("Age: d\n", ptr_stu->age);```而在实际开发中,如果结构体类型名称较长或者需要频繁使用结构体指针,我们可以通过使用typedef来定义结构体指针类型,从而简化代码并提高可读性。
三、typedef关键字的介绍typedef是C语言中的关键字之一,它可以用来为一个已有的数据类型定义一个新的名称。
通过使用typedef,我们可以为复杂的数据类型定义更简洁、更易读的别名,从而提高代码的可维护性和可读性。
四、结构体指针的typedef用法1. 定义结构体指针类型我们可以通过使用typedef来定义结构体指针类型,从而可以直接使用新的类型名称来声明结构体指针变量。
1 数据类型关键字A.基本数据类型(5个)void :声明函数无返回值或无参数,声明无类型指针,显式丢弃运算结果char :字符型类型数据,属于整型数据的一种int :整型数据,通常为编译器指定的机器字长float :单精度浮点型数据,属于浮点数据的一种double :双精度浮点型数据,属于浮点数据的一种B .类型修饰关键字(4个)short :修饰int,短整型数据,可省略被修饰的int。
long :修饰int,长整形数据,可省略被修饰的int。
signed :修饰整型数据,有符号数据类型unsigned :修饰整型数据,无符号数据类型C .复杂类型关键字(5个)struct :结构体声明union :共用体声明enum :枚举声明typedef :声明类型别名sizeof :得到特定类型或特定类型变量的大小D .存储级别关键字(6个)auto :指定为自动变量,由编译器自动分配及释放。
通常在栈上分配static :指定为静态变量,分配在静态变量区,修饰函数时,指定函数作用域为文件内部register :指定为寄存器变量,建议编译器将变量存储到寄存器中使用,也可以修饰函数形参,建议编译器通过寄存器而不是堆栈传递参数extern :指定对应变量为外部变量,即标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义。
const:与volatile合称“cv特性”,指定变量不可被当前线程/进程改变(但有可能被系统或其他线程/进程改变)volatile:与const合称“cv特性”,指定变量的值有可能会被系统或其他进程/线程改变,强制编译器每次从内存中取得该变量的值2 流程控制关键字A .跳转结构(4个)return :用在函数体中,返回特定值(或者是void值,即不返回值)continue :结束当前循环,开始下一轮循环break :跳出当前循环或switch结构goto :无条件跳转语句B .分支结构(5个)if :条件语句,后面不需要放分号else :条件语句否定分支(与if连用)switch :开关语句(多重分支语句)case :开关语句中的分支标记default :开关语句中的“其他”分支,可选。
C++中的extern声明变量详解extern声明变量⽆外乎如下两种:1、声明全局变量2、声明函数今天我们只谈extern,什么const、static之类等等与之相关或不相关的⼀律忽略,下⾯就分别对以上两种情况⼀⼀讲解声明和定义既然提到extern声明变量,那我们就必须搞清楚声明和定义的区别。
这⾥我们将普通数据变量和函数统称变量。
从内存分配⾓度来说,声明和定义的区别在于声明⼀个变量不会分配内存,⽽定义⼀个变量会分配内存。
⼀个变量可以被声明多次,但是只能被定义⼀次。
基于以上前提,我们可以把声明和定义类⽐为指针和内存的关系。
我们知道,指针其实就是指向内存的⼀个符号,变量的定义就好⽐⼀块内存区域,⽽声明就好⽐它的指针,可以有多个指针指向同⼀个内存区域,⽽⼀个指针只能指向⼀个内存区域,这样就很好理解为什么变量只能被定义⼀次,如果被定义多次,那就会分配多个内存,这样你通过变量的声明到底去找哪块内存区域呢,这会是个问题。
对于数据来说,声明和定义往往是同时存在的,⽐如下⾯的⼀⾏语句复制代码代码如下:int data;这样既声明了data同时也定义了data,怎样做到只声明⽽不定义呢,⽤extern就可以了复制代码代码如下:extern int data;对于函数来说,声明和定义就很容易区分了,⼀般我们会将声明放在头⽂件⽽将定义放在源⽂件⾥复制代码代码如下:void hello();这是⼀个函数的声明,⽽复制代码代码如下:void hello(){printf("hello world!\n");}这是⼀个函数的定义。
当然,函数的声明和定义也可以同时发⽣,如果我们没有头⽂件⽽只有源⽂件,并且在源⽂件⾥并没有void hello();这样的语句,那么这个函数的声明和定义就同时发⽣了,此时如果我们在原⽂件⾥想要调⽤函数hello(),你调⽤的代码必须在函数定义之后。
其实上⾯的要点只在于⼀句话:使⽤变量之前必须声明,声明可以有多次,⽽定义只能有⼀次。
曾经碰到过让你迷惑不解、类似于int * (* (*fp1) (int) ) [10];这样的变量声明吗?本文将由易到难,一步一步教会你如何理解这种复杂的C/C++声明。
我们将从每天都能碰到的较简单的声明入手,然后逐步加入const修饰符和typedef,还有函数指针,最后介绍一个能够让你准确地理解任何C/C++声明的“右左法则”。
需要强调一下的是,复杂的C/C++声明并不是好的编程风格;我这里仅仅是教你如何去理解这些声明。
注意:为了保证能够在同一行上显示代码和相关注释,本文最好在至少1024x768分辨率的显示器上阅读。
让我们从一个非常简单的例子开始,如下:
这个应该被理解为“declare n as an int”(n是一个int型的变量)。
接下去来看一下指针变量,如下:
这个应该被理解为“declare p as an int *”(p是一个int *型的变量),或者说p是一个指向一个int型变量的指针。
我想在这里展开讨论一下:我觉得在声明一个指针(或引用)类型的变量时,最好将*(或&)写在紧靠变量之前,而不是紧跟基本类型之后。
这样可以避免一些理解上的误区,比如:
再来看一个指针的指针的例子:
理论上,对于指针的级数没有限制,你可以定义一个浮点类型变量的指针的指针的指针的指针,再来看如下的声明:
这里,p被声明为一个包含5个元素(int类型的指针)的数组。
另外,我们还可以在同一个声明中混合实用*和&,如下:
注:p1是一个int类型的指针的指针;p2是一个int类型的指针的引用;p3是一个int类型引用的指针(不合法!);p4是一个int类型引用的引用(不合法!)。
const修饰符
当你想阻止一个变量被改变,可能会用到const关键字。
在你给一个变量加上const修饰符的同时,通常需要对它进行初始化,因为以后的任何时候你将没有机会再去改变它。
例如:
上述两个变量n和m其实是同一种类型的——都是const int(整形恒量)。
因为C++标准规定,const关键字放在类型或变量名之前等价的。
我个人更喜欢第一种声明方式,因为它更突出了const修饰符的作用。
当const与指针一起使用时,容易让人感到迷惑。
例如,我们来看一下下面的p和q的声明:
他们当中哪一个代表const int类型的指针(const直接修饰int),哪一个代表int类型的const指针(const直接修饰指针)?实际上,p和q都被声明为const int类型的指针。
而int类型的const指针应该这样声明:
这里,p和q都是指向const int类型的指针,也就是说,你在以后的程序里不能改变*p的值。
而r是一个const指针,它在声明的时候被初始化指向变量n (即r=&n;)之后,r的值将不再允许被改变(但*r的值可以改变)。
组合上述两种const修饰的情况,我们来声明一个指向const int类型的const 指针,如下:
下面给出的一些关于const的声明,将帮助你彻底理清const的用法。
不过请注意,下面的一些声明是不能被编译通过的,因为他们需要在声明的同时进行初始化。
为了简洁起见,我忽略了初始化部分;因为加入初始化代码的话,下面每个声明都将增加两行代码。
注:p1是指向char类型的指针的指针;p2是指向const char类型的指针的指针;p3是指向char类型的const指针;p4是指向const char类型的const指针;p5是指向char类型的指针的const指针;p6是指向const char类型的指针的const指针;p7是指向char类型const指针的const指针;p8是指向const char类型的const指针的const指针。
typedef的妙用
typedef给你一种方式来克服“*只适合于变量而不适合于类型”的弊端。
你可以如下使用typedef:
这里的p和q都被声明为指针。
(如果不使用typedef,q将被声明为一个char 变量,这跟我们的第一眼感觉不太一致!)下面有一些使用typedef的声明,并且给出了解释:
typedef经常用在一个结构声明之前,如下。
这样,当创建结构变量的时候,允许你不使用关键字struct(在C中,创建结构变量时要求使用struct关键字,如struct tagPOINT a;而在C++中,struct可以忽略,如tagPOINT b)。
函数指针
函数指针可能是最容易引起理解上的困惑的声明。
函数指针在DOS时代写TSR
程序时用得最多;在Win32和X-Windows时代,他们被用在需要回调函数的场合。
当然,还有其它很多地方需要用到函数指针:虚函数表,STL中的一些模板,Win NT/2K/XP系统服务等。
让我们来看一个函数指针的简单例子:
这里p被声明为一个函数指针,这个函数带一个char类型的参数,并且有一个int类型的返回值。
另外,带有两个float类型参数、返回值是char类型的指针的指针的函数指针可以声明如下:
那么,带两个char类型的const指针参数、无返回值的函数指针又该如何声明呢?参考如下:
“右左法则”是一个简单的法则,但能让你准确理解所有的声明。
这个法则运用如下:从最内部的括号开始阅读声明,向右看,然后向左看。
当你碰到一个括号时就调转阅读的方向。
括号内的所有内容都分析完毕就跳出括号的范围。
这样继续,直到整个声明都被分析完毕。
对上述“右左法则”做一个小小的修正:当你第一次开始阅读声明的时候,你必须从变量名开始,而不是从最内部的括号。
下面结合例子来演示一下“右左法则”的使用。
阅读步骤:
1. 从变量名开始——fp1
2. 往右看,什么也没有,碰到了),因此往左看,碰到一个*——一个指针
3. 跳出括号,碰到了(int)——一个带一个int参数的函数
4. 向左看,发现一个*——(函数)返回一个指针
5. 跳出括号,向右看,碰到[10]——一个10元素的数组
6. 向左看,发现一个*——指针
7. 向左看,发现int——int类型
总结:fp1被声明成为一个函数的指针,该函数返回指向指针数组的指针. 再来看一个例子:
阅读步骤:
1. 从变量名开始——arr
2. 往右看,发现是一个数组——一个5元素的数组
3. 向左看,发现一个*——指针
4. 跳出括号,向右看,发现()——不带参数的函数
5. 向左看,碰到*——(函数)返回一个指针
6. 跳出括号,向右发现()——不带参数的函数
7. 向左,发现*——(函数)返回一个指针
8. 继续向左,发现int——int类型
还有更多的例子:。