目录
1.全局变量
2.命名空间
3.using声明的使用
4.C++语法增强
5.C++中的const
6.引用
7.引用在函数中的使用
8.引用作为函数的返回值类型使用
9.常值引用
10.指针引用
内容
#include
using namespace std;
/****************************************************************************************************
* 1.全局变量的定义与使用:
* ① “::”表示作用域运算符(C++独有,C没有);
****************************************************************************************************/
int a = 30; // 全局变量
void test01()
{
int a = 40; // 局部变量
cout<<"局部变量a="<b?a:b,在C语言中返回的是数值(即变量的值),而在C++中返回的是变量本身(引用)。
* 9.C++中的const:
* ① 字面意思,常数,不变的数值,它是一个限定符,用来限定一个变量不允许改变,将一个对象转换成一个常量;
* ② C++中用const修饰的常量是内部链接常量,即只在本文件中起作用,外部文件不可见(可使用external修饰,用于外部使
* 用),而C中为外部链接,可外部使用;
* ③ C++中const表示常量,不需要分配内存长度,而C中表示只读变量,需要分配内存长度;
* ④ C++中全局const,当声明extern或者对变量进行取地址时,编译器会为其分配内存空间,变量存储在只读数据段,否则不
* 开辟内存空间,内存中的数据不可修改;
* ⑤ 当以变量的形式初始化const修饰的变量时,系统也会为其分配内存空间,如“const int a=b;”,系统直接为a开辟空间
* 而不创建符号表中;
* ⑥ 对于自定义数据类型,如类对象,也会分配内存空间,如:const Person person;
* ⑦ 尽量用const替换宏(#define)。
* 10.C中的const:
* ① const修饰的全局变量,变量名只读,内存空间在文学常量区(只读),不可修改;
* ② const修饰的局部变量,变量名只读,内存分配在栈区(可读写),可通过寻访变量地址来简介修改空间内容;
****************************************************************************************************/
void test05()
{
cout << sizeof (false) << endl; // 输出结果为1,表示bool类型占一个字节大小
bool flag = true; // C语言中没有这种类型
flag = 100; // 给bool类型赋值时,非0值会自动转换成true(1),0值直接转换成false(0)
cout << flag << endl; // 输出结果为1
// C++中的const
const int constA = 10; // 给常量赋值,变量名和值存于符号常量表,此时还没有开辟内存空间
cout << "constA:" << constA << endl; // 输出结果为10,输出符号表中的值
int* p = (int*)&constA; // 寻访变量constA的地址,此时给constA开辟内存空间,并将constA=10存储在开辟的内存空间中
*p = 300; // 并修改constA地址中的值
cout << "*p:" << *p << endl; // 输出结果为300,表示修改成功
cout << "constA:" << constA << endl; // 输出结果仍为10,输出的是符号表中的值,但地址中的值已经被修改。
cout << "*&constA:" << &constA << endl; // 输出结果为:0x61feb8,存储地址
cout << "*p:" << *&p << endl; // 输出结果为:0x61feb8,存储地址
const Student stu = {"Han", 27};
cout << "stu.Name: " << stu.mName << endl << "stu.Age: " << stu.mAge << endl;
}
/****************************************************************************************************
* 11.在C++中尽量使用const替换宏(#define):#define MAX 1024,替换为:const int max=1024
* ① const有类型,可进行编译器类型安全检查,使用时调用的是short类型的函数;
* ② #define无类型,不可进行类型检查,使用时调用的是int类型重载的函数;
* ② const有作用域,可用于顶一个指定作用域下的有效常量,而#define不重视作用域,默认定义处到文件结尾或到
* #undef A(卸载宏常量)处可用;
* ③ 宏可以在命名空间中定义,但是它不属于该命名空间,const属于命名空间中的元素。
****************************************************************************************************/
#define PARAM 128
const short param = 128;
void func(short a){
cout << "short a = " << a << endl;
}
void func(int a){ // 函数的重载,即对上面同一个函数的重新定义,增加其函数功能
cout << "int a = " << a << endl;
}
void test06()
{
func(PARAM); // 输出结果为:int,表示宏调用的是int类型的重载函数
func(param); // 输出结果为:short,表示宏调用的是short类型的重载函数
}
/****************************************************************************************************
* 12.引用(reference),C++中能用引用绝不用指针,它是C++给函数传递地址的途径,可以简单理解为给已有的变量取一个别名。
* ① &别名:表示引用,&在此不是求地址运算,而是起标识作用;
* ② 给某个变量取别名,就定义某个变量,即操作别名就相当于操作原;
* ③ 从上到下替换,之后操作别名就相当于操作变量本身;
* ④ 一个变量可以有多个引用,引用一旦初始化之后,就不能修改。
* ⑤ 不能给“数组”取别名,但是有方法给数组取别名:
* i> 用小括号提高优先级取别名;
* ii> 用typedef给数组类型取个别名;
* ⑥ 其实引用的本质是常量指针:Type& ref = val; // Type* const ref = &val;内部实现用户不可见
****************************************************************************************************/
void test07(){
int a = 10;
int& b = a; // 给变量 a 取一个别名 b
// int *b = &a; // 这是取地址,上面是取别名,即引用变量
cout << "a:" << a << endl; // 10
cout << "b:" << b << endl; // 10
cout << "------------" << endl;
//操作 b 就相当于操作 a 本身
b = 100;
cout << "a:" << a << endl; // 100
cout << "b:" << b << endl; // 100
cout << "------------" << endl;
//一个变量可以有 n 个别名
int& c = a;
c = 200;
cout << "a:" << a << endl; // 200
cout << "b:" << b << endl; // 200
cout << "c:" << c << endl; // 200
cout << "------------" << endl;
//下面是取地址符,a,b,c的地址都是相同的,而变量之间的赋值,其地址是不同的。
cout << "a:" << &a << endl; // 0x61feb4
cout << "b:" << &b << endl; // 0x61feb4
cout << "c:" << &c << endl << endl; // 0x61feb4
/*** 给数组取别名 ***/
int veryLongNameArray[5] = {11, 22, 33, 44, 55};
// 方法i> 用小括号提高优先级取别名;
int (&sArr)[5] = veryLongNameArray; // 因为&引用标识符的优先级<中括号,所以要加个小括号
// 方法ii> 用typedef给数组类型取个别名;
typedef int TYPE_ARR[5]; // 先定义一个数组类型,(TYPE_ARR就是一张数组类型,其中有5个int类型的元素)
TYPE_ARR &new_arr = veryLongNameArray; // 再用该数据类型继承或者赋值veryLongNameArray的值
for(int i=0;i<5;i++){
cout << sArr[i] << " ";
cout << new_arr[i] << "| ";
}
cout << endl;
}
/****************************************************************************************************
* 13.引用(reference)在函数中的使用,引用作为函数的参数使用:
* ① 函数内部修改函数外部的值要么使用地址,要么使用引用(推荐);
* ② 通过引用参数产生的效果同按地址传递是一样的。
* ③ 引用的语法更清楚简单:
* 1) 函数调用时传递的实参不必加“&”符
* 2) 在被调用函数中不必在参数前加“*”符。
* ④ 引用作为其它变量的别名而存在,因此在一些场合可以代替指针。
****************************************************************************************************/
void mySwap1(int a, int b) // 一个简单的交换函数,但是这种交换只在该方法内起作用,对外不起作用
{
int temp = a;
a = b;
b = temp;
}
void mySwap2(int *a, int *b) // a=&data,b=&data2,因为交换的是地址里的内容,所以对外起作用
{
int temp = *a;
*a = *b;
*b = temp;
}
void mySwap3(int &a, int &b) // 给参数取别名
{
int temp = a;
a = b;
b = temp;
}
void test08()
{
int data1 = 10, data2 = 20;
cout << "原始数据:data1 = " << data1 << ", data2 = " << data2 << endl; // 原始数据
// mySwap1(data1, data2);
// cout << "第一种交换:data1 = " << data1 << ", data2 = " << data2 << endl; // 第一种值传递,交换失败
// mySwap2(&data1, &data2);
// cout << "第二种交换:data1 = " << data1 << ", data2 = " << data2 << endl; // 第二种地址传递:交换成功
mySwap3(data1, data2);
cout << "第三种交换:data1 = " << data1 << ", data2 = " << data2 << endl; // 第三种引用传递:交换成功
}
/****************************************************************************************************
* 14.引用作为函数的返回值类型使用:
* ① 不能返回局部变量的引用,因为局部变量在函数结束时,内存内容将被释放,引用无效,所以返回空,但是当不使用引用时
* 局部变量的值可以被返回;
* ② 可以返回静态变量的引用,因为静态变量的生命周期较长,所以引用有效。但是静态变量只会作用1次,所以第二次调用静
* 态变量时,就不起作用了;
* ③ 函数返回,优先返回该变量的值,但是当函数作为左值被赋值时,返回的是引用(当然,如果函数返回值类型不是引用,那
* 么函数定义不合法,报错)而不是值;所以作为右值,返回的自然就是值。
****************************************************************************************************/
//返回局部变量引用
int& func01(){
int a = 10; //局部变量
return a;
}
//返回静态变量引用
int& func02(){
static int a = 20;
cout << "static int a : " << a << endl;
return a;
}
void test09()
{
// //不能返回局部变量的引用
// int& ret01 = func01(); // 现在ret01是函数返回值a的别名
// cout << "func01: ret01 = " << ret01 << endl; // 函数结束局部变量被释放,所以返回空
cout << func02() << endl; // 打印输出20,函数内的cout,和该行的cout输出结果一样
// 如果函数做左值,那么必须返回引用,如下
func02() = 100; //打印输出20,为函数内的cout输出;此句返回的是a,即此句就相当于a=100
cout << func02() << endl; // 打印输出100 100,因为第二次调用静态变量的定义不起作用,所以函数内打印输出为100,此句打印输出也是100
}
/****************************************************************************************************
* 15.常左值引用,即给左值取一个别名:左值一般为变量名a=2,a即为左值,2即为右值。
* ① 定义格式:const Type& ref = val;
* ② 字面量不能赋给引用,但是可以赋给 const 引用 const 修饰的引用,不能修改;
* ③ 将函数的形参定义为常量引用的好处: 引用不产生新的变量,减少形参与实参传递时的开销。
* ④ 由于引用可能导致实参随形参改变而改变,将其定义为常量引可以消除这种副作用。如果希望实参随着形参的改变而改变,
* 那么使用一般的引用,如果不希望实参随着形参改变,那么使用常引用。
* 16.常右值引用,即给右值取一个别名:好处是该引用将不能被修改。
****************************************************************************************************/
typedef struct {
int num;
char name[16];
} STU;
void printSTU01(STU tmp) // 普通结构体变量作为形参,开销太大,使用引用减小开销
{
cout << "姓名: " << tmp.name << ", 年龄: " << tmp.num <<"; 占用空间大小:"<< sizeof (tmp) << endl;
// 姓名: Lucy, 年龄: 16; 占用空间大小:20,其中20字节=name16字节+num4字节,此时形参tmp开辟了存储空间。
}
void printSTU02(STU &tmp) // 使用引用减小开销,此时传入的参数是变量lucy,即“STU &tmp = lucy”,此时tmp只是lucy的别名
{
tmp.num = 80; // 内容可被修改
cout << "姓名: " << tmp.name << ", 年龄: " << tmp.num <<"; 占用空间大小:"<< sizeof (tmp) << endl;
// 姓名: Lucy, 年龄: 80; 占用空间大小:20,其中20字节=name16字节+num4字节,但是此时别名tmp并没有开辟存储空间。
}
void printSTU03(const STU &tmp) // 常引用,使其传入内容不可修改
{
// tmp.num = 80; // 此句报错,说明不可被修改
cout << "姓名: " << tmp.name << ", 年龄: " << tmp.num <<"; 占用空间大小:"<< sizeof (tmp) << endl;
// 姓名: Lucy, 年龄: 16; 占用空间大小:20,其中20字节=name16字节+num4字节,但是此时别名tmp并没有开辟存储空间。
}
void test10()
{
STU lucy = {16, "Lucy"};
printSTU01(lucy);
printSTU02(lucy);
printSTU03(lucy);
const int &num = 10; // 按照C++的语法,此处的10的类型为const int,而非int,所以此句签名使用的是const int
cout << "num = " << num << endl;
}
/****************************************************************************************************
* 17.指针引用:
* ① 好处是传参时,不需要取地址,方便操作;
****************************************************************************************************/
#include
#include
// 在C中使用指针的的实现方法
void myStr1(char **p_str) // 形参p_str == &str,*p_str == *&str ==str
{
*p_str = (char *)calloc(1, 32); // 申请空间用于写入内容
strcpy(*p_str, "Hello World!"); // 写入内容
}
//在C++中使用引用实现该方法,指针的引用
void myStr2(char* &r_str) // char* &r_str == str,即r_str == str
{
r_str = (char *)calloc(1, 32);
strcpy(r_str, "Hello World!");
}
void test11()
{
// 需求:封装一个函数,从堆区给str申请一个空间并赋值为“Hello World!”
char *str = NULL;
// myStr1(&str); // 在C中使用指针就传入地址,并往其地址内写入内容
myStr2(str); // 在C++中使用引用就直接传入变量名即可,
cout << "str = " << str << endl;
free(str); // 释放堆区
}
int main()
{
// test01();
// test02();
// test03();
// test04();
// test05();
// test06();
// test07();
// test08();
// test09();
// test10();
// test11();
return 0;
}