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

《第四章-筋骨淬炼》 C++修炼生涯笔记(基础篇)数组与函数

数组与函数详解

一、数组

1. 一维数组

(1) 定义方式

一维数组是存储相同类型元素的线性集合,定义时需要指定数据类型数组长度

定义方式示例说明
数据类型 数组名[数组长度];int arr1[5];定义长度为5的整型数组,未初始化
数据类型 数组名[数组长度] = {值1, 值2, ...};int arr2[5] = {1,2,3,4,5};定义并初始化(全部元素)
数据类型 数组名[] = {值1, 值2, ...};int arr3[] = {1,2,3};定义并初始化,数组长度自动推断

特点

  • 数组元素在内存中连续存储
  • 数组索引从0开始
  • 数组长度必须是常量表达式
// 正确示例
const int SIZE = 5;
int arr[SIZE] = {10, 20, 30, 40, 50};// 错误示例:长度不能是变量
int n = 5;
int wrong_arr[n];  // 编译错误
(2) 数组名的意义

数组名代表数组的首地址,是一个常量指针(不能修改指向)。

int arr[5] = {1, 2, 3, 4, 5};cout << arr << endl;       // 输出数组首地址
cout << &arr[0] << endl;  // 输出第一个元素地址(与上面相同)
cout << sizeof(arr) << endl; // 输出整个数组大小(5*4=20字节)
(3) 元素逆置案例

将数组元素顺序颠倒的实现:

#include <iostream>
using namespace std;int main() {int arr[] = {1, 2, 3, 4, 5};int length = sizeof(arr) / sizeof(arr[0]); // 计算数组长度// 逆置算法:首尾交换for (int start = 0, end = length - 1; start < end; start++, end--) {int temp = arr[start];arr[start] = arr[end];arr[end] = temp;}// 输出结果:5 4 3 2 1for (int i = 0; i < length; i++) {cout << arr[i] << " ";}return 0;
}
(4) 一维数组冒泡排序⭐⭐⭐

冒泡排序是一种简单直观的排序算法:

#include <iostream>
using namespace std;int main() {int arr[] = {5, 3, 1, 4, 2};int length = sizeof(arr) / sizeof(arr[0]);// 冒泡排序(升序)for (int i = 0; i < length - 1; i++) {         // 外层循环:排序轮数for (int j = 0; j < length - 1 - i; j++) { // 内层循环:每轮比较次数if (arr[j] > arr[j+1]) {               // 前一个比后一个大,交换int temp = arr[j];arr[j] = arr[j+1];arr[j+1] = temp;}}}// 输出排序结果:1 2 3 4 5for (int i = 0; i < length; i++) {cout << arr[i] << " ";}return 0;
}

2. 二维数组

(1) 定义方式

二维数组可看作"数组的数组",常用于表示表格数据。

定义方式示例说明
数据类型 数组名[行数][列数];int arr1[2][3];定义2行3列的二维数组,未初始化
数据类型 数组名[行数][列数] = {{值1,值2,...}, {...}};int arr2[2][3] = {{1,2,3},{4,5,6}};初始化所有元素(推荐)
数据类型 数组名[行数][列数] = {值1,值2,...};int arr3[2][3] = {1,2,3,4,5,6};按顺序初始化(自动填充)
数据类型 数组名[][列数] = {值1,值2,...};int arr4[][3] = {1,2,3,4,5,6};行数自动推断(列数必须指定)

内存布局

行0: [1][2][3]
行1: [4][5][6]
(2) 数组名的意义

二维数组名代表整个二维数组的首地址,是一个指向一维数组的指针

int arr[2][3] = {{1,2,3},{4,5,6}};cout << arr << endl;        // 二维数组首地址
cout << arr[0] << endl;     // 第一行首地址
cout << arr[1] << endl;     // 第二行首地址
cout << &arr[0][0] << endl; // 第一个元素地址
(3) 考试成绩统计案例

统计3名学生4门课程的成绩:

#include <iostream>
#include <iomanip> // 用于格式化输出
using namespace std;int main() {const int STUDENTS = 3;const int COURSES = 4;int scores[STUDENTS][COURSES] = {{85, 90, 78, 92},{76, 88, 95, 80},{92, 85, 90, 88}};// 打印表头cout << "学号\t语文\t数学\t英语\t综合\t总分\t平均分" << endl;// 计算每个学生的总分和平均分for (int i = 0; i < STUDENTS; i++) {int total = 0;cout << i+1 << "\t";// 输出各科成绩并计算总分for (int j = 0; j < COURSES; j++) {cout << scores[i][j] << "\t";total += scores[i][j];}// 计算并输出总分和平均分double average = static_cast<double>(total) / COURSES;cout << total << "\t" << fixed << setprecision(1) << average << endl;}return 0;
}

输出结果:

学号    语文    数学    英语    综合    总分    平均分
1       85      90      78      92      345     86.2
2       76      88      95      80      339     84.8
3       92      85      90      88      355     88.8

二、函数

1. 函数定义与调用

函数是完成特定功能的代码块,提高代码复用性。

基本结构

返回值类型 函数名(参数列表) {// 函数体return 返回值; // void函数可省略return
}

示例

#include <iostream>
using namespace std;// 函数定义
int add(int a, int b) {return a + b;
}int main() {// 函数调用int result = add(3, 4);cout << "3+4=" << result << endl; // 输出:7return 0;
}

2. 参数传递方式

传递方式特点示例是否影响实参
值传递复制参数值void func(int a) {...}
地址传递传递内存地址(指针)void func(int *p) {...}
引用传递创建变量的别名(推荐)void func(int &r) {...}

对比示例

#include <iostream>
using namespace std;// 值传递(不影响实参)
void changeValue(int a) {a = 100;
}// 地址传递(影响实参)
void changeByPointer(int *p) {*p = 200;
}// 引用传递(影响实参)
void changeByReference(int &r) {r = 300;
}int main() {int num = 10;changeValue(num);cout << num << endl; // 输出:10changeByPointer(&num);cout << num << endl; // 输出:200changeByReference(num);cout << num << endl; // 输出:300return 0;
}

3. 函数常见样式

函数样式示例说明
无参无返回值void sayHello() {...}执行操作,不返回结果
有参无返回值void printSum(int a, int b)接收参数,执行操作
无参有返回值int getRandom() {...}生成并返回数据
有参有返回值int max(int a, int b) {...}接收参数,计算并返回结果

4. 函数声明

函数声明告诉编译器函数的存在,定义可以放在后面或其他文件。

#include <iostream>
using namespace std;// 函数声明(可以多次)
int multiply(int a, int b);int main() {cout << multiply(5, 6) << endl; // 输出:30return 0;
}// 函数定义(只能一次)
int multiply(int a, int b) {return a * b;
}

5. 函数的分文件编写⭐⭐⭐

将函数声明放在头文件(.h),定义放在源文件(.cpp),提高代码组织性。

项目结构

project/
├── main.cpp
├── math_utils.h  // 头文件
└── math_utils.cpp // 源文件

math_utils.h

#pragma once // 防止头文件重复包含// 函数声明
int add(int a, int b);
int subtract(int a, int b);

math_utils.cpp

#include "math_utils.h"// 函数定义
int add(int a, int b) {return a + b;
}int subtract(int a, int b) {return a - b;
}

main.cpp

#include <iostream>
#include "math_utils.h" // 包含自定义头文件using namespace std;int main() {cout << "10 + 5 = " << add(10, 5) << endl;cout << "10 - 5 = " << subtract(10, 5) << endl;return 0;
}

6. 综合函数知识点案例

实现数组处理工具集:

array_utils.h

#pragma once// 打印数组
void printArray(int arr[], int size);// 数组求和
int arraySum(int arr[], int size);// 查找最大值
int findMax(int arr[], int size);

array_utils.cpp

#include <iostream>
#include "array_utils.h"using namespace std;void printArray(int arr[], int size) {for (int i = 0; i < size; i++) {cout << arr[i] << " ";}cout << endl;
}int arraySum(int arr[], int size) {int sum = 0;for (int i = 0; i < size; i++) {sum += arr[i];}return sum;
}int findMax(int arr[], int size) {int max = arr[0];for (int i = 1; i < size; i++) {if (arr[i] > max) {max = arr[i];}}return max;
}

main.cpp

#include <iostream>
#include "array_utils.h"using namespace std;int main() {int numbers[] = {12, 45, 23, 67, 34, 89};int size = sizeof(numbers) / sizeof(numbers[0]);cout << "原始数组: ";printArray(numbers, size);cout << "数组总和: " << arraySum(numbers, size) << endl;cout << "最大值: " << findMax(numbers, size) << endl;return 0;
}

三、数组与函数的结合应用

数组作为函数参数时,实际传递的是数组首地址,函数内部可以修改原始数组。

#include <iostream>
using namespace std;// 修改数组元素
void doubleElements(int arr[], int size) {for (int i = 0; i < size; i++) {arr[i] *= 2; // 修改原始数组}
}// 打印数组
void printArray(int arr[], int size) {for (int i = 0; i < size; i++) {cout << arr[i] << " ";}cout << endl;
}int main() {int nums[] = {1, 2, 3, 4, 5};int size = sizeof(nums) / sizeof(nums[0]);cout << "原始数组: ";printArray(nums, size); // 1 2 3 4 5doubleElements(nums, size);cout << "加倍后数组: ";printArray(nums, size); // 2 4 6 8 10return 0;
}

总结

知识点要点
一维数组连续存储、索引从0开始、长度必须常量、数组名是首地址
二维数组行优先存储、可看作一维数组的数组、初始化需指定列数
函数定义包括返回类型、函数名、参数列表和函数体
参数传递值传递(副本)、地址传递(指针)、引用传递(别名)
函数声明提前告知编译器函数存在,定义可后置
分文件编写头文件放声明(.h),源文件放实现(.cpp),提高代码可维护性
数组与函数数组作为参数时传递地址,函数内可修改原始数组

掌握数组和函数是C++编程的基础,通过合理组织代码结构和数据存储,可以构建更复杂、高效的程序系统。

http://www.xdnf.cn/news/1014157.html

相关文章:

  • 砂石骨料数字孪生工厂应用案例:远眺科技三维可视化落地成效
  • 【解决方案】Kali 2022.3修复仓库密钥无交互一键安装docker,docker compose
  • 卷积神经网络(一)基础入门
  • VIC-3D应用指南系列之:DIC数字图像相关技术与热成像(VIC-3D IR System助力热载荷测试)
  • ue5的blender4.1groom毛发插件v012安装和使用方法(排除了冲突错误)
  • 小型化边缘计算设备 特点
  • ubuntu 系统 多条命令通过 bash 脚本执行
  • 深入解析 MySQL 架构:从基础到高级
  • 20250613在Ubuntu20.04.6下编译Rockchip的RK3576原厂Android14的SDK【整理编译】
  • 【Java学习笔记】集合介绍
  • C语言文件操作与预处理详解
  • 面向GPU、CPU及机器学习加速器的机器学习编译器
  • Blender基础知识-操作模式、基本操作、渲染、灯光、材质、粒子系统、动画
  • springboot项目中整合高德地图
  • leetcode题解538:把二叉搜索树转换为累加树
  • 微型导轨在实验室场景中的多元应用
  • 个人支出智能分析系统
  • 【HarmonyOS Next之旅】DevEco Studio使用指南(三十三) -> 构建任务
  • finereport普通报表根据用户权限限制数据查询
  • 动态规划算法的欢乐密码(二):路径问题
  • 【软件开发】什么是DSL
  • Excel大厂自动化报表实战(互联网金融-数据分析周报制作中)
  • 如何使用Postman做接口自动化测试
  • GitHub Actions 深度实践:零运维搭建 CI/CD 流水线
  • OCP 认证培训:踏入 Oracle 数据库专家的殿堂
  • 基于MATLAB的车牌检测系统:传统图像处理与深度学习的创新融合
  • 将MySQL数据库中所有表和字段编码统一改为utf8mb4_unicode_ci
  • 数据库学习(五)——MySQL索引
  • 2025年ASOC SCI2区TOP,强化学习驱动双邻域结构人工蜂群算法RL_DNSABC,深度解析+性能实测
  • React Native 构建与打包发布(iOS + Android)