当前位置: 首页 > news >正文

C++基础(5) - 复合类型(上)

文章目录

  • 数组
    • 1、什么是数组
    • 2、数组的声明
    • 3、数组的初始化
    • 4、数组的访问
    • 5、二维数组
    • 6、`memset` —— 给数组中每一个元素赋同样的值
  • 字符串(字符数组)
    • 1、`string.h` 头文件
      • 1.1 `strlen()`
      • 1.2 `strcmp()`
      • 1.3 `strcpy()`
      • 1.4 `strcat()`
  • string 类简介
    • 1、C++11 字符串初始化
    • 2、赋值、拼接和附加
    • 3、`string` 头文件
      • 3.1 `string` 中内容的访问
      • 3.2 `string` 的比较
      • 3.3 `length()/size()`
      • 3.4 `insert()`
      • 3.5 `erase()`
      • 3.6 `clear()`
      • 3.7 `substr()`
      • 3.8 `string::npos`
      • 3.9 `find()`
      • 3.10 `replace()`
  • 结构体
    • 1、结构体的定义和使用
    • 2、结构体变量的初始化
    • 3、结构体数组
  • 共用体
    • 1、什么是共用体
    • 2、共用体的用途
  • 枚举
    • 1、枚举的定义
    • 2、枚举的使用
    • 3、设置枚举量的值
    • 4、枚举的取值范围

数组

1、什么是数组

数组是一种数据格式,能够存储多个同类型的值。

数组的特点:

  1. 数组中的元素按线性方式排列,可以通过编号来访问数组中的每个元素;
  2. 每个值都存储在一个独立的数组元素中,计算机在内存中依次存储数组的各个元素;

 

2、数组的声明

要创建数组,可使用声明语句。数组声明应指出以下三点:

  1. 存储在每个元素中的值的类型;
  2. 数组名;
  3. 数组中的元素个数;

格式为:数据类型 数组名[数组大小];

注意:数组大小必须是整数常量,不可以是变量。

 

3、数组的初始化

  • C++ 允许在声明语句中初始化数组元素,只需提供一个用逗号分隔的值列表(初始化列表),并将它们用花括号括起来,可省略等号(=);

    int arr[4] = {3, 6, 8, 10};
    int arr[4] {3, 6, 8, 10};	// C++11 新增
    
  • 只有在定义数组时才能使用初始化,此后就不能使用了,也不能将一个数组赋给另一个数组;

    int arr[4] = {3, 6, 8, 10};	// 正确int hand[4];
    hand[4] = {3, 6, 8, 10};	// 错误
    hand = arr;	// 错误
    
  • 初始化数组时,提供的值可以少于数组的元素数目。如果对数组的一部分进行初始化,则编译器将把其他元素设置成0。若想初始化数组为0,则只需要显式将第一个元素初始化为0即可;

    int arr[4] = {3, 6};	// 第1个元素3,第2个元素6,其他元素为0
    int arr[4] = {0};	// 所有元素初始化为0
    int arr[4] = {};	// C++11新增,所有元素初始化为0
    
  • 如果初始化数组时方括号内([])为空,C++ 编译器将计算元素个数;

    int arr[] = {3, 6, 8, 10};	// 编译器会自动识别元素个数为4
    
  • 列表初始化禁止缩窄转换;

    long plifs[] = {25, 92, 3.0};	// 错误,浮点数转换为整型是缩窄操作
    char slifs[4] = {'h', 'i', 1122011, '\0'};	// 错误,1122011超过了char变量的取值范围
    char tlifs[4] = {'h', 'i', 112, '\0'};	// 正确,虽然112是一个int值,但在char类型的取值范围
    

 

4、数组的访问

int arr[indx];

其中 indx 为数组的索引(下标),范围为 0 ~ length-1;数组长度 length = sizeof(arr) / sizeof(int)。这里的 sizeof 之前的文章介绍过,它返回变量或数据类型的字节个数。因为 arrint 型数组,里面每一个元素都是 int 类型,所以 sizeof(arr) 表示所有元素的字节数,sizeof(int) 表示一个元素的字节数,两者之商就是元素的个数。

注意:如果数组大小较大(大概 10610^6106 级别),则需要将其定义在主函数外面,否则会使程序异常退出。因为函数内部申请的局部变量来自系统栈,允许申请的空间较小;而函数外部申请的全局变量来自静态存储区,允许申请的空间较大。

 

5、二维数组

  1. 声明二维数组

数据类型 数组名[m][n];

m 表示二维数组有多少个一维数组,n 表示每个一维数组的元素个数。

二维数组

  1. 二维数组初始化

    创建二维数组时,可以初始化其所有元素,二维数组初始化是建立在一维数组初始化技术的基础之上:提供由逗号分隔的用花括号括起的值列表。

    int scores[3][4] = {{},{},{}
    };
    

    注意:二维数组的行数个数必须全部用大括号指定,不能省略,否则编译出错!【例如上面的 scores 数组,我省略三个大括号中的一个是不行的】

     

6、memset —— 给数组中每一个元素赋同样的值

给数组中每一个元素赋相同的值有两种方法:memsetfill

memset 函数的格式为:memset(数组名, 值, sizeof(数组名));。需要引入 string.h 头文件。

只建议初学者为数组赋 0-1。因为 memset 使用的是按字节赋值,并不是按元素赋值,很容易出错。如果想按元素赋值,可以使用 fill 函数,此时需要引入 algorithm 头文件。

 
 

字符串(字符数组)

  • 字符串是存储在内存连续字节中的一系列字符,又可以称为字符数组;

  • C++ 处理字符串有两种方式,一种是 C 语言风格,另一种是基于 string 类;

  • 可以将字符串存储在 char 数组中,每个字符都位于自己的数组元素中;

  • C 语言风格的字符串是以空字符结尾(空字符被写作 \0,ASCII 码为 0),用来标记字符串的结尾。处理字符串的函数都是根据空字符的位置,而不是数组长度来进行处理;

    存储字符串 “Kitty” 的情况为:char name[6] = {'K', 'i', 't', 't', 'y', '\0'};

  • 字符数组初始化为字符串,可以使用双引号("")将字符串内容括起来【仅限于初始化,程序其他位置不允许这样直接赋值整个字符串】;

    char name[16] = "Hello Kitty";
    char name[] = "Kitty";
    
  • 注意字符常量和字符串常量的区别;

    'S' 是字符常量,"S" 是字符串常量,存储的是 ['S', '\0']

 

1、string.h 头文件

1.1 strlen()

strlen(字符数组):获得字符数组中第一个 \0 之前的字符个数。

1.2 strcmp()

strcmp(字符数组1, 字符数组2):返回两个字符串大小的比较结果,比较规则是字典序。

  1. 字符数组1 < 字符数组2:返回一个负整数;
  2. 字符数组1 < 字符数组2:返回一个正整数;
  3. 字符数组1 < 字符数组2:返回0;

1.3 strcpy()

strcpy(字符数组1, 字符数组2):把字符数组2复制到字符数组1,覆盖复制,这里的复制还包括结束符 \0

1.4 strcat()

strcat(字符数组1, 字符数组2):把字符数组2拼接到字符数组1后面,两者之间的结束符会被覆盖。

 
 

string 类简介

除了使用字符数组之外,还可以使用 string 来存储字符串,而且 string 使用起来会比字符数组要简单。

要使用 string,必须包含头文件 string

 

1、C++11 字符串初始化

C++11 允许将列表初始化用于 C 风格字符串和 string 对象:

char first_data[] = {"Hello World"};
char second_data[] = {"The Elegant Plate"};
string third_data = {"Hello World"};
string third_data {"Hello World"};string str = {'h', 'e', 'l', 'l', 'o'};
string str = "hello world";

 

2、赋值、拼接和附加

  • 不能将一个数组直接赋给另一个数组,但是可以将一个 string 对象直接赋给另一个 string 对象;

    char ch[20] = "hello";
    string str = "world";char ch1[20];
    ch1 = ch;	// 错误!!
    string str1;
    str1 = str;	// 正确!!
    
  • string 简化了合并操作,可以使用运算符 + 来将两个 string 对象合并起来,还可以使用 += 运算符。

    string str2 = str + str2;
    string str += str2;
    

 

3、string 头文件

虽然 string 类也可以使用 C 中的头文件 string.h,但是 C++ 为了更好地使用 string,也提供了自己的 string 操作函数。

3.1 string 中内容的访问

  1. 直接像数组那样通过下标访问 string;

    string str = "abc";
    cout << str[0] << endl;	// 要读入或输出整个字符串,只能使用 cin 和 cout// 还可以用 c_str() 将 string 类型转换为字符数组,然后用 printf 输出
    printf("%s\n", str.c_str());
    
  2. 通过迭代器访问;

    string str = "abc";
    string::iterator it = str.begin();
    // 上一句为了简便,还可以写成:
    auto it = str.begin();
    

3.2 string 的比较

两个 string 类型可以直接使用 关系运算符 进行比较,比较规则是字典序!

3.3 length()/size()

str.length()/str.size():返回字符串 str 的长度,即存放的字符数;

3.4 insert()

  1. str.insert(pos, string):在字符串 strpos 号位置插入字符串 string
  2. str.insert(it, it1, it2)it 为原字符串欲插入的位置,it1it2 为待插字符串的首尾迭代器,用来表示子串 [it1, it2) 将被插入到 it 的位置上;

3.5 erase()

  1. str.erase(it):删除迭代器 it 所在位置的元素;
  2. str.erase(begin, end):删除一个区间 [begin, end) 内的所有元素;
  3. str.erase(pos, length)pos 为开始删除的其实位置,length 为删除的字符个数;

3.6 clear()

str.clear():清空 string 中的数据;

3.7 substr()

str.substr(pos, length):返回从 pos 号位开始、长度为 length 的子串;

3.8 string::npos

一个常数,其本身的值为 -1,但由于是 unsigned_int 类型,因此实际上也可以被认为是 unsigned_int 类型的最大值。string::npos 用来作为 find 函数失配时的返回值。

3.9 find()

  1. str.find(str1):当 str1str 的子串时,返回其在 str 中第一次出现的位置;如果不是,则返回 string::npos
  2. str.find(str1, pos):从 strpos 号位开始匹配 str1,返回值与上相同;

3.10 replace()

  1. str.replace(pos, length, str1):把 strpos 号位开始、长度为 length 的子串转换为 str1
  2. str.replace(it1, it2, str1):把 str 的迭代器 [it1, it2) 范围的子串替换为 str1

 
 

结构体

结构体是由一批不同类型数据组合而成的一种新的数据类型,组成结构体数据的每个数据称为结构体的“成员”(域、元素),通常用来表示类型不同但是又相关的若干数据。

 

1、结构体的定义和使用

struct 结构体名字 {char name[20];	// 一些基本的数据结构或者自定的数据类型int age;...
}[结构体变量1, 结构体变量2, ...];	// [] 中的是可选项

结构体里面可以定义除了自己之外的任何数据类型,因为定义自己本身会引起循环定义的问题,但是可以定义自身类型的指针变量。

  • 假设结构体名字为 student,依次创建结构体类型的变量:

    struct student stu, *stup;	// stu 为结构体变量,stup 为结构体指针变量,两者访问变量方式不同
    student stu;
    
  • 访问结构体类型变量成员:

    cout << stu.name;	// 普通变量的访问方式
    cout << stup->name;	// 指针变量的访问方式
    

 

2、结构体变量的初始化

使用逗号分隔值列表,并将这些值用花括号括起来,值之间用逗号分开:

student stu = {"cat", 19};
student stu {"dog", 18};
student stu {};

 

3、结构体数组

  • 可以创建元素为结构体的数组,方法和创建基本类型数组完全相同:

    student stus[20];
    cin >> stus[0].name;
    cout << stus[19].age << endl;
    
  • 结构体数组初始化:

    student stus[2] = {{"cat", 19},{"dog", 18}
    };
    

 
 

共用体

1、什么是共用体

共用体(联合体)是一种数据格式,它能够存储不同的数据类型,但只能同时存储其中的一种类型。例如一个共用体中定义了 intdouble 类型的变量,存储了 int 型数据之后,再存储 double 型数据,此时 int 型数据会丢失。

union one4two{int id_int;long id_long;};one4two pail;pail.id_int = 3;cout << pail.id_int << endl;	// 3pail.id_long = 3000;cout << pail.id_long << endl;	// 3000cout << pail.id_int << endl;	// 3000

共用体的句法与结构体相似,但是含义不同,共用体的长度为其最大成员的长度:

union one4all
{char char_val;short short_val;int int_val;	// one4all 的长度就是 int 类型的长度
};

 

2、共用体的用途

共用体的用途之一是,当数据项使用两种或更多种格式(但不会同时使用)时,可节省空间。

假设管理一个小商品目录,其中有一些商品的ID为整数,而另外一些的ID为字符串:

struct widget
{char brand[20];int type;union id{long id_num;char id_char[20];} id_val;
};
...
widget prize;
...
if (prize.type == 1) {cin >> prize.id_val.id_num;
} else {cin << prize.id_val.id_char;
}

匿名共用体(anonymous union)没有名称,其成员将成为位于相同地址处的变量。显然,每次只有一个成员是当前的成员:

struct widget
{char brand[20];int type;union{long id_num;char id_char[20];};
};
...
widget prize;
...
if (prize.type == 1) {cin >> prize.id_num;
} else {cin >> prize.id_char;
}

由于共用体是匿名的,因此 id_numid_char 被视为 prize 的两个成员,它们的地址相同,所以不需要用中间标识符 id_val 。程序员负责确定当前哪个成员是活动的。

共用体常用于(但并非只能用于)节省内存。

 
 

枚举

前文提到过符号常量,使用 const#define 进行定义。C++ 的枚举类型 enum 提供了另一种创建符号常量的方式,可以代替 const,它还允许定义新类型。

 

1、枚举的定义

enum 枚举名 {枚举量1, 枚举量2, ...};// 例子如下:red为0,依此类推,数值加1。即 orange 为1,yellow 为2,...
enum colors {red, orange, yellow, green, blue, purple};

 

2、枚举的使用

// 使用枚举名声明该类型的变量
colors band;// 只能将定义枚举时使用的枚举量赋值给这种枚举的变量
band = red;
// band = 5;	// 错误!!// 可以通过强制类型转换,将有效的 int 值转换成枚举量
band = colors(4);// 枚举量是整型,可被提升为 int 类型,但 int 类型不能自动转换为枚举类型
int col = blue;
col = 3 + red;

 

3、设置枚举量的值

  • 可以使用赋值运算符来显式地设置枚举量地值,指定地值必须是整数:

    enum bits{one = 1, two = 2, four = 4, eight = 8};
    
  • 也可以只显式地定义其中一些枚举量的值:

    enum bigstep{first, second = 100, third};	// first 为0,third 为101
    

    first 默认情况下为0,后面没有初始化的枚举量的值比其前面的枚举量大1,third 的值为 101

  • 可以创建多个值相同的枚举量:

    enum{zero, null = 0, one, numero_uno = 1};	// 这里 zero 也为0,one 也为1
    

     

4、枚举的取值范围

  • 每个枚举都有取值范围,通过强制类型转换,可以将取值范围中的任何整数赋值给枚举变量,即使这个值不是枚举值:

    enum bits{one = 1, two = 2, four = 4, eight = 8};
    bits myflag = bits(6);
    
  • 枚举取值范围:

    • 上限:寻找第一个大于枚举量最大值的2的幂次方,上限就是这个值减1;

      例如最大枚举量为 101,第一个大于它的是 27=1282^7 = 12827=128,所以上限就是 128−1=127128 - 1 = 1271281=127

    • 下限:枚举量最小值为0,则下限为0;否则寻找第一个小于它的2的幂次方,加上1;

      例如最小枚举量为 -6,第一小于它的是 −23=−8-2^3 = -823=8,因此下限为 −8+1=−7-8 + 1 = -78+1=7

http://www.lryc.cn/news/3726.html

相关文章:

  • java重写(@Override)介绍及实例说明
  • 基于STM32的虚拟示波器
  • 搭建云端vscode-server,使用web ide进行远程开发
  • Linux clock子系统及驱动实例
  • GIS数据格式坐标转换(地球坐标WGS84、GCJ-02、火星坐标、百度坐标BD-09、国家大地坐标系CGCS2000)
  • 流媒体传输系列文章汇总
  • “万字“ Java I/O流讲解
  • 数据库(Spring)事务的四种隔离级别
  • RabbitMQ详解(一):RabbitMQ相关概念
  • ​ICLR 2023 | GReTo:以同异配关系重新审视动态时空图聚合
  • 线程池分享总结
  • AOSP Android11系统源码和内核源码
  • layui框架学习(6:基础菜单)
  • 第十三届蓝桥杯 C++ B组省赛 C 题——刷题统计(AC)
  • C++中的多态
  • Swift如何保证线程安全
  • 整型提升+算术转换——“C”
  • Freemarker介绍
  • 【软件测试开发】Junit5单元测试框架
  • 【C语言技能树】程序环境和预处理
  • 数据库的三大范式
  • 【MT7628】开发环境搭建-Fedora12安装之后无法上网问题解决
  • [Android Studio]Android 数据存储-文件存储学习笔记-结合保存QQ账户与密码存储到指定文件中的演练
  • 【openGauss实战9】深度分析分区表
  • XSS跨站脚本攻击剖析与防御:初识XSS
  • Python 高级编程之网络编程 Socket(六)
  • centos学习记录
  • 为什么说网络安全是风口行业?
  • 12-PHP使用过的函数 111-120
  • 【JavaWeb项目】简单搭建一个前端的博客系统