ToB企服应用市场:ToB评测及商务社交产业平台

标题: C++ 基础入门篇 [打印本页]

作者: 丝    时间: 2024-8-5 19:58
标题: C++ 基础入门篇
定名空间

   定义:定名空间必要用到namespace关键字,其后跟着定名空间的名字(自定义),再接着就是一对花括号,花括号里的内容可以是定义的变量,函数或者结构体
  1. //对命名空间的定义
  2. namespace fun
  3. {
  4.         int a = 1;
  5.         int add(int left,int right)
  6.         {
  7.                 return left + right;
  8.         }
  9.         struct Node
  10.         {
  11.                 int data;
  12.                 int* next;
  13.         }
  14. }
  15. int main()
  16. {
  17.         // ...
  18.         return 0;
  19. }
复制代码
作用: 定名空间只能使用在全局域中,要知道,不同的域可以定义同一个变量,但在同一个域中不能定义同一个变量,防止在定义某个变量时与全局域产生辩论,空间定名在全局域中就起到了隔离的作用,其在全局域中形成一个定名空间域,该域与全局域互不干涉
若不使用定名空间,在编译时,体系就会出现下面的报错:

产生辩论时,定名空间就起到了作用:

可看到,rang成功打印,并未与在全局域中头文件中的rand产生辩论。在访问定名空间的成员时,必要用到" : : " 域作用限定操纵符,在该操纵符前添上自定义的定名空间名,在其后添上所要访问的成员即可。对于全局域中变量的访问,也可以用此操纵符,在该操纵符前的位置不添任何东西即可
定名空间中定义的变量,函数或结构体作用的范围是全局

可以看到在全局域中定义的函数,也可使用定名空间中定义的变量

一个定名空间中可以有多个子定名空间:

编译查找一个变量/声明时,默认只会在局部和全局域查找,所以想要查找定名空间里的变量/声明的话,有三种方式:
  1. namespace fun
  2. {
  3.         int a = 10;
  4.         int add(int x, int y)
  5.         {
  6.                 return x + y;
  7.         }
  8. }
  9. int main()
  10. {
  11.         printf("%d\n",fun::a); // 指定访问命名空间中的变量a
  12.         return 0;
  13. }
复制代码
  1. namespace fun
  2. {
  3.         int a = 22;
  4.         int b = 10;
  5. }
  6. using namespace fun; //将命名空间域全部展开成全局域
  7. int main()
  8. {
  9.         printf("%d\n",a); // 展开之后命名空间中的变量可当成全局域使用
  10.         printf("%d\n",b);
  11.         return 0;
  12. }
复制代码
全部睁开相当于将定名空间中的成员全部暴露在全局域中,这样的方式一般不适用于多人合作时使用,一般适用于个人做一些训练或简单步伐时
  1. namespace fun
  2. {
  3.         int x = 1;
  4.         int y = 0;
  5. }
  6. unsing fun::x; // 只对命名空间中的x进行展开
  7. int main()
  8. {
  9.         printf("%d\n",x);
  10.         printf("%d\n",fun::y);
  11.         return 0;
  12. }
复制代码
  
  输入与输出

与C语言相比,C++的输入和输出会更加的灵活,来看下面的代码:


与C语言相比,C++的输入和输出不消一个指定格式或者多个格式,在写代码时,更加的方便
  1. int main()
  2. {
  3.         int i = 10;
  4.         char b = 'A';
  5.         std::cout << i << std::endl << b << std::endl; // 不同类型的变量进行同时输出
  6.         return 0;
  7. }
复制代码
缺省参数

   概念:缺省参数是在声明或定义函数时,为函数设定一个缺省值(默认值),在调用函数时,若没有指定实参,则该函数就默认使用设定好的缺省值;若指定了实参,则使用实参的值
  例:

可知,当第一次调用函数时,没有指定实参,a的值默以为缺省值4;当第二次调用函数时,指定了实参44,a的值为指定的实参
分类:


调用全缺省参数时,传参只能从左往右传,不可跳跃传参


设定缺省参数时,只能从右往左依次设定缺省值,不可跳跃设定
   注:在设定缺省参数时,若函数的定义与声明是分离的,那么只能在函数声明时给定缺省值
  函数重载

   概念:C++支持在同一作用域中出现同名的函数,但要求函数的参数范例不同,或者函数的参数个数不同,这样的同名函数就被称作函数重载,函数重载弥补了C语言中出现的功能雷同,但不能定义为同名函数的征象
  分类:



引用和const引用

概念:引用就是给变量取别名
定义:
  1. int main()
  2. {
  3.         int i = 10;
  4.         int& ri = i;
  5.         return 0;
  6. }
复制代码
从上面的代码来看,ri就是 i 的别名, i 是被引用的变量,而 ri 是引用变量,编译器不会给引用变量另外开辟空间,而是与它所引用的变量公用同一个空间,由此可推出,ri 与 i 共用一个空间,若引用变量改变则被引用变量也会改变,反之,亦如此

引用的特性:

引用的使用方式:

我们知道,通常实现两个数交换的函数,实参传的是地址,必要用指针吸收,然后再对指针进行解引用,而上面的代码,传的是值,分别用引用变量来进行吸收,然后进行两个数的交换,体现了改变引用变量就会改变被引用变量这一个特点
  1. typedef struct Node
  2. {
  3.         int data;
  4.         struct Node* next;
  5. }SL,*phead;
  6. void SeqListInit(phead& plist,int x)
  7.                         //这里的引用相当于 phead& plist = list
  8. {
  9.         SL* tmp = (SL*)malloc(sizeof(SL));
  10.         if (tmp == NULL)
  11.         {
  12.                 perror("malloc fail!");
  13.                 exit(-1);
  14.         }
  15.         tmp->data = x;
  16.         tmp->next = NULL;
  17.         plist = tmp;
  18. }
  19. int main()
  20. {
  21.         phead plist = NULL;//plist用来存放链表头结点的地址
  22.         //对链表进行初始化
  23.         SeqListInit(plist, 4);
  24.         return 0;
  25. }
复制代码
代码效果:

我们知道,在对链表进行初始化时,想要改变一级指针的指向就必要取一级指针的地址,然后用二级指针来吸收,通过对二级指针的解引用,就可改变一级指针的指向。但是使用C++的引用,就可以代替指针传参,也可以改变一级指针的指向,这样可较好的简化步伐,避免指针肴杂
但是并不是任何场景都能用引用返回的,例如 :

调用fun()这个函数,在该函数中创建了临时变量 i ,而当调用结束,函数烧毁时,该临时变量 i 也会被烧毁,而对引用了 i 的别名进行修改,会造成越界访问,所以引用临时变量不可作为返回值
   注:引用和指针是相辅相成的,二者皆不可替换,指针可以改变指向的对象,但引用改变不了,C++使用引用能简化步伐,避免使用复杂的指针
  const引用
   概念:被const修饰的变量会受到权限,不可修改,有const的对象必须用const引用
  1. int main()
  2. {
  3.         const int x = 10;
  4.         const int& rx = x; //x受到const的修饰,所以rx也必须用const
  5.         return 0;
  6. }
复制代码
在使用const的过程中必要留意:不可放大const对象的权限,但可以缩小没有const对象的权限,缩小了之后,被引用的对象可修改,引用对象不可修改


上述代码中的x受到了const的权限,当定义x的别名rx时,由于别名是不开辟空间的,与被引用对象共用同一块空间,x的空间受到限制,那么定义的x的别名rx也应受到限制,不可放大x空间的权限

对const运用的延伸
  1. int main()
  2. {
  3.         const int x = 10;
  4.         int y = x;
  5.         return 0;
  6. }
复制代码
上述代码是精确的,不存在权限的放大,const修饰的是x的这块空间,而将x赋给y是一种赋值拷贝,y也不与x共用同一块空间,所以不存在权限的放大
  1. int main()
  2. {
  3.         const int a = 10;
  4.         const int* p1 = &a;
  5.         int* p2 = p1;
  6.         return 0;
  7. }
复制代码
上述代码是错误的,这属于权限的放大,第二条指令中,const修饰的不是p1本身,修饰的是p1所指向的空间a不可被修改,且a受到了const的修饰,而将p1赋值给p2时,p2也应受到const的限制
  1. int main()
  2. {
  3.         int b = 20;
  4.         const int* p1 = &b;
  5.         const int* p2 = p1;
  6.         return 0;
  7. }
复制代码
上述代码精确,属于权限的缩小,变量b可读可写,将b的地址赋给p1后,p1又受到了const的权限,所以p2也应受到const的权限
  1. int main()
  2. {
  3.         int c = 30;
  4.         int* const p1 = &c;
  5.         int* p2 = p1;
  6.         return 0;
  7. }
复制代码
上述代码时精确的,不存在权限的放大,const修饰p1本身,并非其指向的空间c,p2 = p1相当于一个赋值拷贝,不影响p2对空间c的读写
指针和引用的关系
注:引用在语法上是不开辟空间的,但在汇编底层中,实在跟指针是一样的,也必要开辟空间:
  1. int main()
  2. {
  3.         int x = 10;
  4.         int& rx = x;
  5.         rx += 1;
  6.         int* px = &x;
  7.         *px += 1;
  8.         return 0;
  9. }
复制代码
上述代码的汇编层如下:

可以看到,底层的汇编指令一模一样,所以引用的底层实现,与指针的实现是一样的
inline(内联函数)

   概念:inline放在返回值的前面,用来修饰函数。编译器会对调用的inline函数进行睁开,睁开之后就不必要再建立函数栈帧,可进步步伐的效率,inline函数的出现实在就是代替C语言中的宏,宏的使用直接在预处理阶段直接睁开,可进步效率,但是宏的坑太多,所以就有了inline的出现
  例如:
  1. #include <iostream>
  2. using std::cout;
  3. using std::endl;
  4. inline int add(int x,int y)
  5. {
  6.         int ret = x + y;
  7.         return ret;
  8. }
  9. int main()
  10. {
  11.         int sum = add(1,2);
  12.         cout << sum << endl;
  13.         return 0;
  14. }
复制代码
不睁开的汇编指令如下:

不睁开时,会有一条call指令,编译器会进入到call指令中,然后跳转到函数中,创建函数栈帧
睁开的汇编指令如下:

显然睁开后是没有call指令的,直接将实现函数的指令睁开,也不会去建立函数栈帧
注:inline的使用对编译器来说只是一种发起,对函数的睁开与否还是取决于编译器,一般来说,对于内联函数,如果短小的函数,编译器会对其进行睁开,若函数过大,编译器就不会对其进行睁开
在debug版本下,内联函数默认是不睁开的(必要手动设置)

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4