并非从0开始的c++ day8
并非从0开始的c++ day8
- 结构体
- 结构体嵌套二级指针练习
- 结构体偏移量
- 内存对齐
- 内存对齐的原因
- 如何内存对齐
- 文件操作
- 文件的概念
- 流的概念
- 文本流
- 二进制流
- 文件缓冲区
- 文件打开关闭
- 文件关闭fclose
- 文件读写函数回顾
- 按格式化读写文件
- 文件读写注意事项
结构体
结构体嵌套二级指针练习
需求:一个老师数组,老师设计一个结构体,需要有老师的名字和一个学生数组,学生数组可以有若干个学生
//结构体设计
struct Teacher
{//老师姓名char* name;//老师带的学生姓名数组char** students;
};void allocateSpace(struct Teacher *** teacherArray)
{if (teacherArray == NULL){return;}//堆区分配内存struct Teacher** ts = malloc(sizeof(struct Teacher*) * 3);//数据赋值//给老师分配内存for (int i = 0; i < 3; i++){//给老师分配内存ts[i] = malloc(sizeof(struct Teacher));//给老师姓名属性 分配内存ts[i]->name = malloc(sizeof(char) * 64);//给老师姓名赋值sprintf(ts[i]->name, "Teacher_%d", i+1);//给老师带领学生数组分配内存ts[i]->students = malloc(sizeof(char*) * 4);//给学生姓名分配内存 并且赋值for (int j = 0; j < 4; j++){ts[i]->students[j] = malloc(sizeof(char) * 64);sprintf(ts[i]->students[j], "%s_student_%d", ts[i]->name, j+1);}}//建立关系*teacherArray = ts;
}//打印操作
void printTeacherArray(struct Teacher ** teacherArray)
{for (int i = 0; i < 3; i++){//老师姓名printf("%s\n", teacherArray[i]->name);for(int j = 0;j<4;j++)//老师带领的学生printf(" %s\n", teacherArray[i]->students[j]);}
}//释放堆区数据
void freeSpace(struct Teacher** teacherArray)
{if (teacherArray == NULL){return;}for (int i = 0; i < 3; i++){//释放老师姓名if (teacherArray[i]->name != NULL){free(teacherArray[i]->name);teacherArray[i]->name = NULL;}//释放学生姓名for (int j = 0; j < 4; j++){if (teacherArray[i]->students[j] != NULL){free(teacherArray[i]->students[j]);teacherArray[i]->students[j] = NULL;}}//释放学生数组free(teacherArray[i]->students);teacherArray[i]->students = NULL;//释放老师free(teacherArray[i]);teacherArray[i] = NULL;}//释放老师数组free(teacherArray);teacherArray = NULL;
}void test01()
{//老师数组创建struct Teacher** teacherArray = NULL;//分配内存allocateSpace(&teacherArray);//打印所有老师和学生信息printTeacherArray(teacherArray);//释放堆区数据freeSpace(teacherArray);teacherArray = NULL;
}
牢记几个malloc对应几个free
结构体偏移量
printf(“b的偏移量:%d\n”, (int)&(p->b) - (int)p);
printf(“b的偏移量: %d\n”, offsetof(struct teacher, b));
struct teacher
{char a; //0 ~ 3int b; //4 ~ 7};void test01()
{struct teacher t1;struct teacher* p = &t1;printf("b的偏移量:%d\n", (int)&(p->b) - (int)p);printf("b的偏移量: %d\n", offsetof(struct teacher, b));
}
通过偏移量获取数
void test02()
{struct teacher t1 = {'a',1000};struct teacher* p = &t1;printf("b的值为: %d\n", *(int*)((char*)p + offsetof(struct teacher, b)));printf("b的值为: %d\n", ((struct teacher *)((int*)p + 1))-> b);
}//结构体2
struct teacher2
{char a;int b;struct teacher c;
};void test03()
{struct teacher2 t = { 'a',10,'b',20 };int offset1 = offsetof(struct teacher2, c);int offset2 = offsetof(struct teacher, b);printf("属性c中的值为: %d\n", *(int *)((char*)&t + offset1 + offset2));printf("属性c中的值为: %d\n", *(int*)((char*)&t + offset1 + offset2));}
内存对齐
访问特定类型的变量只能在特定的地址访问,这就需要各个变量在空间上按一定的规则排列,而不是简单地顺序排列,这就是内存对齐
内存对齐的原因
我们知道内存的最小单元是一个字节,当CPU从内存中读取数据的时候,是一个一个字节读取
实际上cpu将内存当成多个块,每次从内存中读取一个块,这个块的大小可能是2、4、8、16等
内存对齐是操作系统为了提高访问内存的策略。操作系统在访问内存的时候,每次读取一定长度(这个长度是操作系统默认的对齐数,或者默认对齐数的整数倍)。如果没有对齐,为了访问一个变量可能产生二次访问
内存对齐优点:以空间换时间
如何内存对齐
- 对于标准数据类型,它的地址只要是他的长度的整数倍
- 对于非标准数据类型,比如结构体,要遵循一下对齐原则
内存对齐原则
第一个属性开始 从0开始计算偏移量
第二个属性 要放在该属性的大小 与 对齐模数比 取小的值的 整数倍上
当所有属性都计算完毕之后,整体做二次偏移,将上面计算的结构体
将上面计算的结果 扩充到 这个结构体中最大数据类型的整数倍 与对齐模数比 取小的值 的整数倍
对齐模数默认为8
#pragma pack (show)//查看对齐模数的值 默认为8
#pragma pack (1)typedef struct _STUDENT {int a; // 0 ~ 3 //0~3char b; //4 ~ 7 //4double c;//8 ~ 15 //5~12float d;// 16 ~ 23 //13~16
}Student;void test01()
{printf("student sizeof = %d\n", sizeof(Student));
}
当嵌套结构体时,以子结构体最大数据类型的整数倍来取地址即可
typedef struct _STUDENT2 {char a; //0 ~ 7Student b; //8 ~ 31double c; //32 ~ 39
}Student2;void test02()
{printf("student sizeof = %d\n", sizeof(Student2));
}
文件操作
文件的概念
通过fopen打开文件,中间还有一步open系统函数,open系统函数获取文件后,返回文件的指针FILE*给fopen
流的概念
C语言中,I/O操作可以简单地看做从程序移进或移出字节,这种搬运的过程称为流。I/O都是相对于程序来说,所以o输出为将字节输出到文件里,I输入为从文件读取
文本流
二进制流
我们程序中,经常看到的文本方式打开文件和二进制打开文件仅仅体现在换行符的处理上
输入/输出函数家族
家族名 目的 可用于所有流 只用于stdin和stdout
getchar 字符输入 fgetc、getc getchar
putchar 字符输出 fputc、putc putchar
gets 文本行输入 fgets gets
puts 文本行输出 fputs puts
scanf 格式化输入 fscanf scanf
printf 格式化输出 fprintf pirntf
文件缓冲区
有了缓冲区:提高硬盘寿命、提高运行效率
写文件时需要写fclose,因为可能有些数据留在写文件缓冲区,要是直接结束,可能会留这部分数据在里面没写进去,需要我们手动将其输出到文件
文件打开关闭
“r” 读
“w”写
“a”追加
“rb”二进制只读
“wb”二进制只写
“ab”二进制追加
“r+”允许读和写,文件必须已存在
“w+”允许读和写,如果文件不存在则创建,已存在则把文件长度截断为0字节再重新写
“a+”允许读和追加数据,如果文件不存在则创建
“rb+”以读或写方式打开一个二进制文件
“wb+”以读或写方式建立一个新的二进制文件
“ab+”以读或写方式打开一个二进制文件进行追加
文件关闭fclose
对打开文件进行写入时,若文件缓冲区的空间未被写入的内容填满,这些内容弄不会写到打开的文件中。只有对打开的文件进行关闭操作时,停留在文件缓冲区的内容才能写到改文件中,从而使文件完整。再者一旦关闭了文件,该文件对应的FILE结构将被释放,从而使关闭的文件受到保护,因为这时对该文件的存取操作将不会进行。文件的关闭也意味着释放了该文件的缓冲区
文件读写函数回顾
- 按照字符读写文件:fgetc(),fputc()
- 按照行读写文件:fputs(),fgets()
- 按照块读写文件:fread(),fwrite()
- 按照格式化读写文件:fprintf(),fscanf()
- 按照随机位置读写文件:fseek(),ftell(),rewind()
while ((ch = fgetc(f_read)) != EOF)
判断是否是文件尾
//按字符方式读写
void test01()
{//写文件FILE * f_write = fopen("./test1.txt", "w+");if (f_write == NULL){return;}char buf[] = "hello world";for (int i = 0; i < strlen(buf); i++){fputc(buf[i], f_write);}fclose(f_write);//读文件FILE* f_read = fopen("./test1.txt", "r");if (f_read == NULL){return;}char ch;while ((ch = fgetc(f_read)) != EOF){printf("%c", ch);}printf("\n");fclose(f_read);
}
按格式化读写文件
//移动文件光标
void test05()
{//打开文件FILE* f_write = fopen("./test5.txt", "wb");if (f_write == NULL){return;}struct Hero heros[4] ={{"geats",19},{"revice",20},{"saber",21},{"zero one",22}};for (int i = 0; i < 4; i++)//参数1 写入数据的地址 参数2 块大小 参数3 快个数 参数4 文件指针fwrite(&heros[i], sizeof(struct Hero), 1, f_write);//关闭文件fclose(f_write);//读文件FILE* f_read = fopen("./test5.txt", "rb");struct Hero temp;if (f_read == NULL){perror("文件打开失败");//errno宏 全局变量return;}//移动文件光标//fseek(f_read, sizeof(struct Hero) , SEEK_SET);fseek(f_read, -(long)sizeof(struct Hero) * 2, SEEK_END);//将文件光标置首rewind(f_read);fread(&temp, sizeof(struct Hero), 1, f_read);printf("姓名: %s 年龄:%d\n", temp.name, temp.age);fclose(f_read);
}
//参数1 写入数据的地址 参数2 块大小 参数3 快个数 参数4 文件指针
fwrite(&heros[i], sizeof(struct Hero), 1, f_write);
//移动文件光标
//fseek(f_read, sizeof(struct Hero) , SEEK_SET);
fseek(f_read, -(long)sizeof(struct Hero) * 2, SEEK_END);
前者从前往后移,后者从后往前移,后者需要强制类型转换
//将文件光标置首
rewind(f_read);
perror("文件打开失败");//errno宏 全局变量
对于每一个错误都会有一个相应的代码
文件读写注意事项
void test01()
{//按照字符读test文件FILE* file = fopen("test.txt", "r");if (file == NULL)return;#if 0char ch;while (!feof(file)){ch = fgetc(file);if (feof(file)){break;}printf("%c", ch);}
#endifchar ch;while ((ch = fgetc(file)) != EOF){printf("%c", ch);}fclose(file);
}
-
EOF为结尾,在if内部的代码用feof有滞后性,在文件读取结束后,还会进一次循环后再退出,if下面的代码就可以解决
-
如果结构体中属性,创建在堆区,保存数据到文件中的时候,要将指针放入到文件中
相关文章:

并非从0开始的c++ day8
并非从0开始的c day8结构体结构体嵌套二级指针练习结构体偏移量内存对齐内存对齐的原因如何内存对齐文件操作文件的概念流的概念文本流二进制流文件缓冲区文件打开关闭文件关闭fclose文件读写函数回顾按格式化读写文件文件读写注意事项结构体 结构体嵌套二级指针练习 需求&am…...

ubuntu下用i686-w64-mingw32交叉编译支持SDL、Openssl的ffmpeg库
前言 本篇博客是基于前两篇关于ffmpeg交叉编译下,进行再次编译操作。ubuntu下ffmpeg的交叉编译环境搭建可以参看以下我的这篇博客:https://blog.csdn.net/linyibin_123/article/details/108759367 ; ubuntu下交叉编译openssl及交叉编译支持o…...

对IDEA中断点Suspend 属性理解
suspend的类型分为 1、ALL:有线程进入该断点时,暂停所有线程 2、Thread:有线程进入该断点时,只暂停该线程 讨论下不同线程在同一时间段都遇到断点时,idea的处理方法。假如在执行时间上,thread1会先进入断…...

IM即时通讯开发如何解决大量离线消息导致客户端卡顿的
大部分做后端开发的朋友,都在开发接口。客户端或浏览器h5通过HTTP请求到我们后端的Controller接口,后端查数据库等返回JSON给客户端。大家都知道,HTTP协议有短连接、无状态、三次握手四次挥手等特点。而像游戏、实时通信等业务反而很不适合用…...

【软件测试】测试老鸟的迷途,进军高级自动化测试测试......
目录:导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜)前言 很多从业几年的选手…...

HMM(隐马尔科夫模型)-理论补充2
目录 一.大数定理 二.监督学习方法 1.初始概率 2.转移概率 3.观测概率 三.Baum-Welch算法 1.EM算法整体框架 2. Baum-Welch算法 3.EM过程 4.极大化 5.初始状态概率 6.转移概率和观测概率 四.预测算法 1.预测的近似算法 2.Viterbi算法 1.定义 2. 递推࿱…...

【分布式系统】MinIO之Multi-Node Multi-Drive架构分析
文章目录架构分析节点资源硬盘资源服务安装安装步骤创建系统服务新建用户和用户组创建环境变量启动服务负载均衡代码集成注意最近打算使用MinIO替代原来使用的FastDFS,所以一直在学习MinIO的知识。这篇文章是基于MinIO多节点多驱动的部署进行研究。 架构分析 节点资…...
【无标题】(2019)NOC编程猫创新编程复赛小学组真题含参考
(2019)NOC编程猫创新编程复赛小学组最后6道大题。前10道是选择填空题 略。 这道题是绘图题,没什么难度,大家绘制这2个正十边形要注意:一是不要超出舞台;二是这2个正十边形不要相交。 这里就不给出具体程序了…...

【尚硅谷MySQL入门到高级-宋红康】数据库概述
1、为什么要使用数据库 数据的持久化 2、数据库与数据库管理系统 2.1 数据库的相关概念 2.2 数据库与数据库管理系统的关系 3、 MySQL介绍 MySQL从5.7版本直接跳跃发布了8.0版本 ,可见这是一个令人兴奋的里程碑版本。MySQL 8版本在功能上做了显著的改进与增强&a…...

SpringBoot集成Redis并实现数据缓存
应用场景 存放Token、存放用户信息或字典等需要频繁访问数据库获取但不希望频繁访问增加数据库压力且变化不频繁的数据。 集成步骤 1. 新建 Maven 项目并引入 redis 依赖【部分框架有可能已经集成,会导致依赖文件有差异】 <dependency><groupId>org…...

SpringBoot配置文件(properties yml)
查看官网更多系统配置项:https://docs.spring.io/spring-boot/docs/current/reference/html/application-properties.html#application-properties 1.配置⽂件作⽤ 整个项⽬中所有重要的数据都是在配置⽂件中配置的,⽐如:数据库的连接信息&am…...

css 画图之质感盒子
前言 css 众所周知可以做很多的事情,比如:界面效果、特效、独特的样式等。今天给各位朋友带来的是以box-shadow来画一个很有质感效果的一个盒子。 之前在网上冲浪的时候,发现了这样的一个效果,所以来记录一下。 下面是实现后的…...

面了一个月,终于让我总结出了这份最详细的接口测试面试题
目录 1、你们公司是如何做接口测试的? 2、什么时候开展接⼝测试? 3、接⼝测试和UI测试的工作是否重复? 4、接口测试框架怎么搭建? 5、接⼝之间有依赖时怎么处理? 6、如何判断接⼝测试的结果(成功或失败&a…...

{新}【java开发环境安装】完整工作环境安装配置
公司新发了一台红米笔记本,打算用新的笔记本,开启自己新的工作旅程,其中把做个的事都记录一边,以便实现,听、读、视频图像、讨论、实践、教人的一个学习过程。 一、Java开发环境安装 找到安装包下载;在官…...
Python|每日一练|数组|数学|图算法|字符串|动态规划|单选记录:加一|迷宫问题|扰乱字符串
1、加一(数组,数学) 给定一个由 整数 组成的 非空 数组所表示的非负整数,在该数的基础上加一。 最高位数字存放在数组的首位, 数组中每个元素只存储单个数字。 你可以假设除了整数 0 之外,这个整数不会以…...
MySQL 使用IF判断
mysql判断语句 1、IF 和IFNULL IF(表达式1,表达式2,表达式3); 含义:如果表达式1为true,则返回表达式2的值,否则返回表达式3的值,表达式的值类型可以为数字或字符串 例:判断对错 SELECT IF(TRUE…...

C++类与对象(上)【详析】
目录1.面向过程和面向对象初步认识2.类的引入3.类的定义4.类的访问限定符及封装4.1访问限定符4.2封装5.类的作用域6.类的实例化7.类对象模型7.1 如何计算类对象的大小8.this关键字如果说我们对C的初步认识,是觉得C是对C语言不足之处的进行修补,在认识完类…...

AIR系列|板载LED|gpio引脚选择|GPIO|流水灯|LuatOS-SOC接口|官方demo|学习(20-1):GPIO库基础
AIR系列各型号开发板板载LED对应管脚及GPIO控制代码 AIR103: rtos_bsp "AIR103" then -- Air103开发板LED引脚编号--return pin.PB26, pin.PB25, pin.PB24return 42,41,40 AIR105: rtos_bsp "AIR105" then -- Air105开发板LED引…...

MySQL数据库中的函数怎样使用?
函数 是指一段可以直接被另一段程序调用的程序或代码。 也就意味着,这一段程序或代码在MySQL中已经给我们提供了,我们要做的就是在合适的业务场景调用对应的函数完成对应的业务需求即可。 那么,函数到底在哪儿使用呢?我们先来看两个场景&…...
命名空间的使用大全
概述 在C中,我们会使用变量、常量、函数、类、对象、结构体等各种元素。随着工程越来越庞大,代表这些元素的标识符冲突的概率也越来越大。为了解决标识符命名冲突的问题,C标准在1995年引入了关键字namespace,也叫做命名空间。使用…...
uniapp 对接腾讯云IM群组成员管理(增删改查)
UniApp 实战:腾讯云IM群组成员管理(增删改查) 一、前言 在社交类App开发中,群组成员管理是核心功能之一。本文将基于UniApp框架,结合腾讯云IM SDK,详细讲解如何实现群组成员的增删改查全流程。 权限校验…...

基于Flask实现的医疗保险欺诈识别监测模型
基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施,由雇主和个人按一定比例缴纳保险费,建立社会医疗保险基金,支付雇员医疗费用的一种医疗保险制度, 它是促进社会文明和进步的…...
工程地质软件市场:发展现状、趋势与策略建议
一、引言 在工程建设领域,准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具,正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...

【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...

令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...
laravel8+vue3.0+element-plus搭建方法
创建 laravel8 项目 composer create-project --prefer-dist laravel/laravel laravel8 8.* 安装 laravel/ui composer require laravel/ui 修改 package.json 文件 "devDependencies": {"vue/compiler-sfc": "^3.0.7","axios": …...

九天毕昇深度学习平台 | 如何安装库?
pip install 库名 -i https://pypi.tuna.tsinghua.edu.cn/simple --user 举个例子: 报错 ModuleNotFoundError: No module named torch 那么我需要安装 torch pip install torch -i https://pypi.tuna.tsinghua.edu.cn/simple --user pip install 库名&#x…...
服务器--宝塔命令
一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行! sudo su - 1. CentOS 系统: yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...
在Ubuntu24上采用Wine打开SourceInsight
1. 安装wine sudo apt install wine 2. 安装32位库支持,SourceInsight是32位程序 sudo dpkg --add-architecture i386 sudo apt update sudo apt install wine32:i386 3. 验证安装 wine --version 4. 安装必要的字体和库(解决显示问题) sudo apt install fonts-wqy…...

JVM 内存结构 详解
内存结构 运行时数据区: Java虚拟机在运行Java程序过程中管理的内存区域。 程序计数器: 线程私有,程序控制流的指示器,分支、循环、跳转、异常处理、线程恢复等基础功能都依赖这个计数器完成。 每个线程都有一个程序计数…...