C++基础
C++基础
1、C++初识
1.1注释
两种格式
- 单行注释:
// 描述信息
- 通常放在一行代码的上方,或者一条语句的末尾,对该行代码说明
- 多行注释:
/* 描述信息 */
- 通常放在一段代码的上方,对该段代码做整体说明
1.2变量
作用:给一段指定的内存空间起名,方便操作这段内存
语法:数据类型 变量名 = 初始值;
示例:
#include<iostream>
using namespace std;
int main(){
int a = 10;
cout<<"a = "<<a<<endl;
system("pause");
return 0;
}
注意:C++在创建变量时,必须给变量一个初始值,否则会报错
system("pause")意思就是让程序暂停一下,然后按任意键继续,初学的时候最多见于程序的末尾处,用于看运行结果,避免程序一闪而过。相同的我们还可以用getchar(),避免程序运行完直接结束而看不到运行结果。
1.3常量
作用:用于记录程序中不可更改的数据
C++定义常量两种方式
- #define 宏常量:
#define 常量名 常量值
- 通常在文件上方定义,表示一个常量
- const修饰的变量
const 数据类型 常量名 = 常量值
- 通常在变量定义前加关键字const,修饰该变量为常量,不可修改
示例:
#include<iostream>
using namespace std;
// 1、宏常量 #define 常量名 常量值
#define day 7
int main(){
cout<<"一周之内一共有"<<day<<" 天"<<endl;
// day = 8 报错,宏常量不可以修改
//2、const 常量
const int month = 12;
cout<<"一年之内一共有"<<month<<"个月份"<<endl;
// month = 24; 报错因为常量不可以修改
}
1.4关键字
作用:关键字是C++中预先保留的单词(标识符)
- 在定义变量或者常量时候,不要用关键字
C++关键字如下:
asm | do | if | return | typedef |
---|---|---|---|---|
auto | double | inline | short | typeid |
bool | dynamic_cast | int | signed | typename |
break | else | long | sizeof | union |
case | enum | mutable | static | unsigned |
catch | explicit | namespace | static_cast | using |
char | export | new | struct | virtual |
class | extern | operator | switch | void |
const | false | private | template | volatile |
const_cast | float | protected | this | wchar_t |
continue | for | public | throw | while |
default | friend | register | true | |
delete | goto | reinterpret_cast | try |
1.5标识符命名规则
作用:C++规定给标识符(变量、常量)命名时,有一套自己的规则
- 标识符不能是关键字
- 标识符只能由字母、数字、下划线组成
- 第一个字符必须为字母或下划线
- 标识符中字母区分大小写
2、数据类型
C++规定在创建一个变量或者常量时,必须要指定出相应的数据类型,否则无法给变量分配内存
2.1整型
作用:整型变量表示的是整数类型的数据
C++中能够表示整型的类型有以下几种方式,区别在于所占内存空间不同:
数据类型 | 占用空间 | 取值范围 |
---|---|---|
short(短整型) | 2字节 | (-2^15 ~ 2^15-1) |
int(整型) | 4字节 | (-2^31 ~ 2^31-1) |
long(长整形) | Windows为4字节,Linux为4字节(32位),8字节(64位) | (-2^31 ~ 2^31-1) |
long long(长长整形) | 8字节 | (-2^63 ~ 2^63-1) |
2.2sizeof关键字
作用:利用sizeof关键字可以统计数据类型所占内存大小
语法: sizeof( 数据类型 / 变量)
示例:
#include<iostream>
using namespace std;
int main(){
cout<<"short 类型所占的内存空间为: "<<sizeof(short)<<endl;
cout<<"int 类型所占的内存空间为: "<<sizeof(int)<<endl;
cout<<"long 类型所占的内存空间为: "<<sizeof(long)<<endl;
cout<<"long long 类型所占的内存空间为: "<<sizeof(long long)<<endl;
}
注意这里是long long不是longlong
2.3实型(浮点型)
作用:用于表示小数
浮点型变量分为两种:
- 单精度float
- 双精度double
两者的区别在于表示的有效数字范围不同。
数据类型 | 占用空间 | 有效数字范围 |
---|---|---|
float | 4字节 | 7位有效数字 |
double | 8字节 | 15~16位有效数字 |
示例:
#include<iostream>
using namespace std;
int main(){
float f1 = 3.14f;
double d1 = 3.14;
cout<<f1<<endl;
cout<<d1<<endl;
cout<<"float sizeof = "<<sizeof(f1)<<endl;
cout<<"double sizeof = "<<sizeof(d1)<<endl;
//科学计数法
float f2 = 3e2; //3*10^2
cout<<"f2 = "<<f2<<endl;
float f3 = 3e-2;//3*10^(-2)
cout<<"f3 = "<<f3<<endl;
system("pause");
return 0;
}
2.4字符型
作用:字符型变量用于显示单个字符
语法:char ch = 'a';
注意1:在显示字符型变量时,用单引号将字符括起来,不要用双引号
注意2:单引号内只能有一个字符,不可以是字符串
- C和C++中字符型变量只占用1个字节。
- 字符型变量并不是把字符本身放到内存中存储,而是将对应的ASCII编码放入到存储单元
示例:
#include<iostream>
using namespace std;
int main(){
char ch = 'a';
cout<<ch<<endl;
cout<<sizeof(ch)<<endl;
// ch = "abcde"; // 不可以使用双引号
// ch = 'abcde'; // 单引号里面只能引用一个字符
cout<<(int)ch<<endl; //查看字符所对应的ASCII编码
ch = 97; //可以直接使用ASCII给字符型变量赋值
cout <<ch<<endl;
}
2.5转义字符
作用:用于表示一些不能显示出来的ASCII字符
现阶段我们常用的转义字符有: \n \\ \t
转义字符 | 含义 | ASCII码值(十进制) |
---|---|---|
\a | 警报 | 007 |
\b | 退格(BS) ,将当前位置移到前一列 | 008 |
\f | 换页(FF),将当前位置移到下页开头 | 012 |
\n | 换行(LF) ,将当前位置移到下一行开头 | 010 |
\r | 回车(CR) ,将当前位置移到本行开头 | 013 |
\t | 水平制表(HT) (跳到下一个TAB位置) | 009 |
\v | 垂直制表(VT) | 011 |
\ | 代表一个反斜线字符"" | 092 |
’ | 代表一个单引号(撇号)字符 | 039 |
" | 代表一个双引号字符 | 034 |
? | 代表一个问号 | 063 |
\0 | 数字0 | 000 |
\ddd | 8进制转义字符,d范围0~7 | 3位8进制 |
\xhh | 16进制转义字符,h范围09,af,A~F | 3位16进制 |
示例:
int main() {
cout << "\\" << endl;
cout << "\tHello" << endl;
cout << "\n" << endl;
system("pause");
return 0;
}
2.6字符串型
作用:用于表示一串字符
两种风格
- C风格字符串:
char 变量名[] = "字符串值"
示例:
#include<iostream>
using namespace std;
int main(){
char str1[] = "hello world";
cout<<str1<<endl;
system("pause");
return 0;
}
- C++风格字符串:
string 变量名 = "字符串值"
示例:
#include<iostream>
#include<string>
using namespace std;
int main(){
string str = "hello world";
cout<<str<<endl;
system("pause");
return 0;
}
注意:C++风格字符串,需要加入头文件#include
2.7布尔类型 bool
作用:布尔数据类型代表真或假的值
bool类型只有两个值:
- true — 真(本质是1)
- false — 假(本质是0)
bool类型占1个字节大小
示例:
#include<iostream>
using namespace std;
int main(){
bool flag = true;
cout<<"flag = "<<flag<<endl; //1
flag = false;
cout<<"flag = "<<flag<<endl; //0
cout<<"size of bool = "<<sizeof(bool) <<endl; //1
system("pause");
return 0;
}
2.8数据的输入
作用:用于从键盘获取数据
关键字:cin
语法: cin >> 变量
示例:
#include<iostream>
#include<stdlib.h>
using namespace std;
int main(){
// 整型输入
int a = 0;
cout<<"请输入整型变量"<<endl;
cin>>a;
cout<<a<<endl;
//浮点型输入
float f = 0;
cout<<"请输入浮点型变量"<<endl;
cin>>f;
cout<<f<<endl;
//字符型输出
char c = 0;
cout<<"请输入字符:"<<endl;
cin>>c;
cout<<c<<endl;
//字符串输出
string str = "";
cout<<"请输入字符串"<<endl;
cin>>str;
cout<<str<<endl;
//布尔类型输出
bool flag =true;
cout<<"请输入布尔变量"<<endl;
cin>>flag;
cout<<flag<<endl;
system("pause");
return EXIT_SUCCESS;
}
stdlib.h头文件中 定义了两个变量:
#define EXIT_SUCCESS 0
#define EXIT_FAILURE 1
1、C++bool类型全局变量,如果不赋初始值的话,默认为false;
#include<iostream>
#include<stdlib.h>
using namespace std;
int main(){
//布尔类型输出
bool flag ;
cout<<flag<<endl; //0
system("pause");
return EXIT_SUCCESS;
}
2、如果bool类型变量使用cin等等进行赋值true和false的话,编译器会把true和false当成字符串,最终导致被赋值的变量输出结果为0(false);
#include<iostream>
#include<stdlib.h>
using namespace std;
int main(){
//布尔类型输出
bool flag ;
cin>>flag; //true
cout<<flag<<endl; //0
system("pause");
return EXIT_SUCCESS;
}
3、我们一般定义bool变量的时候都会赋初始值,并且再之后给bool变量赋值的时候我们一般都是直接赋值,比如aaa=true;而不是cin>>aaa;如果使用cin赋值bool的话,我们一般输入0(false)或者1(true);
#include<iostream>
#include<stdlib.h>
using namespace std;
int main(){
//布尔类型输出
bool flag ;
cin>>flag; //1
cout<<flag<<endl; //1
system("pause");
return EXIT_SUCCESS;
}
4、true、false、字符串、0这些内容在控制台给一个bool类型变量赋值时,返回结果都为false(0);
任意除0以外的数字,在控制台给bool类型变量赋值时,返回结果都为true(1);
#include<iostream>
#include<stdlib.h>
using namespace std;
int main(){
//布尔类型输出
bool flag ;
cin>>flag; //2
cout<<flag<<endl; //1
system("pause");
return EXIT_SUCCESS;
}
3、运算符
作用:用于执行代码的运算
本章我们主要讲解以下几类运算符:
运算符类型 | 作用 |
---|---|
算术运算符 | 用于处理四则运算 |
赋值运算符 | 用于将表达式的值赋给变量 |
比较运算符 | 用于表达式的比较,并返回一个真值或假值 |
逻辑运算符 | 用于根据表达式的值返回真值或假值 |
3.1算术运算符
作用:用于处理四则运算
算术运算符包括以下符号:
运算符 | 术语 | 示例 | 结果 |
---|---|---|---|
+ | 正号 | +3 | 3 |
- | 负号 | -3 | -3 |
+ | 加 | 10 + 5 | 15 |
- | 减 | 10 - 5 | 5 |
* | 乘 | 10 * 5 | 50 |
/ | 除 | 10 / 5 | 2 |
% | 取模(取余) | 10 % 3 | 1 |
++ | 前置递增 | a=2; b=++a; | a=3; b=3; |
++ | 后置递增 | a=2; b=a++; | a=3; b=2; |
– | 前置递减 | a=2; b=–a; | a=1; b=1; |
– | 后置递减 | a=2; b=a–; | a=1; b=2; |
示例1:
#include<iostream>
using namespace std;
int main(){
int a1 = 10;
int b1 = 3;
cout<<a1 + b1<<endl;
cout<<a1 - b1<<endl;
cout<<a1 * b1<<endl;
cout<<a1 / b1<<endl;
int a2 = 10;
int b2 = 20;
cout<<a2/b2<<endl;
int a3 = 10;
int b3 = 0;
// cout<<a3/b3<<endl; 除0异常
double d1 = 0.5;
double d2 = 0.25;
cout<<d1/d2<<endl;
system("pause");
return 0;
}
示例2:
#include <iostream>
using namespace std;
int main(){
int a1 = 10;
int b1 = 3;
cout<<10%3<<endl; //1
int a2 = 10;
int b2 = 20;
cout<<a2%b2<<endl; //10
int a3 = 10;
int b3 = 0;
// cout<<a3%b3<<endl; //取模的时候也能为0
double d1 = 3.14;
double d2 = 1.1;
// cout<<d1%d2<<endl; //两个小数不可以取模
system("pause");
return 0;
}
总结:只有整型变量可以进行取模运算
示例3:
#include<iostream>
using namespace std;
int main(){
int a = 1;
a++;
cout<<"a = "<<a<<endl;
int b = 2;
--b;
cout<<"b = "<<b<<endl;
}
总结:前置递增先对变量进行++,再计算表达式,后置递增相反
3.2赋值运算符
作用:用于将表达式的值赋给变量
赋值运算符包括以下几个符号:
运算符 | 术语 | 示例 | 结果 |
---|---|---|---|
= | 赋值 | a=2; b=3; | a=2; b=3; |
+= | 加等于 | a=0; a+=2; | a=2; |
-= | 减等于 | a=5; a-=3; | a=2; |
*= | 乘等于 | a=2; a*=2; | a=4; |
/= | 除等于 | a=4; a/=2; | a=2; |
%= | 模等于 | a=3; a%2; | a=1; |
示例:
#include<iostream>
using namespace std;
int main(){
//赋值运算符
//=
int a = 10;
a = 100;
cout<<"a = "<<a<<endl;
//+=
int b = 10;
b += 2; //b = b+2;
cout<<"b = "<<b<<endl;
//-=
int c = 10;
c -= 10;//c = c-10;
cout<<"c = "<<c<<endl;
//*=
int d = 10;
d *=2; //d = d*2
cout<<"d = "<<d<<endl;
// /=
int e = 10;
e /=2; //e = e/2;
cout<<"e = "<<e<<endl;
// %=
int f = 10;
f %= 3;// f = f%3;
cout<<"f = "<<f<<endl;
system("pause");
return 0;
}
3.3比较运算符
作用:用于表达式的比较,并返回一个真值或假值
比较运算符有以下符号:
运算符 | 术语 | 示例 | 结果 |
---|---|---|---|
== | 相等于 | 4 == 3 | 0 |
!= | 不等于 | 4 != 3 | 1 |
< | 小于 | 4 < 3 | 0 |
> | 大于 | 4 > 3 | 1 |
<= | 小于等于 | 4 <= 3 | 0 |
>= | 大于等于 | 4 >= 1 | 1 |
示例:
#include<iostream>
using namespace std;
int main(){
int a = 10;
int b = 20;
cout<< (a==b)<<endl; //0
cout<< (a!=b)<<endl; //1
cout<< (a<b)<<endl; //1
cout<<(a>b)<<endl; //0
cout<<(a<b)<<endl; //1
cout<<(a<=b)<<endl; //1
cout<<(a>=b)<<endl; //0
}
注意:C和C++ 语言的比较运算中, “真”用数字“1”来表示, “假”用数字“0”来表示。
3.4逻辑运算符
作用:用于根据表达式的值返回真值或假值
逻辑运算符有以下符号:
运算符 | 术语 | 示例 | 结果 |
---|---|---|---|
! | 非 | !a | 如果a为假,则!a为真; 如果a为真,则!a为假。 |
&& | 与 | a && b | 如果a和b都为真,则结果为真,否则为假。 |
|| | 或 | a || b | 如果a和b有一个为真,则结果为真,二者都为假时,结果为假。 |
示例1:逻辑非
#include<iostream>
using namespace std;
int main(){
int a = 10;
cout<<!a<<endl; //0
cout<<!!a<<endl; //1
system("pause");
return 0;
}
任意除0以外的数字,返回结果都为true(1);
示例2:逻辑与
#include<iostream>
using namespace std;
int main(){
int a = 10;
int b =10;
cout<<(a&&b)<<endl; //1
a = 10;
b = 0;
cout<<(a&&b)<<endl; //0
system("pause");
return 0;
}
总结:逻辑与运算符总结: 同真为真,其余为假
示例3:逻辑或
#include<iostream>
using namespace std;
int main(){
int a = 10;
int b = 20;
cout<<(a||b)<<endl; //1
a = 0;
b = 10;
cout<<(a||b)<<endl;//1
a = 0;
b = 0;
cout<<(a||b)<<endl; //0
system("pause");
return 0;
}
逻辑或运算符总结: 同假为假,其余为真
4、程序流程结构
C/C++支持最基本的三种程序运行结构:顺序结构、选择结构、循环结构
- 顺序结构:程序按顺序执行,不发生跳转
- 选择结构:依据条件是否满足,有选择的执行相应功能
- 循环结构:依据条件是否满足,循环多次执行某段代码
4.1选择结构
4.1.1 if语句
作用:执行满足条件的语句
if语句的三种形式
- 单行格式if语句
- 多行格式if语句
- 多条件的if语句
-
单行格式if语句:
if(条件){ 条件满足执行的语句 }
示例:
#include<iostream>
using namespace std;
int main(){
int score = 0;
cout<<"请输入您的分数:"<<endl;
cin>>score;
if(score>=514){
cout<<"你的分数超过一本分数线"<<endl;
}
system("pause");
return 0;
}
- 多行格式if语句:
if(条件){ 条件满足执行的语句 }else{ 条件不满足执行的语句 };
示例:
#include<iostream>
using namespace std;
int main(){
int score = 0;
cout<<"请输入您的分数:"<<endl;
cin>>score;
if(score>=514){
cout<<"你的分数超过一本分数线"<<endl;
}
else{
cout<<"继续加油"<<endl;
}
system("pause");
return 0;
}
- 多条件的if语句:
if(条件1){ 条件1满足执行的语句 }else if(条件2){条件2满足执行的语句}... else{ 都不满足执行的语句}
#include<iostream>
using namespace std;
int main(){
int score = 0;
cout<<"请输入您的分数:"<<endl;
cin>>score;
if(score>=514){
cout<<"你的分数超过一本分数线"<<endl;
}else if(score>400){
cout<<"你过了二本线"<<endl;
}else{
cout<<"继续加油"<<endl;
}
system("pause");
return 0;
}
嵌套if语句:在if语句中,可以嵌套使用if语句,达到更精确的条件判断
案例需求:
- 提示用户输入一个高考考试分数,根据分数做如下判断
- 分数如果大于600分视为考上一本,大于500分考上二本,大于400考上三本,其余视为未考上本科;
- 在一本分数中,如果大于700分,考入北大,大于650分,考入清华,大于600考入人大。
示例:
#include<iostream>
using namespace std;
int main(){
int score = 0;
cout<<"请输入你的高考考试分数"<<endl;
cin>>score;
if(score>600){
cout<<"你考上了一本"<<endl;
if(score>700){
cout<<"你考上了北大"<<endl;
}else if(score > 650){
cout<<"你考上了清华"<<endl;
}else{
cout<<"你考入了人大"<<endl;
}
}else if(score>500){
cout<<"你考上了二本"<<endl;
}else if(score>400){
cout<<"你考上了三本"<<endl;
}else{
cout<<"你没有考上本科"<<endl;
}
}
4.1.2三目运算符
作用: 通过三目运算符实现简单的判断
语法:表达式1 ? 表达式2 :表达式3
解释:
如果表达式1的值为真,执行表达式2,并返回表达式2的结果;
如果表达式1的值为假,执行表达式3,并返回表达式3的结果。
示例:
#include<iostream>
using namespace std;
int main(){
int a = 1;
int b = 2;
int c = 3;
c = a<b ? b:a;
cout<<"c = "<<c<<endl;
(a > b ? a : b) = 100;
cout << "a = " << a << endl;//a = 1
cout << "b = " << b << endl; //b = 100
cout << "c = " << c << endl; //c = 2
system("pause");
return 0;
}
总结:和if语句比较,三目运算符优点是短小整洁,缺点是如果用嵌套,结构不清晰
4.1.3switch语句
作用:执行多条件分支语句
语法:
switch(表达式)
{
case 结果1:执行语句;break;
case 结果2:执行语句;break;
...
default:执行语句;break;
}
示例:
#include<iostream>
using namespace std;
int main(){
int score = 0;
cout<<"请输入你对这个电影的评分:"<<endl;
cin>>score;
switch(score){
case 10:
case 9:
cout<<"经典"<<endl;
break;
case 8:
case 7:
cout<<"非常好"<<endl;
break;
case 6:
cout<<"一般"<<endl;
break;
default:
cout<<"烂片"<<endl;
break;
}
system("pause");
return 0;
}
//请给电影评分
//10 ~ 9 经典
// 8 ~ 7 非常好
// 6 ~ 5 一般
// 5分以下 烂片
注意1:switch语句中表达式类型只能是整型或者字符型
注意2:case里如果没有break,那么程序会一直向下执行
总结:与if语句比,对于多条件判断时,switch的结构清晰,执行效率高,缺点是switch不可以判断区间
4.2循环结构
4.2.1 while循环语句
作用:满足循环条件,执行循环语句
语法: while(循环条件){ 循环语句 }
解释:只要循环条件的结果为真,就执行循环语句
示例:
#include<iostream>
using namespace std;
int main(){
int num = 0;
while(num<10){
cout<<"num = "<<num<<endl;
num++;
}
system("pause");
return 0;
}
注意:在执行循环语句时候,程序必须提供跳出循环的出口,否则出现死循环
4.2.2 do…while循环语句
作用: 满足循环条件,执行循环语句
语法: do{ 循环语句 } while(循环条件);
注意:与while的区别在于do…while会先执行一次循环语句,再判断循环条件
示例:
int main() {
int num = 0;
do
{
cout << num << endl;
num++;
} while (num < 10);
system("pause");
return 0;
}
特别注意while后面有一个;
总结:与while循环区别在于,do…while先执行一次循环语句,再判断循环条件
4.2.3 for循环语句
作用: 满足循环条件,执行循环语句
语法: for(起始表达式;条件表达式;末尾循环体) { 循环语句; }
示例:
#include<iostream>
using namespace std;
int main(){
for(int i = 0;i<10;i++){
cout<<i<<endl;
}
system("pause");
return 0;
}
详解:
注意:for循环中的表达式,要用分号进行分隔
总结:while , do…while, for都是开发中常用的循环语句,for循环结构比较清晰,比较常用
4.3 跳转语句
4.3.1 break语句
作用: 用于跳出选择结构或者循环结构
break使用的时机:
- 出现在switch条件语句中,作用是终止case并跳出switch
- 出现在循环语句中,作用是跳出当前的循环语句
- 出现在嵌套循环中,跳出最近的内层循环语句
示例1:
#include<iostream>
using namespace std;
int main(){
//1.在switch 语句中使用break
cout<<"请输入你所选择的游戏难度:"<<endl;
cout<<"1、简单"<<endl;
cout<<"2、普通"<<endl;
cout<<"3、困难"<<endl;
int num = 0;
cin>>num;
switch(num){
case 1:
cout<<"你选择的是简单难度"<<endl;
break;
case 2:
cout<<"你选择的是普通难度"<<endl;
break;
case 3:
cout<<"你选择的是困难难度"<<endl;
break;
}
system("pause");
return 0;
}
示例2:
#include<iostream>
using namespace std;
int main(){
//2.在循环语句中用break
for(int i = 0;i < 10;i++){
if(i == 5){
break; //跳出循环语句
}
cout<<i<<endl; //输出0 1 2 3 4
}
system("pause");
return 0;
}
注意这里的 if(i == 5)判断是两个等于号
示例3:
#include<iostream>
using namespace std;
int main() {
//在嵌套循环语句中使用break,退出内层循环
for (int i = 0; i < 10; i++)
{
for (int j = 0; j < 10; j++)
{
if (j == 5)
{
break;
}
cout << "*" << " ";
}
cout << endl;
}
system("pause");
return 0;
}
4.3.2 continue语句
作用:在循环语句中,跳过本次循环中余下尚未执行的语句,继续执行下一次循环
示例:
#include<iostream>
using namespace std;
int main(){
//实现效果只输出奇数
for(int i = 0;i<100;i++){
if(i%2 == 0){
continue;
}
cout<<i<<endl;
}
system("pause");
return 0;
}
注意:continue并没有使整个循环终止,而break会跳出循环
4.3.3goto语句
作用:可以无条件跳转语句
语法: goto 标记;
解释:如果标记的名称存在,执行到goto语句时,会跳转到标记的位置
示例:
#include<iostream>
using namespace std;
int main(){
cout<<"1"<<endl;
goto FlAG;
cout<<"2"<<endl;
cout<<"3"<<endl;
cout<<"4"<<endl;
FlAG:
cout<<"5"<<endl;
return 0;
}
注意:在程序中不建议使用goto语句,以免造成程序流程混乱
5、数组
5.1概述
所谓数组,就是一个集合,里面存放了相同类型的数据元素
特点1:数组中的每个数据元素都是相同的数据类型
特点2:数组是由连续的内存位置组成的
5.2一维数组
5.2.1 一维数组定义方式
一维数组定义的三种方式:
数据类型 数组名[ 数组长度 ];
数据类型 数组名[ 数组长度 ] = { 值1,值2 ...};
数据类型 数组名[ ] = { 值1,值2 ...};
示例
#include<iostream>
using namespace std;
int main(){
//定义方式1
//数据类型 数据名[元素个数]
int score[10];
//利用下标赋值
score[0] = 100;
score[1] = 99;
score[2] = 98;
//利用下标输出
cout<< score[0]<<endl;
cout<< score[1]<<endl;
cout<< score[2]<<endl;
//第二种定义方式
//数据类型 数组名[元素个数] = {值1,值2,值3...}
//如果{}的不够10个数据,剩余数据用0补全
int score2[10] ={100,90,80,70,60,50,40,30,20,10};
//循环输出
for(int i = 0;i<10;i++){
cout<<score2[i]<<endl;
}
//定义方式3
//数据类型 数组名[] = {值1,值2,值3,...}
int score3[] = {100,90,80,70,60,50,40,30,20,10};
for(int i = 0;i<10;i++){
cout<<score3[i]<<endl;
}
system("pause");
return 0;
}
5.2.2 一维数组数组名
一维数组名称的用途:
- 可以统计整个数组在内存中的长度
- 可以获取数组在内存中的首地址
示例:
#include<iostream>
using namespace std;
int main(){
//数组名用途
//1、可以获取整个数组占用内存大小
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
cout<<"整个数组所占内存空间为:"<<sizeof(arr)<<endl;
cout<<"每个元素所占内存空间为: "<<sizeof(arr[0])<<endl;
cout<<"数组元素的个数为:"<<sizeof(arr)/sizeof(arr[0])<<endl;
//2、可以通过数组名获取到数组首地址
cout<<"数组首地址为:"<<(long long)arr<<endl;
cout<<"数组中第一个元素地址为:"<<(long long)&arr[0]<<endl;
cout<<"数组中第二个元素地址为:"<<(long long)&arr[1]<<endl;
//arr = 100; 错误,数组名是常量,因此不可以赋值
system("pause");
return 0;
}
注意:数组名是常量,不可以赋值
总结1:直接打印数组名,可以查看数组所占内存的首地址
总结2:对数组名进行sizeof,可以获取整个数组占内存空间的大小
error: cast from ‘int*’ to ‘int’ loses precision [-fpermissive]
目的是将16进制的地址,转化为10进制的
问题:因为基于Linux内核的64位系统上指针类型占用8个字节,而int类型占用4个字节,所以会出现loses precision。但亲身遇到的问题并不是在linux而是win,或许编译内核是linux,才疏学浅,尚未理解。
解决:可以先将int* 转成long类型,long类型可以隐式类型转换到int类型。直接修改为long long即可,而单纯long类型是不可以的,依然不理解为什么。
5.2.3 冒泡排序
作用: 最常用的排序算法,对数组内元素进行排序
- 比较相邻的元素。如果第一个比第二个大,就交换他们两个。
- 对每一对相邻元素做同样的工作,执行完毕后,找到第一个最大值。
- 重复以上的步骤,每次比较次数-1,直到不需要比较
示例: 将数组 { 4,2,8,0,5,7,1,3,9 } 进行升序排序
#include<iostream>
using namespace std;
int main(){
int arr[9] = {4,1,2,7,8,9,3,5,6};
for(int i = 0;i<9-1;i++){
for(int j = 0;j<9-1-i;j++){
if(arr[j]>arr[j+1]){
int temp =arr[j];
arr[j] = arr[j+1];
arr[j+1] = temp;
}
}
}
for(int i = 0;i<9;i++){
cout<<arr[i]<<endl;
}
system("pause");
return 0;
}
5.3 二维数组
二维数组就是在一维数组上,多加一个维度。
5.3.1 二维数组定义方式
二维数组定义的四种方式:
数据类型 数组名[ 行数 ][ 列数 ];
数据类型 数组名[ 行数 ][ 列数 ] = { {数据1,数据2 } ,{数据3,数据4 } };
数据类型 数组名[ 行数 ][ 列数 ] = { 数据1,数据2,数据3,数据4};
数据类型 数组名[ ][ 列数 ] = { 数据1,数据2,数据3,数据4};
建议:以上4种定义方式,利用第二种更加直观,提高代码的可读性
示例:
#include<iostream>
using namespace std;
int main(){
// 数据类型 数组名[ 行数 ][ 列数 ];
int arr1[2][3];
arr1[0][0]=1;
arr1[0][1]=2;
arr1[0][2]=3;
arr1[1][0]=4;
arr1[1][1]=5;
arr1[1][2]=6;
for(int i = 0;i < 2;i++){
for(int j = 0;j < 3;j++ ){
cout<<arr1[i][j]<<" ";
}
cout<<endl;
}
// 数据类型 数组名[ 行数 ][ 列数 ] = { {数据1,数据2 } ,{数据3,数据4 } };
int arr2[2][3]={{1,2,3},{4,5,6}};
for(int i = 0;i < 2;i++){
for(int j = 0;j < 3;j++ ){
cout<<arr2[i][j]<<" ";
}
cout<<endl;
}
// 数据类型 数组名[ 行数 ][ 列数 ] = { 数据1,数据2,数据3,数据4};
int arr3[2][3]={1,2,3,4,5,6};
for(int i = 0;i < 2;i++){
for(int j = 0;j < 3;j++ ){
cout<<arr3[i][j]<<" ";
}
cout<<endl;
}
// 数据类型 数组名[ ][ 列数 ] = { 数据1,数据2,数据3,数据4};
int arr4[][3]={1,2,3,4};
for(int i = 0;i < 2;i++){
for(int j = 0;j < 3;j++ ){
cout<<arr4[i][j]<<" "; //123 400
}
cout<<endl;//列不够的用0填充
}
}
总结:在定义二维数组时,如果初始化了数据,可以省略行数
二维数组数组名
- 查看二维数组所占内存空间
- 获取二维数组首地址
示例:
#include<iostream>
using namespace std;
int main(){
int arr[2][3] = {
{1,2,3},
{4,5,6}
};
cout<<"二维数组大小"<<sizeof(arr)<<endl;
cout<<"二维数组一行大小"<<sizeof(arr[0])<<endl;
cout<<"二维数组元素大小"<<sizeof(arr[0][0])<<endl;
cout<<"二维数组行数:"<<sizeof(arr)/sizeof(arr[0])<<endl;
cout<<"二维数组列数:"<<sizeof(arr[0])/sizeof(arr[0][0])<<endl;
//地址
cout<<"二维数组首地址:"<<arr<<endl;
cout<<"二维数组第一行地址"<<arr[0]<<endl;
cout<<"二维数组第二行地址"<<arr[1]<<endl;
cout<<"二维数组第一列地址"<<&arr[0][0]<<endl;
cout<<"二维数组第二列地址"<<&arr[0][1]<<endl;
system("pause");
return 0;
}
总结1:二维数组名就是这个数组的首地址
总结2:对二维数组名进行sizeof时,可以获取整个二维数组占用的内存空间大小
6、函数
6.1概述
作用:将一段经常使用的代码封装起来,减少重复代码
一个较大的程序,一般分为若干个程序块,每个模块实现特定的功能。
6.2函数的定义
函数的定义一般主要有5个步骤:
1、返回值类型
2、函数名
3、参数表列
4、函数体语句
5、return 表达式
语法:
返回值类型 函数名 (参数列表)
{
函数体语句
return表达式
}
- 返回值类型 :一个函数可以返回一个值。在函数定义中
- 函数名:给函数起个名称
- 参数列表:使用该函数时,传入的数据
- 函数体语句:花括号内的代码,函数内需要执行的语句
- return表达式: 和返回值类型挂钩,函数执行完后,返回相应的数据
示例:定义一个加法函数,实现两个数相加
//函数定义
int add(int num1, int num2)
{
int sum = num1 + num2;
return sum;
}
6.3 函数的调用
功能:使用定义好的函数
语法: 函数名(参数)
示例:
//函数定义
int add(int num1, int num2) //定义中的num1,num2称为形式参数,简称形参
{
int sum = num1 + num2;
return sum;
}
int main() {
int a = 10;
int b = 10;
//调用add函数
int sum = add(a, b);//调用时的a,b称为实际参数,简称实参
cout << "sum = " << sum << endl;
a = 100;
b = 100;
sum = add(a, b);
cout << "sum = " << sum << endl;
system("pause");
return 0;
}
总结:函数定义里小括号内称为形参,函数调用时传入的参数称为实参
6.4 值传递
- 所谓值传递,就是函数调用时实参将数值传入给形参
- 值传递时,如果形参发生,并不会影响实参
示例:
#include<iostream>
using namespace std;
void swap(int a,int b){
cout<<"交换前"<<endl;
cout<<"a"<<a<<endl;
cout<<"b"<<a<<endl;
int temp = a;
a = b;
b = temp;
cout<<"交换后"<<endl;
cout<<"a"<<a<<endl;
cout<<"b"<<b<<endl;
}
int main(){
int a = 10;
int b = 20;
swap(a,b);
cout<<"main函数中的a:"<<a<<endl; //10
cout<<"main函数中的b:"<<b<<endl; //20
system("pause");
return 0;
}
总结: 值传递时,形参是修饰不了实参的
6.5函数的常见样式
常见的函数样式有4种
- 无参无返
- 有参无返
- 无参有返
- 有参有返
示例:
#include<iostream>
using namespace std;
//无参无返
void fun1(){
cout<<"这是一个无参无返的函数"<<endl;
}
//有参无返
void fun2(int num){
num++;
cout<<"这是一个有参无返的函数"<<endl;
cout<<"num = "<<num<<endl;
}
//无参有返
int fun3(){
cout<<"这是一个无参有返的函数"<<endl;
return 10;
}
//有参有返
int fun4(int num){
num++;
cout<<"这是一个有参有返的函数"<<endl;
return num;
}
int main(){
fun1();
fun2(1);
fun3();
int num1 = fun3();
cout<<num1<<"这是fun3返回值"<<endl;
int num2 = fun4(1);
cout<<num2<<"这是fun4返回值"<<endl;
}
6.6 函数的声明
作用: 告诉编译器函数名称及如何调用函数。函数的实际主体可以单独定义。
- 函数的声明可以多次,但是函数的定义只能有一次
示例:
#include<iostream>
using namespace std;
//函数的声明可以有多个,但是函数的定义只能有一次
//声明
int max(int a,int b);
int max(int a,int b);
//定义
int max(int a,int b){
return a>b ? a:b;
}
int main(){
int a = 100;
int b = 200;
cout<<max(a,b)<<endl;
system("pause");
return 0;
}
6.7 函数的分文件编写
作用:让代码结构更加清晰
函数分文件编写一般有4个步骤
- 创建后缀名为.h的头文件
- 创建后缀名为.cpp的源文件
- 在头文件中写函数的声明
- 在源文件中写函数的定义
示例:
//swap.h文件
#include<iostream>
using namespace std;
//实现两个数字交换的函数声明
void swap(int a,int b);
#include<iostream>
using namespace std;
#include "swap.h"
void swap(int a,int b){
int temp = a;
a = b;
b = temp;
cout<<"a = "<<a<<endl;
cout<<"b = "<<b<<endl;
}
//main函数文件
#include "swap.h"
#include "swap.cpp"
#include<iostream>
using namespace std;
int main(){
int a = 100;
int b = 200;
swap(a,b);
system("pause");
return 0;
}
7、指针
7.1 指针的基本概念
指针的作用: 可以通过指针间接访问内存
- 内存编号是从0开始记录的,一般用十六进制数字表示
- 可以利用指针变量保存地址
7.2 指针变量的定义和使用
指针变量定义语法: 数据类型 * 变量名;
示例:
#include<iostream>
using namespace std;
int main(){
//1、指针的定义
int a = 10;
// 指针定义语法:数据类型 * 变量名
int *p;
//指针变量赋值
p = &a;//指针指向变量a的地址
cout<< &a<<endl; //打印数据a的地址
cout<< p <<endl; //打印指针变量p
//2.指针的使用
//通过*操作指针变量指向内存
cout<<"*p = " <<*p<<endl;
system("pause");
return 0;
}
指针变量和普通变量的区别
- 普通变量存放的是数据,指针变量存放的是地址
- 指针变量可以通过" * "操作符,操作指针变量指向的内存空间,这个过程称为解引用
总结1: 我们可以通过 & 符号 获取变量的地址
总结2:利用指针可以记录地址
总结3:对指针变量解引用,可以操作指针指向的内存
7.3 指针所占内存空间
提问:指针也是种数据类型,那么这种数据类型占用多少内存空间?
示例:
#include<iostream>
using namespace std;
int main(){
int a = 10;
int *p;
p = &a;//指针指向数据a的地址
cout<<*p<<endl;// *解引用
cout<<sizeof(p)<<endl;
cout<<sizeof(char *)<<endl;
cout<<sizeof(float *)<<endl;
cout<<sizeof(double *)<<endl;
system("pause");
return 0;
}
总结:所有指针类型在32位操作系统下是4个字节
64位操作系统下是8个字节
7.4 空指针和野指针
空指针:指针变量指向内存中编号为0的空间
用途:初始化指针变量
注意:空指针指向的内存是不可以访问的
示例1:空指针
#include<iostream>
using namespace std;
int main(){
//指针变量p指向内存地址编号为0的空间
int *p = NULL;
//访问空指针报错
//内存编号为0~255为系统占用内存,不允许用户访问
cout<<*p<<endl;
system("pause");
return 0;
}
野指针:指针变量指向非法的内存空间
示例2:野指针
#include<iostream>
using namespace std;
int main(){
//指针变量p指向内存地址编号为0x1100的空间
int *p = (int *)0x1100;
//访问野指针报错
cout<<*p<<endl;
system("pause");
return 0;
}
总结:空指针和野指针都不是我们申请的空间,因此不要访问。
7.5 const修饰指针
const修饰指针有三种情况
- const修饰指针 — 常量指针
- const修饰常量 — 指针常量
- const即修饰指针,又修饰常量
示例:
#include<iostream>
using namespace std;
int main(){
int a = 10;
int b = 10;
//const修饰的是指针,指针指向可以改,但是指针所指的值不可以更改
const int * p1 = &a;
p1 = &b; //正确
//*p1 = 100; 报错
//const修饰的是常量,指针指向不可以改,指针指向的值可以更改
int * const p2 = &a;
*p2 = 100; //正确
cout<<"p2 = "<<*p2<<endl; ///100
cout<<"a = "<<a<<endl; //100
a = 200;
cout<<"p2 = "<<*p2<<endl; //200
cout<<"a = "<<a<<endl; //200
//const 即修饰常量又修饰指针
const int * const p3 = &a;
// p3 = &b; //错误
// *p3 = 100; //错误
a = 1000;
cout<<"p3 = "<<*p3<<endl; ///1000
cout<<"a = "<<a<<endl; //1000
system("pause");
return 0;
}
技巧:看const右侧紧跟着的是指针还是常量, 是指针就是常量指针,是常量就是指针常量
7.6 指针和数组
作用:利用指针访问数组中元素
示例:
#include<iostream>
using namespace std;
int main(){
int arr[] = {1,2,3,4,5,6,7,8,9,10};
int *p = arr; //指向数组的指针
cout<<"第一个元素:"<<arr[0]<<endl;
cout<<"指针访问第一个元素:"<<*p<<endl;
for(int i = 0;i<10;i++){
//利用指针遍历数组
cout<<*p<<endl;
p++;
}
system("pause");
return 0;
}
7.7 指针和函数
作用:利用指针作函数参数,可以修改实参的值
示例:
#include<iostream>
using namespace std;
//值传递
void swap1(int a,int b){
int temp = a;
a = b;
b = temp;
}
//地址传递
void swap2(int *p1,int *p2){
int temp = *p1;
*p1 = *p2;
*p2 = temp;
}
int main(){
int a = 10;
int b = 20;
swap1(a,b); //值传递不会改变实参
cout<<"a = "<<a<<endl;
cout<<"b = "<<b<<endl;
swap2(&a,&b);//地址传递会改变实参
cout<<"a = "<<a<<endl;
cout<<"b = "<<b<<endl;
system("pause");
return 0;
}
总结:如果不想修改实参,就用值传递,如果想修改实参,就用地址传递
7.8 指针、数组、函数
案例描述:封装一个函数,利用冒泡排序,实现对整型数组的升序排序
例如数组:int arr[10] = { 4,3,6,9,1,2,10,8,7,5 };
示例:
#include<iostream>
using namespace std;
//冒泡排序函数
void bubbleSort(int *arr,int len)// int*arr也可以写为int arr[]
{
for(int i = 0;i<len-1;i++){
for(int j = 0;j<len-i-1;j++){
int temp = 0;
if(arr[j]>arr[j+1]){
temp = arr[j];
arr[j] = arr[j+1];
arr[j+1] = temp;
}
}
}
}
//打印数组函数
void printArray(int arr[],int len){
for(int i = 0;i<len;i++){
cout<<arr[i]<<endl;
}
}
int main(){
int arr[10] = {3,7,4,5,6,1,2,8,9,10};
int len = sizeof(arr) /sizeof(int); //C++没有专门求数组长度的内置函数
bubbleSort(arr,len);
printArray(arr,len);
system("pause");
return 0;
}
总结:当数组名传入到函数作为参数时,被退化为指向首元素的指针
8、结构体
8.1 结构体基本概念
结构体属于用户自定义的数据类型,允许用户存储不同的数据类型
8.2 结构体定义和使用
语法:struct 结构体名 { 结构体成员列表 };
通过结构体创建变量的方式有三种:
- struct 结构体名 变量名
- struct 结构体名 变量名 =
- 定义结构体时顺便创建变量
示例:
#include<iostream>
using namespace std;
//结构体定义
struct student{
//成员列表
string name;
int age;
int score;
}stu3;//定义结构体时顺便创建变量
int main(){
//struct 结构体名 变量名
struct student stu1;//struct关键字可以省略
stu1.name = "张三";
stu1.age = 21;
stu1.score = 500;
cout<<"姓名:"<<stu1.name<<" 年龄:"<<stu1.age<<" 分数:"<<stu1.score<<endl;
//struct 结构体名 变量名 = { 成员1值 , 成员2值…}
student stu2 = {"李四",22,499};
cout<<"姓名:"<<stu2.name<<" 年龄:"<<stu2.age<<" 分数:"<<stu2.score<<endl;
system("pause");
return 0;
}
总结1:定义结构体时的关键字是struct,不可省略
总结2:创建结构体变量时,关键字struct可以省略
总结3:结构体变量利用操作符 ‘’.‘’ 访问成员
8.3 结构体数组
作用:将自定义的结构体放入到数组中方便维护
语法: struct 结构体名 数组名[元素个数] = { {} , {} , ... {} }
示例:
#include<iostream>
using namespace std;
struct student {
string name;
int age;
int score;
};
int main(){
// struct 结构体名 数组名[元素个数] = { {} , {} , ... {} }
struct student arr[3] = {
{"tom",10,100},{"amy",11,100},{"jerry",10,100}
};
for(int i = 0;i<3;i++){
cout<<"姓名"<<arr[i].name<<" 年龄: "<<arr[i].age<<" 分数:"<<arr[i].score<<endl;
}
system("pause");
return 0;
}
8.4 结构体指针
作用:通过指针访问结构体中的成员
- 利用操作符
->
可以通过结构体指针访问结构体属性
#include<iostream>
using namespace std;
//结构体定义
struct student{
//成员列表
string name; //姓名
int age; //年龄
int score; //分数
};
int main(){
struct student stu = {
"张三",21,100
};
struct student *p = &stu;
p->score = 80; //指针通过操作符可以访问成员
cout<<"姓名:" <<p->name<<"年龄:"<<p->age<<"分数:"<<p->score<<endl;
system("pause");
return 0;
}
总结:结构体指针可以通过 -> 操作符 来访问结构体中的成员
8.5 结构体嵌套结构体
作用: 结构体中的成员可以是另一个结构体
例如:每个老师辅导一个学员,一个老师的结构体中,记录一个学生的结构体
示例:
#include<iostream>
using namespace std;
//学生结构体定义
struct student{
string name;
int age;
int score;
};
//教师结构体定义
struct teacher{
int id;
string name;
int age;
struct student stu;//子结构 学生
};
int main(){
struct teacher t1;
t1.id = 3000;
t1.name = "李华";
t1.age = 40;
t1.stu.name = "李四";
t1.stu.age = 20;
t1.stu.score = 100;
cout<<"教师 职工编号:"<<t1.id<<" 姓名: "<<t1.name<<" 年龄"<<t1.age<<endl;
cout<<"学生姓名:"<<t1.stu.name<<" 学生年龄:"<<t1.stu.age<<" 学生分数:"<<t1.stu.score<<endl;
system("pause");
return 0;
}
总结: 在结构体汇总可以定义另一个结构体作为成员,用来解决实际问题
8.6 结构体做函数参数
作用:将结构体作为参数向函数中传递
传递方式有两种:
- 值传递
- 地址传递
示例:
#include<iostream>
using namespace std;
struct student{
string name;
int age;
int score;
};
void printStudent1(student stu){
stu.age = 18;
cout<<"姓名:"<<stu.name<<" 年龄:"<<stu.age<<" 分数:" <<stu.score<<endl;
}
void printStudent2(student *stu){
stu->age = 21;
cout<<"姓名:"<<stu->name<<" 年龄:"<<stu->age<<" 分数:" <<stu->score<<endl;
}
int main(){
student stu = {"amy",99,100};
//值传递
printStudent1(stu);
cout<<"主函数中姓名:"<<stu.name<<" 年龄:"<<stu.age<<" 分数:" <<stu.score<<endl;
//地址传递
printStudent2(&stu);
cout<<"主函数中姓名:"<<stu.name<<" 年龄:"<<stu.age<<" 分数:" <<stu.score<<endl;
system("pause");
return 0;
}
总结:如果不想修改主函数中的数据,用值传递,反之用地址传递
8.7 结构体中 const使用场景
作用:用const来防止误操作
示例:
#include<iostream>
using namespace std;
struct student{
string name;
int age;
int score;
};
void printStudent(const student *stu){
// stu->age = 21; //操作失败,因为加了const修饰
cout<<"姓名:"<<stu->name<<" 年龄:"<<stu->age<<" 分数:" <<stu->score<<endl;
}
int main(){
student stu = {"amy",99,100};
printStudent(&stu);
system("pause");
return 0;
}