C++入门篇---(命名空间、缺省参数、以及输入、输出)

news2024/12/24 0:14:02

前言

  c++ 我来了,恭喜牛牛解锁新世界.开启c++的学习之旅.

🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨
🐻推荐专栏: 🍔🍟🌯C语言进阶
🔑个人信条: 🌵知行合一
🍉本篇简介:>:讲解C++入门知识,命名空间的引入,C++中的输入输出,缺省参数的出现,函数重载等.
金句分享:
✨花会沿路盛开,你以后的路也是!✨

目录

  • 前言
  • 一、命名空间
    • 1.1 " 命名空间"出现的原因:
    • 1.2 "命名空间"的访问
    • 1.3 "命名空间"中可以放什么?
    • 1.4 "命名空间"的名称会冲突吗?
  • 二、C++中的"输入"与"输出"
    • 2.1 为啥C++中的头文件头文件有的没有`.h`?
    • 2.2 学习新语言(C++)了,向世界问个好吧!
  • 三、缺省参数
    • 3.1 缺省参数的分类:
  • 四、"关键字"表(资料)
      • c语言"关键字":
      • c++ "关键字"表:

一、命名空间

1.1 " 命名空间"出现的原因:

大家可以先看下面这一段代码:

#include <stdio.h>
int time = 0;

int main()
{
	
	printf("%d", time + 66);
	return 0;
}

  这代码可以正常运行,但是我们引用头文件#include <time.h>后,代码就不能正常运行了.

#include <time.h>
#include <stdio.h>
int time = 0;

int main()
{
	
	printf("%d", time + 66);
	return 0;
}

原因是time变量在<time.h>中已经有定义了,所以这里就会报变量重命名错误.
在这里插入图片描述

不要以为我们只需要修改time变量,例如改为:time1就可以解决问题.

  因为可能只是解决这一个当前问题,但是如果有一天,在某个工程中,包含某个头文件之后,代码就出现了一堆错误,那时候就有你头痛的时候了.
在这里插入图片描述

除此之外,往往一个大型的项目是由多个人即一个团队组合完成的,程序猿A程序猿B可能会使用同一个名称去定义变量,这是难以预的.在C/C++中,变量函数和C++中“类” 都是大量存在的,它们的名称将都存在于全局作用域中,可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化(在本地范围内保持不重名,在外面重名无所谓,只要加上作用域限定符即可),以避免命名冲突或名字污染,namespace关键字的出现就是针对这种问题的.

介绍" :: "域作用限定符

#include <stdio.h>

int a = 10;

int main()
{
	int a = 5;
	printf("a=%d\n", a);	//1
	printf("a=%d\n", ::a);	//2
}

结果:

5
10

  在C语言中,如果局部全局都定义了同一个变量,局部优先,所以1处打印5.

  在C中如果不能屏蔽这个局部变量,还想优先访问全局域的10似乎很难办到,
而C++中,引入了域作用限定符,2处的a前面有" :: "域作用限定符,虽然是 空格+:: ,这里空格就代表全局域,所以这里打印10.

#include <iostream>
int main()
{
	int a = 5;
	//std是库里面的一个命名空间
	std::cout << a << std::endl;//cout函数后续会介绍
	return 0;
}

1.2 "命名空间"的访问

命名空间好似一堵墙,他将变量,函数等对象围了起来,防止与别处的污染,要想使用其中的变量,函数等内容有三种方法.

1.使用域作用限定符号:

命名空间的名字::+变量名

2.展开命名空间(将墙拆掉):

using namespace +命名空间的名字

3,展开命名空间的部分

using std::cout;

🌰栗子1:使用域作用限定符(指定空间访问):

//不推荐
#include <stdio.h>
namespace cjn
{
	int a = 66;
}
int main()
{
	printf("a=%d\n", cjn::a);
}
return 0;

🌰栗子2:将命名空间展开:(将墙拆掉)

//推荐
#include <stdio.h>
namespace cjn
{
	int a = 66;
}
using namespace cjn;//将命名空间展开
int main()
{
	printf("a=%d\n", a);
	return 0;
}

  在写项目时,不适合使用第二种方法,因为将墙拆掉后,命名空间中的所有成员都暴露在全局域中,这样依旧会产生命名冲突.所以一般采用指定的命名空间域限定符来指定访问.

虽然不建议展开全部的命名空间,但是我们可以展开部分常用的,也很方便.
🌰栗子3:

#include <iostream>
//展开部分常用的
using std::cout;
using std::endl;
int main()
{
	int a = 0, b = 1;
	cout <<"abcdef" << " " << "a" << endl;
	cout << a << endl;
	cout << b << endl;
	return 0;
}

总结:

在这里插入图片描述

展示解决命名冲突问题:

#include <stdio.h>
int a = 10;
namespace cjn
{
	int a = 66;
}
//using namespace cjn;//将命名空间展开,不推荐
int main()
{
	int a = 5;
	printf("a1=%d\n", a);
	printf("a2=%d\n", ::a);//推荐
	printf("a3=%d\n", cjn::a);//推荐
	return 0;
}

注意:

如果命名空间未展开,编译器在默认情况下并不会搜索命名空间中的变量.(即如果在命名空间的定义的变量,不指定访问,编译器会找不到).

访问优先级:

优先级:局部域>全局域

小试牛刀:下面这段代码打印的结果是什么?

#include <stdio.h>
int a = 10;
namespace cjn
{
	int a = 66;
}
int main()
{
	int a = 5;
	printf("a=%d\n", a);
	return 0;
}

在这里插入图片描述

答案:

5

  因为局部域优先,并且cjn命名空间没有展开编译器不会去里面搜索,所以也不会与全局变量冲突.

1.3 "命名空间"中可以放什么?

.  命名空间中可以定义很多东西,可以有函数,变量,结构体等,甚至可以嵌套其他命名空间等.

#include <iostream>
namespace cjn
{
	//定义函数
	void swap(int* e1, int* e2)
	{
		int tmp = *e1;
		*e1 = *e2;
		*e2 = tmp;
	}
	//声明类型
	struct student
	{
		char name[10];
		int age;
	};

	//定义变量
	int time = 0;
	
	//嵌套定义命名空间
	namespace cjn2
	{
		int time = 2;
		int c=1;
	}

}

int main()
{
	//printf("%d", cjn::time + 66);
	std::cout << cjn::time + 66 << std::endl;
	return 0;
}

1.4 "命名空间"的名称会冲突吗?

命名空间是为了解决全局变量的命名冲突问题,那它自己的名字会被冲突吗?

🌰栗子1:同一个源文件中,定义相同名称的命名空间

#include <iostream>

namespace cjn
{
	int a = 66;
}

namespace cjn
{
	int b = 7;
}


int main()
{
	std::cout << cjn::a <<" "<< cjn::b << std::endl;
	return 0;
}

🌰栗子2:同一个源文件中,定义相同名称的命名空间

#include <iostream>
//test1.cpp
namespace cjn
{
	int a = 66;
}
//test2.cpp
namespace cjn
{
	int b = 7;
}


int main()
{
	std::cout << cjn::a <<" "<< cjn::b << std::endl;
	return 0;
}

运行:

66 7

  答案是并不冲突的,对于在同一个文件或在不同文件中定义的同名命名空间,会被自动合并在一起.

总结:

  1. 命名空间是为了解决名称冲突而出现的,当然,也要学会合理使用命名空间,指定访问是更加推荐的这一种写法.

  2. 命名空间中可以定义很多东西,可以有函数,变量,结构体等,也可以嵌套其他命名空间等.

  3. 在不同文件中定义同一名称的命名空间不会报错,而是会被合并!

二、C++中的"输入"与"输出"

在讲上面的命名空间的时候,牛牛刚刚使用了cout函数,有没有友友好奇是怎么回事呢?

其实cincout是C++中的"输入"和"输出"函数.
使用时需要注意以下几点:

  1. 使用cout标准输出对象(控制台)和cin标准输入对象(键盘)之前,需要包含头文件< iostream >头文件,并且使用命名空间std
  1. coutcin是全局的流对象,endl是特殊的C++符号,与C语言中的"\n"类似,表示换行,他们都包含在包含< iostream >头文件中。
  2. <<流插入运算符>>流提取运算符。是一种很形象的"输入"和"输出"符号.
  3. 很明显使用C++的cincout更方便,不需要像c语言中的printf/scanf输入输出时使用格式输出符(%d,%c,%lf等等).cincout可以自动识别变量类型
  4. 实际上coutcin分别是ostream和istream类型的对象.

2.1 为啥C++中的头文件头文件有的没有.h?

  早期标准库将所有功能在全局域中实现,声明在.h后缀的头文件中,使用时只需包含对应头文件即可.
  后来C++中出现了命名空间的概念,就将实现方在std命名空间下,为了和C头文件区分,也为了正确使用命名空间,规定C++头文件不带.h;旧编译器(vc 6.0)中还支持<iostream.h>格式,后续编译器已不支持,因此推荐使用<iostream>+std的这种不带.h方式。

2.2 学习新语言(C++)了,向世界问个好吧!

#include <iostream>

using namespace std;//使用std命名空间域,cin和cout在其中被定义

int main()
{
	cout << "hello world" << endl;
	//endl代表换行
	return 0;
}

但是对于输出特定格式的内容,cincout就没有那么方便了,比如:输出一个浮点型数字,保留3位小数.

printf表示:(建议)

#include <stdio.h>
using namespace std;
int main(){
	float a=0;
	scanf("%f",&a);
	printf("%0.3f",a);
}

cout表示:(不建议)

#include <iostream>
#include <iomanip>
using namespace std;
int main(){
	float a;
	cin>>a;
	cout<<fixed<<setprecision(5)<<a;
}

这种情况建议使用printf,毕竟C++是兼容c的,可以混着用.😍

三、缺省参数

  缺省参数是声明或定义函数时为函数的参数指定一个缺省值。在调用该函数时,如果没有指定实参则采用该形参的缺省值,否则使用指定的实参

  缺省参数舔狗或者可以说是备胎,如果有合适的人选(实参),则就会立马抛弃缺省参数,毫不犹豫选择实参,实在没有合适的人选(实参),才会没办法选择缺省参数.

🌰栗子

#include <iostream>
using std::cout;
using std::endl;
void fun(int a=3)
{
	cout << a << endl;
}
int main()
{
	fun();		//1
	fun(66);	//2
	return 0;
}

运行结果:

3
66

1.调用函数时,并没有传参,则会采用缺省值(默认值)a=3.
2.调用函数时,实参传入了66,则采用实参.

  还记得顺序表的初始化操作吗?

//顺序表的初始化操作
void InitSQL(SQL* SL)
{
	assert(SL);//防止传入空指针
	SL->capacity = MAX;
	SL->size = 0;//顺序表初始状态,当前数据量为0
	SL->data = (DataType*)malloc(sizeof(DataType) * MAX);//初始化顺序表大小
	if (SL->data == NULL)
	{
		printf("初始化申请空间失败.\n");
	}
}

此时我们可以使用缺省参数将这个函数优化一下.

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
typedef int DataType;

typedef struct SQList
{
	DataType* data;//指向一段连续的内存空间
	int size;//代表当前顺序表的长度
	int capacity;//表示最大容量
}SQL;

//优化后
void InitSQL(SQL* SL,int defaultsize=4)//这里设置了缺省参数
{
	assert(SL);//防止传入空指针
	SL->capacity = defaultsize;
	SL->size = 0;//顺序表初始状态,当前数据量为0
	SL->data = (DataType*)malloc(sizeof(DataType) * defaultsize);//初始化顺序表大小
	if (SL->data == NULL)
	{
		printf("初始化申请空间失败.\n");
	}
}

int main()
{
	SQL SL;//用顺序表类型创建一个SL顺序表
	InitSQL(&SL);			//1
	//InitSQL(&SL,100);		//2
	return 0;
}

  这里我们就可以很灵活的设置顺序表的初始大小,如果我们确定这个顺序表至少要100大小的空间,我们可以采用方法2,直接传入实参过去.这样就减少了一次次扩容至100,更加方便高效.

如果不确定大小,这里不传参也没事,采用方法1,默认会设置为4大小的空间.

这种方式更加灵活,C语言中采用宏替换则没有这么灵活.

除此之外,缺省参数还有一些需要注意:

3.1 缺省参数的分类:

1.全缺省
2.半缺省

#include <iostream>
using std::cout;
using std::endl;

void fun1(int a=1, int b=2, int c=3,int d=4)//全缺省
{
	cout << "fun1:" << endl;
	cout << a << endl;
	cout << b << endl;
	cout << c << endl;
	cout << d << endl << endl;
}

void fun2(int a , int b , int c = 3, int d = 4)//半缺省
{
	cout << "fun2:" << endl;
	cout << a << endl;
	cout << b << endl;
	cout << c << endl;
	cout << d << endl;
}

int main()
{
	fun1();
	fun2(6,7);
	return 0;
}

注意1:

半缺省值不是指一定非要缺省一半的参数,而是指缺省部分参数.
半缺省参数必须从右往左依次来给出,不能间隔着给.
传参是从左到右依次赋值,这点需要注意.这里需要理解,不是考记.

比如:如果赋值传参也是从右往左,则缺省的意义不大.

在这里插入图片描述

注意2:
缺省值必须是常量或者全局变量

注意3:
缺省参数,在函数的声明定义不能同时给出,只能在声明中给出.

原因:
如果两个地方都给出缺省参数,则双方给的默认参数不同时,编译器不知道选择哪一个.
只有在声明给出缺省参数,是因为声明在.h文件中,而定义在另外一个.cpp文件中,
在编译阶段后,不同的.cpp文件会生成不同的目标文件.如果在定义处写则会出错,具体看下图.

(1)定义处写缺省参数:(报错)
在这里插入图片描述
(2)声明处写缺省参数:(正确写法)

在这里插入图片描述

声明中给出缺省参数:

  1. 如果不传参,编译后会默认替换为默认值,
  2. 如果传参过去,则编译器会采用实参.

其中对于函数定义部分从始至终都是没有缺省值的,因为此处无论上面声明是何种情况,都会传足够的参数过来.

C++入门第一篇就讲到这里了,后续会讲解函数重载,引用、内联函数等.请保持持续关注哦!!!
如果文章对大家有用的话记得一键三连哦!💗💗💗
如果文章中有部分错误之处,可以私信牛牛,互相讨论哦!!!

在这里插入图片描述

最后附上C语言和C++关键字的资料

四、"关键字"表(资料)

c语言"关键字":

(图片来源于:百度)
在这里插入图片描述

c++ "关键字"表:

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

30年后,茶产业规模是现在的10倍

做个预言&#xff1a;30年后&#xff0c;茶产业是现在的10倍 【5.21是世界茶日】 杭州中国茶博会&#xff0c;我来啦 人工智能越让生产效率越来越高 常用物质将会唾手可得 人闲着&#xff0c;无法体会活着的意义&#xff0c;才是挑战 田园诗茶生活方式会有一席之地 趣讲大白话&…

【EMC专题】案例:读一读TI的按接口选择ESD器件指南

在TI的官网上看到一份ESD by Interface Selection Guide,也就是按接口选择ESD器件指南。因此想读一读看看一起学习一下。 首先看一下文档,是比较简明的。可以看到不同的接口推荐了一些不同的保护器件。因为应用环境不一样,所有有不同的器件封装(如单体、集成TVS等),这样在…

Spyder可在线使用!?

不同安装&#xff0c;如果想使用spyder进行编程&#xff0c;可以用其在线版&#xff0c;和本地版功能一样&#xff0c;就是有点慢。 另外需要用chrome浏览器&#xff0c;用火狐没法正常访问。 Spyder可以在线使用&#xff0c;所以在没有安装python环境的电脑上&#xff0c;想…

Linux常用命令——hostname命令

在线Linux命令查询工具 hostname 显示和设置系统的主机名 补充说明 hostname命令用于显示和设置系统的主机名称。环境变量HOSTNAME也保存了当前的主机名。在使用hostname命令设置主机名后&#xff0c;系统并不会永久保存新的主机名&#xff0c;重新启动机器之后还是原来的主…

字符串匹配--BF算法和KMP算法

0.前言 字符串函数strstr相信大家都不陌生–就是在一个字符串&#xff08;主串&#xff09;中找查找另一个字符串&#xff08;子串&#xff09;&#xff0c;并返回子串在主串中的位置。那么这个函数是怎么实现的呢&#xff1f;这就涉及字符串匹配的问题&#xff0c;本章就让我们…

Node.js 事件循环和事件派发器

目录 1、process.nextTick() 介绍 2、setTimeout() 3、零延迟 4、setInterval() 5、递归setTimeout 6、setImmediate() 7、Node.js 事件派发器 1、process.nextTick() 介绍 Node.js中 process.nextTick函数以一种特殊的方式与事件循环交互。 当你试图理解Node.js事件循…

Redis数据结构——QuickList、SkipList、RedisObjective

承接上文&#xff0c;本文主要介绍QuickList、SkipList、RedisObjective 四、 Redis数据结构-QuickList 问题1&#xff1a;ZipList虽然节省内存&#xff0c;但申请内存必须是连续空间&#xff0c;如果内存占用较多&#xff0c;申请内存效率很低。怎么办&#xff1f; ​ 答&a…

计算机操作系统(慕课版)第二章课后题答案

一、简答题 (1)什么是前趋图&#xff1f;试画出下面四条语句的前趋图. S1&#xff1a;axy&#xff1b; S2&#xff1a;bz1&#xff1b; S3&#xff1a;ca-b&#xff1b; S4&#xff1a;wc1&#xff1b; 答&#xff1a;前趋图(Precedence Graph)是一个有向无循环图&#xff0c;…

chatgpt赋能Python-pythondataframe取出一列

用 Python Dataframe 取出一列 数据分析中&#xff0c;用到的数据往往是有多列多行的。而在实际的分析过程中&#xff0c;我们需要针对其中的某一列进行处理。这个时候&#xff0c;Python中的Dataframe就成了我们的利器。 在这篇文章中&#xff0c;我们将教你如何使用Python …

chatgpt赋能Python-pythongit

PythonGit&#xff1a;使Git操作更加高效 Git作为目前最流行的版本控制工具之一&#xff0c;已经被广泛应用于软件开发、Web开发等领域。PythonGit则是一个基于Python编写的Git客户端库&#xff0c;可以让开发者们更加高效地进行Git操作&#xff0c;提高开发效率。 PythonGit…

Qt Quick系列(2)—核心元素类型(1)

作者&#xff1a;CCAccept 专栏&#xff1a;Qt Quick 文章目录 前言ItemRectangleTextImageMouseArea 总结 前言 Qt Quick的元素分为 1、视觉元素&#xff08;如Rectangle&#xff09;具有几何属性 2、非视觉元素&#xff08;如Timer&#xff09;提供一般功能&#xff0c;用…

learn C++ NO.5 ——类和对象(3)

日期类的实现 在前面类和对象的学习中&#xff0c;由于知识多比较多和碎&#xff0c;需要一个能够将之前所学知识融会贯通的东西。下面就通过实现日期类来对类和对象已经所学的知识进行巩固。 日期类的基本功能&#xff08;.h文件&#xff09; //Date.h//头文件内容 #includ…

makefile 学习(4): makefile基础

0. 官方文档 GNU Make 官方网站: https://www.gnu.org/software/makeGNU Make 官方文档下载地址: https://www.gnu.org/software/make/manual/Makefile Tutorial:https://makefiletutorial.com/ 1.基本要求 1.1 基本格式 targets : prerequisties [tab键] command target : …

一、MongoDB简介

文章目录 一、MongoDB简介1、NoSQL简介2、什么是MongoDB ?3、MongoDB 特点4、安装mongodb5、MongoDB 概念解析5.1 数据库5.2 文档5.3 集合5.4 MongoDB 数据类型 6、适用场景 一、MongoDB简介 1、NoSQL简介 NoSQL(NoSQL Not Only SQL)&#xff0c;意即反SQL运动&#xff0c;…

关于在spyder,jupyter notebook下创建虚拟环境(pytorch,tensorflow)均有效

anaconda下载地址 https://www.anaconda.com/download/ 下载完成后打开anaconda目录下的 anaconda prompt 在命令行中输入下面的命令创建一个叫tf2.0的虚拟环境&#xff08;“tf2.0”是建立的Conda虚拟环境的名字&#xff0c;可以自拟&#xff09; conda create -n tf2.0 p…

chatgpt赋能Python-pythonfor遍历

Python for 遍历&#xff1a;优雅地遍历数据结构 对于任何编程语言来说&#xff0c;遍历是一项基本操作。而在 Python 中&#xff0c;遍历是一项非常简单和优雅的操作。Python 提供了多种遍历数据结构的方法&#xff0c;包括 for 循环、while 循环、迭代器和生成器等。本文将介…

模板和STL【C++初阶】

目录 一、前言 二、函数模板 三、类模板 四、STL 一、前言 以前我们写swap函数时&#xff0c;对每一种类型的变量都要写一份swap函数&#xff0c;但是他们的格式都是一样的&#xff0c;未免有些麻烦 因此&#xff0c;我们今天学习的模板就可以针对广泛的类型而不是具体的类…

chatgpt赋能Python-pythondir

Python dir命令&#xff1a;探索Python模块的秘密 如果你是一名Python开发者&#xff0c;那么你一定或多或少接触过dir这个命令。但是&#xff0c;你了解dir到底能做什么吗&#xff1f;这篇文章将会介绍dir命令的用途、用法以及一些有趣的技巧。 什么是dir命令 简单来说&…

chatgpt赋能Python-pythonfind

Python文件搜索工具Pythonfind 在开发过程中&#xff0c;文件搜索工具是一个非常重要的工具。在大型项目中&#xff0c;可能需要查找特定类型的文件或者在代码库中查找特定的代码块。 Pythonfind是一个非常强大和灵活的python文件搜索工具&#xff0c;可以帮助我们简化这个过程…

chatgpt赋能Python-pythonforelse

Python For Else 详解&#xff1a;用 Python 的人都应该了解的语法结构 在 Python 中&#xff0c;一个常见的语法结构是 for...else。这种语法结构让循环变得更加直接明了&#xff0c;也让代码更加易读和易懂。 什么是 Python For Else 在 Python 中&#xff0c;for...else …