[C++]指针与结构体
标题
- 一.指针
- 1.指针的定义和使用
- 2.指针所占的内存空间
- 3.空指针与野指针
- 4.const修饰指针
- 5.指针和数组
- 6.指针和函数
- 二.结构体
- 1.结构体的定义与使用
- 2.结构体数组
- 3.结构体指针
- 4.结构体的嵌套使用
- 5.结构体做函数参数
- 6.结构体中const使用场景
- 7.案例练习
一.指针
- 作用: 可以通过指针间接的访问一段内存
- 内存编号是从0开始记录的,一般用16进制的数字表示
- 可以利用指针变量保存地址
1.指针的定义和使用
- 定义
指针类型 指针名;(指针类型指int *,float *等等) - 使用:可以通过解引用的方法
*来操作指针所指向的内存
#include <iostream>
using namespace std;int main() {int a = 12;// 定义指针: 数据类型 * 指针名;int * p;p = &a; // &为取地址符号,可以取到a的地址cout << "&a = " << &a << endl;cout << " p = " << p << endl; // 二者打印的内容相同,均为变量a的地址// 使用指针*p = 12345; // *代表解引用,可以通过它找到指针指向的内存中的数据cout << "*p = " << *p << endl;cout << " a = " << a << endl; // 修改*p的值也间接的修改了变量a的值system("pause");return 0;
}
2.指针所占的内存空间
- 32位系统:不管什么类型的指针都占用4个字节空间
- 64位:8字节
可以在这里调整编译器编译时的32或64位选项

int a = 99;int* p = &a;// 具体输出和操作系统有关cout << "sizeof(int *) = " << sizeof(p) << endl;cout << "sizeof(bool *) = " << sizeof(bool *) << endl;cout << "sizeof(char *) = " << sizeof(char *) << endl;cout << "sizeof(double *) = " << sizeof(double *) << endl;cout << "sizeof(long long *) = " << sizeof(long long *) << endl;
3.空指针与野指针
- 空指针:指向的内存编号为0的指针
用途:不知道指针的具体赋值时,进行指针的初始化
注意:空指针指向的内存是不可以被访问的!!!(0~255之间的内存编号是系统占用的,不能被访问)
这段代码在编写的时候没有问题,但是在编译运行的时候会报异常
#include <iostream>
using namespace std;int main() {int* p = NULL;int a = *p;system("pause");return 0;
}
- 野指针:指针变量指向了非法的内存空间
int * p = (int *)0x1100;
// 这里会报访问权限异常,因为这个地址就不是由你本人申请的!
int c = *p;
空指针与野指针,都不是我们申请的空间,请不要随意的访问它!!!!
4.const修饰指针
- const修饰指针有三种情况:
1.const修饰指针—常量指针
2.const修饰常量—指针常量
3.const即修饰指针也修饰常量
5.指针和数组
- 可以利用指针访问数组内的元素
int arr[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };int* p = arr; // 将这个指针指向数组的首地址cout << "指针访问第一个元素:" << *p << endl; // 输出0/*// 如果需要用指针来访问第二个元素,需要让指针指向后移4个字节p++; // 自增已经可以让指针向后移动4字节cout << "第二个元素: " << *p << endl; // 输出1*/// 利用指针遍历数组元素for (int i = 0; i < size(arr); i++) {cout << "arr[" << i << "] = " << *p << endl;cout << "p = " << p << endl;cout << "&arr[" << i << "] = " << &arr[i] << endl;p++;}
6.指针和函数
- 利用指针作为函数的参数, 通过解引用可以修改实参的值!
#include <iostream>
using namespace std;void swap01(int* a, int* b);// 值传递,并不会改变传入的形参的值
void swap02(int a, int b) {int temp = a;a = b;b = temp;
}int main() {int a = 33;int b = 99;cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "值传递" << endl;swap02(a, b);// 不会改变传入的实参的值cout << "a = " << a << endl; // 33cout << "b = " << b << endl; // 99cout << "地址传递" << endl;swap01(&a, &b);// 可以看到这里的值已经被改变了cout << "a = " << a << endl; // 99cout << "b = " << b << endl; // 33system("pause");return 0;
}// 交换两个数的函数, 通过解引用的方式交换两个数
void swap01(int* a, int* b) { // 把房间号传递过来了,可以借助房间号,改变房间里放的东西int temp = *a;*a = *b;*b = temp;
}
案例描述:实现一个函数,利用冒泡排序对整数数组进行降序排序
#include <iostream>
using namespace std;// 输出数组内容
void printArray(int* arr, int size) {for (int i = 0; i < size; i++) {cout << arr[i] << "\t";}cout << endl;
}// 冒泡排序
void bubbleSort(int* arr, int size) {for (int i = 0; i < size - 1; i++) {for (int j = 0; j < size - 1 - i; j++) {if (arr[j] < arr[j + 1]) {int temp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = temp;}}}
}int main() {int arr[] = { 9, 34, 56, 7, 123, 45, 67, 98, 33, 12, 55, 987, 11, -2 };int* arrP = arr; // arr就是数组的首地址bubbleSort(arr, size(arr));printArray(arr, size(arr));system("pause");return 0;
}
二.结构体
- 结构体属于用户自定义的数据类型, 允许用户存储不同的数据类型
1.结构体的定义与使用
- 语法
struct 结构体名 { 结构体成员列表 }; - 通过结构体创建变量(使用结构体)的方式有三种:
1.struct 结构体名 变量名;
2.struct 结构体名 变量名 = { 成员值1, 成员值2... };
3.定义结构体是顺便创建变量
#include <iostream>
using namespace std;// 定义结构体
struct Student {string name;int age;double score;
} s0; // 创建结构体的时候顺便创建结构变量(不推荐)void printStudent(Student stu) {cout << "name: " << stu.name << " age: " << stu.age << " score: " << stu.score << endl;
}int main() {// 这种方式struct关键字能省略不写struct Student s1;s1.name = "叶子";s1.age = 18;s1.score = 100;printStudent(s1);// 按顺序填入结构体数据Student s2 = { "老黄", 25, 60 };printStudent(s2);// 创建的时候顺便创建的量s0.name = "YeZi";s0.age = 25;s0.score = 100;printStudent(s0);system("pause");return 0;
}
2.结构体数组
- 将自定义的结构体放入数组中方便维护
- 语法
struct 结构体名 数组名[元素个数] = { {} , {}, ... {} }
#include <iostream>
using namespace std;struct Student {// 成员列表string name; // 姓名int age; // 年龄int score; // 分数
};int main() {// 结构体数组Student stus[3] = {{"叶子", 18, 100},{"老黄", 20, 99},{"罗磊", 21, 100}};// 遍历结构体数组for (int i = 0; i < size(stus); i++) {cout << "姓名:" << stus[i].name<< " 年龄:" << stus[i].age<< " 分数:" << stus[i].score << endl;}system("pause");return 0;
}
3.结构体指针
- 通过指针来访问结构体的元素
- 利用操作符
->可以通过结构体指针访问结构体的属性
#include <iostream>
using namespace std;struct Student {// 成员列表string name; // 姓名int age; // 年龄int score; // 分数
};int main() {Student stu0 = { "老黄", 20, 99 };// 指针指向结构体变量Student* p = &stu0;// 通过指针访问结构体变量中的数据cout << "姓名: " << p->name << " 年龄: " << p->age << " 分数: " << p->score << endl;system("pause");return 0;
}
4.结构体的嵌套使用
- 结构体也可以结合实际情况嵌套使用
#include <iostream>
using namespace std;struct Student {// 成员列表string name; // 姓名int age; // 年龄int score; // 分数
};struct Teacher {string name;int age;Student stu;
};int main() {Teacher teacher = {"娟", 18, {"叶子", 18, 100}};cout << "老师姓名: " << teacher.name << " 老师年龄: " << teacher.age<< " 学生姓名: " << teacher.stu.name << " 学生年龄: " << teacher.stu.age << " 学生分数: " << teacher.stu.score << endl;system("pause");return 0;
}
5.结构体做函数参数
- 将结构体作为参数向函数中传递
- 方式有两种:
值传递
地址传递(形参修改影响实参的值)
#include <iostream>
using namespace std;struct Student {string name;int age;double score;
};// 输出函数,值传递不影响实参变量
void printStudent(Student stu, string tag) {cout <<"tag: " << tag << " name: " << stu.name << " age : " << stu.age << " score : " << stu.score << endl;
}// 值传递,在函数中修改结构体不会影响具体值
void printStudent1(Student stu) {stu.name = "luilui";stu.age = 20;stu.score = 99;printStudent(stu, "printStudent1");
}// 引用传递,函数中修改结构体会影响实参的值
void printStudent2(Student* ps) {ps -> name = "老黄";ps -> age = 20;ps -> score = 99;printStudent(*ps, "printStudent2");
}int main() {Student stu;stu.name = "叶子";stu.age = 18;stu.score = 100;cout << "值传递: " << endl;printStudent(stu, "main");printStudent1(stu);printStudent(stu, "main"); // 函数里修改了具体的值,但是值传递不影响实参结构体cout << "\n\n引用传递: " << endl;printStudent(stu, "main");printStudent2(&stu); // 取地址符,传入的是对应结构体的指针printStudent(stu, "main"); // 引用传递,函数里边的修改影响实参结构体system("pause");return 0;
}
6.结构体中const使用场景
- 作用: 用const来防止误操作(可以结合前文中的常量指针进行理解)
示例:
#include <iostream>
using namespace std;struct Student {string name;int age;double score;
};// 3值传递时,会将实参复制给形参,数据量越大,占用的内存越多
void printStruct(Student stu) {cout << "name: " << stu.name << " age: " << stu.age << " score: " << stu.score << endl;
}// 4将函数中的形参改为指针,不会复制,所以会介绍内存空间的占用
void printStruct(const Student *stu) { // 6加入const(常量指针)可以防止在形参修改,影响到实参的值// 5这样会留下一个隐患,地址传递时对形参指针的修改会影响到实参的值// stu->name = "老黄"; 7加入const之后会报错,防止误操作cout << "name: " << stu->name << " age: " << stu->age << " score: " << stu->score << endl;
}int main() {//1创建结构体变量Student s = { "叶子", 18, 100.00 };//2通过函数打印结构体的信息printStruct(s);printStruct(&s);system("pause");return 0;
}
7.案例练习
- 案例1:
#include <iostream>
using namespace std;struct Student {string name;int score;
};struct Teacher {string name;Student stus[5];
};void input(Teacher* tchs, int sizeT) {for (int i = 0; i < sizeT; i++) {string name = "";cout << "请输入第" << i << "位老师的姓名:";cin >> name;tchs[i].name = name;cout << "开始录入 " << name << " 老师的学生们!" << endl;for (int j = 0; j < size(tchs[i].stus); j++) {string name = "";cout << "输入学生" << j << "的姓名:";cin >> name;tchs[i].stus[j].name = name;cout << "输入 " << name << " 的成绩:" << endl;int score = 0;cin >> score;tchs[i].stus[j].score = score;}}
}void input1(Teacher tchs[], int sizeT) {for (int i = 0; i < sizeT; i++) {string name = "";cout << "请输入第" << i << "位老师的姓名:";cin >> name;tchs[i].name = name;cout << "开始录入 " << name << " 老师的学生们!" << endl;for (int j = 0; j < size(tchs[i].stus); j++) {string name = "";cout << "输入学生" << j << "的姓名:";cin >> name;tchs[i].stus[j].name = name;cout << "输入 " << name << " 的成绩:" << endl;int score = 0;cin >> score;tchs[i].stus[j].score = score;}}
}void print(Teacher* tchs, int sizeT) {for (int i = 0; i < sizeT; i++) {cout << tchs[i].name << " 老师的学生信息:" << endl;for (int j = 0; j < size(tchs[i].stus); j++) {string name = "";cout << tchs[i].stus[j].name << " 的成绩是: " << tchs[i].stus[j].score << endl;}}}int main() {Teacher teachers[3];input(teachers, size(teachers));print(teachers, size(teachers));system("pause");return 0;
}
- 案例2
#include<iostream>
#include<ctime>
using namespace std;/*设计一个英雄的结构体,包括成员 姓名,年龄,性别创建结构体数组,数组中存放5名英雄。通过冒泡排序法将数组中的英雄按照年龄进行升序排序,打印最终结果。*/// 英雄结构体
struct Hero {string name;int age;string sex;
};// 随机获取英雄的年龄
int getRandAge() {int age = rand() % 51;if (age <= 21) { // 小于201岁重新获取return getRandAge();}return age;
}// 初始化英雄数据
void initHeroInfo(Hero heros[], int len) {srand((unsigned int)time(NULL));heros[0] = { "刘备", getRandAge(), "男"};heros[1] = { "关羽", getRandAge(), "男" };heros[2] = { "张飞", getRandAge(), "男" };heros[3] = { "赵云", getRandAge(), "男" };heros[4] = { "貂蝉", getRandAge(), "女" };heros[5] = { "西施", getRandAge(), "女" };}// 冒泡排序
void sortHeros(Hero *heros, int len) {for (int i = 0; i < len - 1; i++) {for (int j = 0; j < len - i - 1; j++) {if (heros[j].age > heros[j + 1].age) {Hero temp = heros[j];heros[j] = heros[j + 1];heros[j + 1] = temp;}}}
}// 打印英雄信息
void printHeros(const Hero *heros, int len) {for (int i = 0; i < len; i++) {cout << "姓名: " << heros[i].name << " 年龄: " << heros[i].age << " 性别: " << heros[i].sex << endl;}cout << endl;
}int main() {Hero heros[6];initHeroInfo(heros, size(heros));printHeros(heros, size(heros));sortHeros(heros, size(heros));printHeros(heros, size(heros));system("pause");return 0;
}
学习笔记与课程计划
B站视频链接
相关文章:
[C++]指针与结构体
标题 一.指针1.指针的定义和使用2.指针所占的内存空间3.空指针与野指针4.const修饰指针5.指针和数组6.指针和函数 二.结构体1.结构体的定义与使用2.结构体数组3.结构体指针4.结构体的嵌套使用5.结构体做函数参数6.结构体中const使用场景7.案例练习 一.指针 作用: 可以通过指针…...
注解原理是什么 Spring MVC常用的注解有哪些?
文章目录 注解原理是什么Spring MVC常用的注解有哪些? 通过这篇文章来和大家一起认识springMVC常用的注解,那么首先需要来了解注解。 注解原理是什么 注解本质是一个继承了Annotation的特殊接口,其具体实现类是Java运行时生成的动态代理类。…...
【数据结构】树与二叉树(廿五):树搜索指定数据域的结点(算法FindTarget)
文章目录 5.3.1 树的存储结构5. 左儿子右兄弟链接结构 5.3.2 获取结点的算法1. 获取大儿子、大兄弟结点2. 搜索给定结点的父亲3. 搜索指定数据域的结点a. 算法FindTargetb. 算法解析c. 代码实现a. 使用指向指针的指针b. 直接返回找到的节点 4. 代码整合 5.3.1 树的存储结构 5.…...
深度学习图像风格迁移 - opencv python 计算机竞赛
文章目录 0 前言1 VGG网络2 风格迁移3 内容损失4 风格损失5 主代码实现6 迁移模型实现7 效果展示8 最后 0 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 深度学习图像风格迁移 - opencv python 该项目较为新颖,适合作为竞赛课题…...
提高SQL语句执行效率的8个方法
提高SQL语句执行效率的8个方法 在日常的数据库操作中,如何提高SQL语句的执行效率是每个程序员都需要关注的问题,SQL语句的执行效率对系统的性能有着重要影响,本文将介绍8种提高SQL语句执行效率的方法。 合理使用索引 索引介绍 索引是数据…...
C语言,通过数组实现循环队列
实现循环队列最难的地方就在于如何判空和判满,只要解决了这两点循环队列的设计就没有问题。接下来我们将会使用数组来实现循环队列。 接下来,为了模拟实现一个容量为4的循环队列,我们创建一个容量为4 1 的数组。 接下来我们将会对这个数组…...
python+pygame+opencv+gpt实现虚拟数字人直播(一)
AI技术突飞猛进,不断的改变着人们的工作和生活。数字人直播作为新兴形式,必将成为未来趋势,具有巨大的、广阔的、惊人的市场前景。它将不断融合创新技术和跨界合作,提供更具个性化和多样化的互动体验,成为未来的一种趋…...
c语言:模拟实现各种字符串函数(2)
strncpy函数: 功能:拷贝指定长度的字符串a到字符串b中 代码模拟实现: //strncpy char* my_strncpy(char* dest, char* str,size_t num) {char* ret dest;assert(dest && str);//断言,如果其中有一个为空指针ÿ…...
【Proteus仿真】【STM32单片机】感应水龙头设计
文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真STM32单片机控制器,使用LCD1602液晶模块、HCSR04超声波等。 主要功能: 系统运行后,LCD1602显示超声波模块检测的距离,若检测距离小…...
P15 C++ 枚举
The ChenPi 前言 今天我们要讲的是 C 中的枚举。 enum 是 enumeration 的缩写,基本上可以说,它就是一个数值集合。如果你想要给枚举一个更实际的定义,它们是给一个值命名的一种方法。 所以我们不用一堆叫做 A、B、C 的整数。我们可以有一个…...
深入理解路由协议:从概念到实践
路由技术是Internet得以持续运转的关键所在,路由是极其有趣而又复杂的课题,永远的话题。 SO:这是一个解析路由协议的基础文章。 目录 前言路由的概念路由协议的分类数据包在网络中的路由过程理解路由表的结构路由器关键功能解析 前言 在互联…...
Qt 串口编程-从入门到实战
1. Qt 串口通信流程解析 1.1 串行通信和并行通信对比 并行通信适合距离较短的通信,且信号容易受干扰,成本高串口通讯-设备(蓝牙, wifi, gprs, gps) 1.2 Qt 串口通信具体流程 1. 创建 QSerial…...
如何获得微软MVP徽章
要成为微软MVP,需要在特定领域成为专家,并积极参与社区,为其他人提供帮助和支持。以下是一些步骤可以帮助你成为MVP: 在特定领域成为专家:要成为MVP,需要在某个领域具有专业知识和经验。这可以通过阅读相关…...
Java架构师软件架构开发
目录 1 基于架构的软件开发导论2 ABSD架构方法论3 ABSD方法论具体实现4 ABSD金融业案例5 基于特定领域的软件架构开发导论6 DSSA领域分析7 DSSA领域设计和实现8 DSSA国际电商平台架构案例9 架构思维方法论概述10 AT方法论和案例想学习架构师构建流程请跳转:Java架构师系统架构…...
西南科技大学数字电子技术实验一(数字信号基本参数与逻辑门电路功能测试及FPGA 实现 )预习报告
手写报告稍微认真点写,80+随便有 目录 一、计算/设计过程 1、通过虚拟示波器观察和测量信号 2、通过实际电路(电阻、开关、发光二极管)模拟逻辑门电路 二、画出并填写实验指导书上的预表...
Java八股文面试全套真题【含答案】- SpringMVC篇
以下是一些关于Spring MVC语言的经典面试题以及它们的答案: 什么是Spring MVC框架?它的特点是什么? Spring MVC是基于Java的一种Web应用框架,用于开发基于MVC(模型-视图-控制器)模式的Web应用程序。它的特…...
Spring第二课响应的完全,如何理解前后端互联
目录 一、响应 Control,RestController 1.Controller的源码,代表什么意思 2.返回数据 Responsebody 3.返回HTML片段 4.返回JSON 5.那么假如我们使用集合会怎么样呢 设置状态码,虽然不影响展示,但是确实显示起来也就是401的情况。 2.我…...
html实现各种瀑布流(附源码)
文章目录 1.设计来源1.1 动态响应瀑布流1.2 分页瀑布流1.3 响应瀑布流 2.效果和源码2.1 动态效果2.2 源代码 源码下载 作者:xcLeigh 文章地址:https://blog.csdn.net/weixin_43151418/article/details/134613121 html实现各种瀑布流(附源码),…...
万字解析设计模式之责任链模式、状态模式
目录 一、责任链模式 1.1概述 1.2结构 1.3实现 1.4 优缺点 1.5应用场景 1.6源码解析 二、状态模式 2.1概述 2.2结构 2.3实现 2.4优缺点 2.5应用场景 三、责任链模式实验 任务描述 实现方式 编程要求 测试说明 四、状态模式实验 任务描述 实现方式 编程要…...
二十三种设计模式全面解析-深入探讨状态模式的高级应用技术:释放对象行为的无限可能
在软件开发中,状态管理是一个常见的挑战。当对象的行为随着内部状态的变化而变化时,有效地管理对象的状态和相应的行为变得至关重要。在这方面,状态模式提供了一种优雅而灵活的解决方案。它允许对象在运行时根据内部状态的改变而改变其行为&a…...
浅谈 React Hooks
React Hooks 是 React 16.8 引入的一组 API,用于在函数组件中使用 state 和其他 React 特性(例如生命周期方法、context 等)。Hooks 通过简洁的函数接口,解决了状态与 UI 的高度解耦,通过函数式编程范式实现更灵活 Rea…...
PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建
制造业采购供应链管理是企业运营的核心环节,供应链协同管理在供应链上下游企业之间建立紧密的合作关系,通过信息共享、资源整合、业务协同等方式,实现供应链的全面管理和优化,提高供应链的效率和透明度,降低供应链的成…...
visual studio 2022更改主题为深色
visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中,选择 环境 -> 常规 ,将其中的颜色主题改成深色 点击确定,更改完成...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
在四层代理中还原真实客户端ngx_stream_realip_module
一、模块原理与价值 PROXY Protocol 回溯 第三方负载均衡(如 HAProxy、AWS NLB、阿里 SLB)发起上游连接时,将真实客户端 IP/Port 写入 PROXY Protocol v1/v2 头。Stream 层接收到头部后,ngx_stream_realip_module 从中提取原始信息…...
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
使用 Streamlit 构建支持主流大模型与 Ollama 的轻量级统一平台
🎯 使用 Streamlit 构建支持主流大模型与 Ollama 的轻量级统一平台 📌 项目背景 随着大语言模型(LLM)的广泛应用,开发者常面临多个挑战: 各大模型(OpenAI、Claude、Gemini、Ollama)接口风格不统一;缺乏一个统一平台进行模型调用与测试;本地模型 Ollama 的集成与前…...
保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek
文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama(有网络的电脑)2.2.3 安装Ollama(无网络的电脑)2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...
iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈
在日常iOS开发过程中,性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期,开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发,但背后往往隐藏着系统资源调度不当…...
