初识C++--C++入门

一、命名空间

        在c语言中存在着名字冲突的问题,即不能出现同名,会出现错误。而在c++中变量、函数和后⾯要学到的类都是⼤量存在的,这些变量、函数和类的名称将都存在于全局作⽤域中,可能会导致很多冲突。为了解决这个问题,所以出现了namespace这个关键字--定义命名空间的关键字。

        使⽤命名空间的⽬的是对标识符的名称进⾏本地化,以避免命名 冲突或名字污染。

        定义命名空间,需要使⽤到namespace关键字,后⾯跟命名空间的名字,然后接⼀对{}即可,{}中 即为命名空间的成员。命名空间中可以定义变量/函数/类型等。 

                                              

        namespace本质是定义出⼀个域,这个域跟全局域各⾃独⽴,不同的域可以定义同名变量,所以下⾯的rand不在冲突了。

        C++中域有函数局部域,全局域,命名空间域,类域;域影响的是编译时语法查找⼀个变量/函数/ 类型出处(声明或定义)的逻辑,所有有了域隔离,名字冲突就解决了。局部域和全局域除了会影响编译查找逻辑,还会影响变量的⽣命周期,命名空间域和类域不影响变量⽣命周期。

        namespace只能定义在全局,当然他还可以嵌套定义。

        项⽬⼯程中多⽂件中定义的同名namespace会认为是⼀个namespace,不会冲突。 

        C++标准库都放在⼀个叫std(standard)的命名空间中。

        ⼀般开发中是⽤项⽬名字做命名空间名。

命名空间的使用

        编译查找⼀个变量的声明/定义时,默认只会在局部或者全局查找,不会到命名空间⾥⾯去查找。所以我们要使⽤命名空间中定义的变量/函数,有三种⽅式:

        指定命名空间访问,项⽬中推荐这种⽅式。 

namespace ts
{
	int  rand = 10;
}
int main()
{
	printf("%d\n", ts::rand);
	
	return 0;
}

        其中printf("%d\n", ts::rand);的::符号,叫做域作用限定符,在进行嵌套使用时,使用时name+::+name::+对象就可以了。

        using将命名空间中某个成员展开,项⽬中经常访问的不存在冲突的成员推荐这种⽅式。

        

namespace ts
{
	int a = 10;
	int b = 20;
	double c = 3.0;
}
//展开命名空间中的a
using ts::a;
int main()
{
//不使用域作用限定符也可以使用a
	printf("%d\n", a);
	printf("%d\n", ts::b);
	printf("%.2lf\n", ts::c);
	return 0;
}

         展开命名空间中全部成员,项⽬不推荐,冲突⻛险很⼤,⽇常⼩练习程序为了⽅便推荐使⽤。

namespace ts
{
	int a = 10;
	int b = 20;
	double c = 3.0;
}
//将ts命名域全部展开
using namespace ts;
int main()
{
	printf("%d\n", a);
	printf("%d\n", b);
	printf("%.2lf\n", c);
	return 0;
}

        总而言之,命名空间就像是给一些变量、函数建立一度围墙,使得外面不能直接拿走使用,使用域作用限定符就像是在墙上开了一扇门,可以拿出指定的对象,展开命名空间,就是推倒这堵墙,使得与外面相交,可以随意使用其中的成员。

二、C++的输入输出

        <iostream>是InputOutputStream的缩写,是标准的输⼊、输出流库,定义了标准的输⼊、输 出对象。

         std::cin 是istream类的对象,它主要⾯向窄字符(narrowcharacters(oftypechar))的标准输 ⼊流。

        std::cout 是ostream类的对象,它主要⾯向窄字符的标准输出流。

        std::endl 是⼀个函数,流插⼊输出时,相当于插⼊⼀个换⾏字符加刷新缓冲区。

        <<是流插入运算符, >>是流提取运算符。(C语⾔还⽤这两个运算符做位运算左移/右移)

        使⽤C++输⼊输出更⽅便,不需要像printf/scanf输⼊输出时那样,需要⼿动指定格式,C++的输⼊输出可以⾃动识别变量类型(本质是通过函数重载实现的),其实最重要的是 C++的流能更好的⽀持⾃定义类型对象的输⼊输出。  

        cout/cin/endl等都属于C++标准库,C++标准库都放在⼀个叫std(standard)的命名空间中,所以要 通过命名空间的使⽤⽅式去⽤他们。

        ⼀般⽇常练习中我们可以usingnamespacestd,实际项⽬开发中不建议usingnamespacestd。

using namespace std;
int main()
{
	int a = 1;
	int b = 2;
	double c = 3.0;
	char d = 'd';
	char e[] = "hello world";
	cout << a << endl;
	cout << b << endl;
	cout << c << endl;
	cout << d << endl;
	cout << e << endl;
	return 0;
}

        

using namespace std;
int main()
{
	int a = 0;
	char b = 0;
	cout << "请输入a的值:";
	cin >> a;
	cout << a << endl;
	cout << "请输入b的值:";
	cin >> b;
	cout << b << endl;
	return 0;
}

三、缺省参数

        缺省参数是声明或定义函数时为函数的参数指定⼀个缺省值。在调⽤该函数时,如果没有指定实参则采⽤该形参的缺省值,否则使⽤指定的实参,缺省参数分为全缺省和半缺省参数。(有些地⽅把 缺省参数也叫默认参数)

        全缺省就是全部形参给缺省值,半缺省就是部分形参给缺省值。C++规定半缺省参数必须从右往左依次连续缺省,不能间隔跳跃给缺省值。

        

#include <iostream>
#include <assert.h>
using namespace std;

//全缺省
void My_print(int a = 1, int b = 2, char c = 'd', double d = 3.0)
{
	cout << "a = " << a << endl;
	cout << "b = " << b << endl;
	cout << "c = " << c << endl;
	cout << "d = " << d << endl;
	cout << endl;
}
int main()
{
	My_print();
	My_print(10);
	My_print(10,20);
	My_print(10,20,'c');
	My_print(10,20,'c', 6.7);
	return 0;
}

        

#include <iostream>
#include <assert.h>
using namespace std;

//半缺省
void My_print(int a, int b, char c = 'd', double d = 3.0)
{
	cout << "a = " << a << "  ";
	cout << "b = " << b << "  ";
	cout << "c = " << c << "  ";
	cout << "d = " << d << "  ";
	cout << endl;
}
int main()
{
	My_print(10, 20);
	My_print(10, 20, 'c');
	My_print(10, 20, 'c', 6.7);

	return 0;
}

        带缺省参数的函数调⽤,C++规定必须从左到右依次给实参,不能跳跃给实参。

        函数声明和定义分离时,缺省参数不能在函数声明和定义中同时出现,规定必须函数声明给缺省值。

四、函数重载

        C++⽀持在同⼀作⽤域中出现同名函数,但是要求这些同名函数的形参不同,可以是参数个数不同或者类型不同。这样C++函数调⽤就表现出了多态⾏为,使⽤更灵活。C语⾔是不⽀持同⼀作⽤域中出现同名函数的。

#include <iostream>
#include <assert.h>
using namespace std;

void Add(int a, int b)
{
	cout << a + b << endl;
}
void Add(int a,int b,int c)
{
	cout << a + b + c << endl;

}
void Add(double a,double b)
{
	cout << a + b << endl;

}
int main()
{
	Add(1, 2);
	Add(1, 2, 3);
	Add(1.2, 1.3);
	return 0;
}

五、引用

        概念以及定义

        引⽤不是新定义⼀个变量,⽽是给已存在变量取了⼀个别名,编译器不会为引⽤变量开辟内存空间, 它和它引⽤的变量共⽤同⼀块内存空间。

        类型& 引⽤别名=引⽤对象;

        

#include <iostream>
#include <assert.h>
using namespace std;

int main()
{
	int a = 1;
	cout <<"a = " << a << endl;
	int& b = a;
	b = 3;
	cout <<"a = " << a << endl;
	cout <<"b = " << b << endl;
	return 0;
}

        这里我们对a和b取地址,可以看到,两者的地址是一样的,这也印证了引用与被引用的对象使用的是同一块地址。

        引⽤的特性

        引⽤在定义时必须初始化

        

#include <iostream>
#include <assert.h>
using namespace std;
int main()
{
	int a = 1;
	int& ra = a;
	//int& rb;
    //error C2530: “rb”: 必须初始化引用
	return 0;
}

        ⼀个变量可以有多个引⽤

        

#include <iostream>
#include <assert.h>
using namespace std;
int main()
{
	int a = 1;
	int& ra = a;
	int& b = a;
	int& c = a;
	cout << a << endl;  //1
	cout << ra << endl; //1
	cout << b << endl;  //1
	cout << c << endl;  //1
	return 0;
}

        引⽤⼀旦引⽤⼀个实体,再不能引⽤其他实体

        引⽤的使⽤

        引⽤在实践中主要是于引⽤传参和引⽤做返回值中减少拷⻉提⾼效率和改变引⽤对象时同时改变被引⽤对象。

        引⽤传参跟指针传参功能是类似的,引⽤传参相对更⽅便⼀些。

        引⽤和指针在实践中相辅相成,功能有重叠性,但是各有特点,互相不可替代。C++的引⽤跟其他语⾔的引⽤(如Java)是有很⼤的区别的,除了⽤法,最⼤的点,C++引⽤定义后不能改变指向, Java的引⽤可以改变指向。

        

#include <iostream>
#include <assert.h>
using namespace std;
void Swap(int*p1,int*p2)
{
	int tmp = *p1;
	*p1 = *p2;
	*p2 = tmp;
}
void _Swap(int& p1, int& p2)
{
	int tmp = p1;
	p1 = p2;
	p2 = tmp;
}
int main()
{
	//交换两者数据
	int a = 1;
	int b = 2;
	cout << "交换前a = " << a << "  ";
	cout << "b = " << b << endl;
	Swap(&a,&b);
	cout << "交换后a = " << a << "  ";
	cout << "b = " << b << endl;
	int c = 3;
	int& rc = c;
	int d = 4;
	int& rd = d;
	cout << "交换前c = " << c << "  ";
	cout << "d = " << d << endl;
	_Swap(rc,rd);
	cout << "交换后c = " << c << "  ";
	cout << "d = " << d << endl;
	return 0;
}

        可以看出引用的效果跟指针是一样的,并且传参还不需要取地址,使用时也不需要解引用,会简单方便一些,也更不容易出错。

        const引⽤

        可以引⽤⼀个const对象,但是必须⽤const引⽤。const引⽤也可以引⽤普通对象,因为对象的访问权限在引⽤过程中可以缩⼩,但是不能放⼤。

        不需要注意的是类似 int& rb = a*3; double d = 12.34; int& rd = d; 这样⼀些场 景下a*3的和结果保存在⼀个临时对象中, int& rd = d 也是类似,在类型转换中会产⽣临时对象存储中间值,也就是时,rb和rd引⽤的都是临时对象,⽽C++规定临时对象具有常性,所以这⾥就触发了权限放⼤,必须要⽤常引⽤才可以。    

        所谓临时对象就是编译器需要⼀个空间暂存表达式的求值结果时临时创建的⼀个未命名的对象, C++中把这个未命名对象叫做临时对象。

#include <iostream>
#include <assert.h>
using namespace std;
//const引用
int main()
{
	const int a = 10;
	//int& ra = a;  error C2440: “初始化”: 无法从“const int”转换为“int &”
	//在这里的引用是对a的访问权限的放大
	const int& ra = a;
	//这样就可以了

	//这里是对b的访问权限的缩小
	int b = 2;
	const int& rb = b;
	
	int c = 3;
	//int& rc = c * 3;  error C2440: “初始化”: 无法从“int”转换为“int &”
	//c进行了运算,产生了临时变量,而临时变量具有常性,须用const进行修饰
	const int& rc = c;

	double d = 1.2;
	//int& rd = d;  error C2440: “初始化”: 无法从“double”转换为“int &”
	//dounle类型转为int类型,在类型转换中会产⽣临时对象存储中间值
	const int& rd = d;

	//简单总结产生临时对象的情况
	//1.表达式相加(运算)
	//2.调用一个函数 函数传值返回--回生成一个临时拷贝 这个临时拷贝是临时对象
	//3.类型转换
	return 0;
}
        指针和引⽤的关系--相似以及区别

        C++中指针和引⽤在实践中他们相辅相成,功 能有重叠性,但是各有⾃⼰的特点,互相不可替代。

        语法概念上引⽤是⼀个变量的取别名不开空间,指针是存储⼀个变量地址,要开空间。

        引⽤在定义时必须初始化,指针建议初始化,但是语法上不是必须的。

        引⽤在初始化时引⽤⼀个对象后,就不能再引⽤其他对象;⽽指针可以在不断地改变指向对象。

        引⽤可以直接访问指向对象,指针需要解引⽤才是访问指向对象。

        sizeof中含义不同,引⽤结果为引⽤类型的⼤⼩,但指针始终是地址空间所占字节个数(32位平台下 占4个字节,64位下是8byte)

        指针很容易出现空指针和野指针的问题,引⽤很少出现,引⽤使⽤起来相对更安全⼀些。

六、inline

        ⽤inline修饰的函数叫做内联函数,编译时C++编译器会在调⽤的地⽅展开内联函数,这样调⽤内联函数就需要建⽴栈帧了,就可以提⾼效率(类似于宏)。

        inline对于编译器⽽⾔只是⼀个建议,也就是说,你加了inline编译器也可以选择在调⽤的地⽅不展 开,不同编译器关于inline什么情况展开各不相同,因为C++标准没有规定这个。inline适⽤于频繁 调⽤的短⼩函数,对于递归函数,代码相对多⼀些的函数,加上inline也会被编译器忽略(这样看来,编译器也有它自己的小脾气)。

        C语⾔实现宏函数也会在预处理时替换展开,但是宏函数实现很复杂很容易出错的,且不⽅便调 试,C++设计了inline⽬的就是替代C的宏函数。

        vs编译器debug版本下⾯默认是不展开inline的,这样⽅便调试,debug版本想展开需要设置⼀下 以下两个地⽅。

        inline不建议声明和定义分离到两个⽂件,分离会导致链接错误。因为inline被展开,就没有函数地址,链接时会出现报错。

七、nullptr

        C++中NULL可能被定义为字⾯常量0,或者C中被定义为⽆类型指针(void*)的常量。不论采取何种定义,在使⽤空值的指针时,都不可避免的会遇到⼀些⿇烦,本想通过f(NULL)调⽤指针版本的 f(int*)函数,但是由于NULL被定义成0,调⽤了f(intx),因此与程序的初衷相悖。f((void*)NULL); 调⽤会报错。

        C++11中引⼊nullptr,nullptr是⼀个特殊的关键字,nullptr是⼀种特殊类型的字⾯量,它可以转换成任意其他类型的指针类型。使⽤nullptr定义空指针可以避免类型转换的问题,因为nullptr只能被隐式地转换为指针类型,⽽不能被转换为整数类型。

        

void f(int a )
{
	cout << "int a" << endl;
}
void f(int* a)
{
	cout << "int*a" << endl;
}
int main()
{
	f(1);
	f(NULL);
	f(nullptr);
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/893629.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

十三、Python基础语法(字符串str-中)

一、切片 使用下标可以获得字符串中指定的一个字符&#xff0c;使用切片可以获取字符中多个字符。 字符串[start: end: step] start&#xff1a;开始位置的下标 end&#xff1a;结束位置的下标&#xff08;end对应的位置数据取不到&#xff09; step&#xff1a;步长&#…

要在 Git Bash 中使用 `tree` 命令,下载并手动安装 `tree`。

0、git bash 安装 git(安装,常用命令,分支操作,gitee,IDEA集成git,IDEA集成gitee,IDEA集成github,远程仓库操作) 1、下载并手动安装 tree 下载 tree.exe 从 tree for Windows 官方站点 下载 tree 的 Windows 可执行文件。tree for Window&#xff1a;https://gnuwin32.source…

FreeRTOS学习笔记1

结合汇编 ldr r3, pxCurrentTCB ldr r2 R3 value0x20000054,R2 value0x2002B950 pxCurrentTCB 020028950 pxTopOfStsck 0x2002B8FC 解释这些寄存器的值是怎么变化的 1. ldr r3, pxCurrentTCB 这一行指令将 全局变量 pxCurrentTCB 的地址加载到寄存器 r3 中。pxCu…

【论文精读】RELIEF: Reinforcement Learning Empowered Graph Feature Prompt Tuning

RELIEF: Reinforcement Learning Empowered Graph Feature Prompt Tuning 前言AbstractMotivationSolutionRELIEFIncorporating Feature Prompts as MDPAction SpaceState TransitionReward Function Policy Network ArchitectureDiscrete ActorContinuous ActorCritic Overall…

【C++】精妙的哈希算法

&#x1f680;个人主页&#xff1a;小羊 &#x1f680;所属专栏&#xff1a;C 很荣幸您能阅读我的文章&#xff0c;诚请评论指点&#xff0c;欢迎欢迎 ~ 目录 一、哈希结构1、哈希概念2、哈希函数3、哈希冲突3.1 闭散列3.2 开散列 4、完整代码 一、哈希结构 1、哈希概念 A…

C# WPF 仿 Android Toast 效果

转载请注明出处: https://blog.csdn.net/hx7013/article/details/142860084 主职Android, 最近需要写一些WPF的程序作为上位机&#xff0c;目前WPF的MessageBox过于臃肿&#xff0c;且想找一个内置的非阻塞的简单提示一直找不到&#xff0c;想到了Android的Toast所以写了这个扩…

低代码可视化-uniapp购物车页面-代码生成器

购物车页面是电子商务网站或应用程序中的一个关键功能页面&#xff0c;它允许用户查看、编辑和管理他们选择加入购物车的商品。下面通过低代码可视化实现一个uniapp购物车页面&#xff0c;把购物车整个事件都集成进去。实现完成后可以保存为页面模板。 收货地址选择 如果尚未…

yolov9目标检测/分割预测报错AttributeError: ‘list‘ object has no attribute ‘device‘常见汇总

这篇文章主要是对yolov9目标检测和目标分割预测测试时的报错&#xff0c;进行解决方案。 在说明解决方案前&#xff0c;严重投诉、吐槽一些博主发的一些文章&#xff0c;压根没用的解决方法&#xff0c;也不知道他们从哪里抄的&#xff0c;误人子弟、浪费时间。 我在解决前&…

JVM 实战篇(一万字)

此笔记来至于 黑马程序员 内存调优 内存溢出和内存泄漏 内存泄漏&#xff08;memory leak&#xff09;&#xff1a;在Java中如果不再使用一个对象&#xff0c;但是该对象依然在 GC ROOT 的引用链上&#xff0c;这个对象就不会被垃圾回收器回收&#xff0c;这种情况就称之为内…

Rust usize类型(用来表示地址的类型)Rust地址与指针的区别(Rust指针)

文章目录 Rust usize类型Rust地址与指针的区别&#xff08;指针有数据类型&#xff0c;而地址只是一个数字&#xff09;指针地址使用场景示例 Rust usize类型 在Rust中&#xff0c;地址通常表示为usize类型&#xff0c;这是因为usize是专门设计用来存储指针大小的无符号整型&a…

vue综合指南(五)

​&#x1f308;个人主页&#xff1a;前端青山 &#x1f525;系列专栏&#xff1a;Vue篇 &#x1f516;人终将被年少不可得之物困其一生 依旧青山,本期给大家带来Vuet篇专栏内容:vue综合指南 目录 81 简述每个周期具体适合哪些场景 82、Vue $forceUpdate的原理 83、vue获取数…

MySQL—关于数据库的CRUD—(增删改查)

文章目录 关于数据库的使用&#xff1a;1. 数据库的背景知识&#xff1a;2. MYSQL数据库软件的使用&#xff08;MYSQL安装的问题在另一篇博客中讲解&#xff09;。&#xff08;1&#xff09;启动MYSQL数据库软件&#xff08;2&#xff09;开始使用数据库程序&#xff1a;1&…

leetcode动态规划(一)-理论基础

本节主要参考&#xff1a;代码随想录 题目分类 动态规划释义 动态规划&#xff0c;英文&#xff1a;Dynamic Programming&#xff0c;简称DP&#xff0c;如果某一问题有很多重叠子问题&#xff0c;使用动态规划是最有效的。 动态规划中每一个状态一定是由上一个状态推导出来…

车辆管理的SpringBoot技术革新

摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了车辆管理系统的开发全过程。通过分析车辆管理系统管理的不足&#xff0c;创建了一个计算机管理车辆管理系统的方案。文章介绍了车辆管理系统的系统分析部分&…

使用 OpenWebUI 一键部署 Mistral-Large-Instruct-2407-AWQ

教程及模型简介 该教程是使用 OpenWebUI 一键部署 Mistral-Large-Instruct-2407-AWQ&#xff0c;相关环境和配置已经搭建完成&#xff0c;只需克隆启动容器即可进行推理体验。 Mistral-Large-Instruct-2407-AWQ 是法国人工智能公司 Mistral AI 发布的新一代旗舰 AI 模型&…

操作系统简介:作业管理

作业管理 一、作业管理1.1 作业控制1.2 作业的状态及其转换1.3 作业控制块和作业后备队列 二、作业调度2.1 调度算法的选择2.2 作业调度算法2.3 作业调度算法性能的衡量指标 三、人机界面 作业&#xff1a;系统为完成一个用户的计算任务&#xff08;或一次事务处理&#xff09;…

RabbitMQ 核心功能详解

引言 在现代分布式系统中&#xff0c;消息队列已经成为一种不可或缺的组件。它不仅能够实现应用之间的解耦&#xff0c;还能提高系统的灵活性和可扩展性。RabbitMQ 是一款基于 AMQP&#xff08;Advanced Message Queuing Protocol&#xff09;协议的消息中间件&#xff0c;以其…

【人工智能】人工智能的10大算法详解(优缺点+实际案例)

人工智能&#xff08;AI&#xff09;是现代科技的重要领域&#xff0c;其中的算法是实现智能的核心。本文将介绍10种常见的人工智能算法&#xff0c;包括它们的原理、训练方法、优缺点及适用场景。 1. 线性回归&#xff08;Linear Regression&#xff09; 模型原理 线性回归…

2021年10月自考《软件开发工具》03173试题

目录 一.选择题 二.填空题 三.简答题 五.综合题 一.选择题 1.下列各项属于集成化开发工具的是 &#xff08;书中&#xff09;P96页 A.WORDSTAR B.FLOW C.Dictionary/3000 D.Visual Studio 2.软件工程的思想主要服务于 &#xff08;书中&#xff09;P84页面 A.用户 B.项目…

虚拟现实辅助工程技术在现代汽车制造中的重要性

虚拟现实辅助工程&#xff08;VR Aided Engineering&#xff09;&#xff0c;简称VAE&#xff0c;作为数字化转型的重要手段&#xff0c;在各行各业被越来越广泛的应用。随着汽车变得越来越复杂&#xff0c;虚拟现实辅助工程技术逐渐成为汽车行业产品开发过程中不可或缺的一部分…