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

【数据结构】动态数组(vector)的基本操作,包括插入、删除、扩容、输出、释放内存等。以下是代码的解释和注释:

这段C代码实现了一个动态数组(vector)的基本操作,包括插入、删除、扩容、输出、释放内存等。以下是代码的解释和注释:

// 引入标准输入输出库和标准库函数,用于后续的内存分配和打印输出等操作  
#include <stdio.h>  
#include <stdlib.h>  // 引入时间库,用于生成随机数(这里并未使用,但保留了引入头文件)  
#include <time.h>  // 定义一个名为vector的结构体,该结构体有三个成员:size表示数组的大小,count表示数组中元素的数量,data是一个指向整型数组的指针,存储数组中的元素  
typedef struct vector {  int size, count;  int *data;  
} vector;  // getNewVector函数用于创建一个新的动态数组,并返回其指针  
vector *getNewVector(int n) {  // 使用malloc函数为vector结构体分配内存  vector *p = (vector *)malloc(sizeof(vector));  // 设置新创建的vector的大小为n,元素数量为0,并为data指针分配n个int类型的内存空间  p->size = n;  p->count = 0;  p->data = (int *)malloc(sizeof(int) * n);  // 返回新创建的vector的指针  return p;  
}  // expand函数用于扩容动态数组,将数组的大小翻倍  
int expand(vector *v) {  // 检查传入的指针是否为空,如果为空则返回0  if (v == NULL) return 0;  // 打印一条消息表示开始扩容  printf("expand v from %d to %d\n", v->size, 2 * v->size);  // 使用realloc重新分配足够的内存来存储int类型的2n个元素,并将这些元素的地址赋值给data指针  int *p = (int *)realloc(v->data, sizeof(int) * 2 * v->size);  // 如果realloc失败(返回NULL),则返回0;否则,将新分配的内存地址赋值给data,将数组的大小乘以2,并返回1表示扩容成功  if (p == NULL) return 0;  v->data = p;  v->size *= 2;  return 1;  
}  // insert函数用于在动态数组的指定位置插入一个元素  
int insert(vector *v, int pos, int val) {  // 检查插入的位置是否合法,如果不合法则返回0  if (pos < 0 || pos > v->count) return 0;  // 检查数组是否需要扩容,如果需要扩容但是扩容失败则返回0  if (v->size == v->count && !expand(v)) return 0;  // 从数组的末尾开始向前遍历每个元素,将每个元素向后移动一个位置  for (int i = v->count - 1; i >= pos; i--) {  v->data[i + 1] = v->data[i];  }  // 在指定的位置插入新的元素  v->data[pos] = val;  // 将元素数量加1,然后返回1表示插入成功  v->count += 1;  return 1;  
}  // erase函数用于从动态数组中删除指定位置的元素  
int erase(vector *v, int pos) {  // 检查删除的位置是否合法,如果不合法则返回0  if (pos < 0 || pos >= v->count) return 0;  // 从删除位置的下一个位置开始遍历每个元素,将每个元素向前移动一个位置  for (int i = pos + 1; i < v->count; i++) {  v->data[i - 1] = v->data[i];  }  // 将元素数量减1,然后返回1表示删除成功  v->count -= 1;  return 1;  
}  // 定义一个名为output_vector的函数,它接受一个指向vector结构体的指针作为参数  
void output_vector(vector* v) {  // 初始化一个整型变量len,用于存储要输出的整数之和  int len = 0;  // 遍历vector的大小(即其可以容纳的元素数量)  for (int i = 0; i < v->size; i++) {  // 在每次循环中,将整数i的值加到len上,同时输出i的值(格式化为三个数字宽)  len += printf("%3d", i);  }  // 输出一个换行符  printf("\n");  // 根据前面输出的整数数量,输出相应数量的短横线(-)以形成一个框架  for (int i = 0; i < len; i++) printf("-");  // 再输出一个换行符  printf("\n");  // 遍历vector中的元素(只遍历实际存在的元素,即count个)  for (int i = 0; i < v->count; i++) {  // 输出vector中第i个元素的值(格式化为三个数字宽)  printf("%3d", v->data[i]);  }  // 输出一个换行符  printf("\n");  // 输出两个空行,可能是为了创建视觉分隔或提供一些视觉清晰度  printf("\n\n");  // 函数结束,返回无值(void)  return;  
} // 释放动态数组内存  
void clear(vector *v) {  // 如果传入的指针为NULL,则直接返回,不进行任何操作  if (v == NULL) return ;  // 释放data指针指向的内存空间  free(v->data);  // 释放v指针指向的内存空间  free(v);  // 返回  return ;  
}  // 程序从main函数开始执行  
int main() {  // 使用当前时间作为随机数生成器的种子,这样可以使得每次运行程序时生成的随机数都不同  srand(time(0));  // 定义常量MAX_OP为20,表示要进行的操作次数  #define MAX_OP 20  // 调用getNewVector函数创建一个新的动态数组,并返回指向该数组的指针,数组的大小为2  vector *v = getNewVector(2);  // 执行MAX_OP次循环,每次循环都会随机生成一个操作和相应的参数  for (int i = 0; i < MAX_OP; i++) {  // 生成一个介于0到3之间的随机数,这个随机数将用于决定执行哪种操作  int op = rand() % 4, pos, val, ret;  // 根据随机数决定执行哪种操作  switch (op) {  // 如果操作是0、1或2,表示要进行插入操作  case 0:  case 1:  case 2:  // 生成一个介于0到(数组大小+1)之间的随机数,作为插入位置  pos = rand() % (v->count + 2);  // 生成一个介于0到99之间的随机数,作为要插入的值  val = rand() % 100;  // 调用insert函数进行插入操作,并把返回值保存在ret变量中  ret = insert(v, pos, val);  // 输出插入操作的信息,包括插入的值、插入的位置以及插入是否成功的返回值  printf("insert %d at %d to vector = %d\n", val, pos, ret);  break;  // 如果操作是3,表示要进行删除操作  case 3:  // 生成一个介于0到(数组大小+1)之间的随机数,作为删除位置  pos = rand() % (v->count + 2);  // 调用erase函数进行删除操作,并把返回值保存在ret变量中  ret = erase(v, pos);  // 输出删除操作的信息,包括删除位置以及删除是否成功的返回值  printf("erase item at %d in vector = %d\n", pos, ret);  break;  }  // 输出当前动态数组的内容  output_vector(v);  }  // 调用clear函数释放动态数组所占用的内存空间  clear(v);  // 程序正常结束,返回0  return 0;  
}

这段代码实现了一个简单的动态数组(vector),包含插入、删除和打印数组元素的功能。下面是各个函数的功能解释:

  1. getNewVector(int n):这个函数创建了一个新的动态数组,并为其分配了指定数量的整数存储空间。它返回一个指向新创建的动态数组的指针。
  2. expand(vector *v):这个函数用于将动态数组的存储空间扩大一倍。如果当前的存储空间已经足够,那么它什么都不做。否则,它会使用 realloc 函数来重新分配两倍于当前大小的存储空间,并将旧的数据复制到新的存储空间。如果扩大存储空间失败,它会返回0,否则返回1。
  3. insert(vector *v, int pos, int val):这个函数在动态数组中插入一个新的元素。它首先检查插入的位置是否有效,然后检查是否需要扩大存储空间。如果需要扩大存储空间且扩大操作失败,它会返回0。否则,它会将数组中的所有元素向后移动一位,然后在指定的位置插入新的元素。最后,它会返回1表示插入成功。
  4. erase(vector *v, int pos):这个函数从动态数组中删除一个元素。它首先检查删除的位置是否有效,然后删除元素并返回1表示删除成功。
  5. output_vector(vector *v):这个函数打印动态数组的所有元素和它们的位置(包括空位)。
  6. clear(vector *v):这个函数释放动态数组的内存空间。

相关文章:

【数据结构】动态数组(vector)的基本操作,包括插入、删除、扩容、输出、释放内存等。以下是代码的解释和注释:

这段C代码实现了一个动态数组&#xff08;vector&#xff09;的基本操作&#xff0c;包括插入、删除、扩容、输出、释放内存等。以下是代码的解释和注释&#xff1a; // 引入标准输入输出库和标准库函数&#xff0c;用于后续的内存分配和打印输出等操作 #include <stdio.…...

[unity]三角形顶点顺序

序 详见官方文档&#xff1a;Unity - Manual: Mesh data (unity3d.com) Topology&#xff1a;拓扑结构 翻译&#xff1a; 拓扑描述网格具有的面类型。 网格的拓扑定义了索引缓冲区的结构&#xff0c;索引缓冲区又描述了顶点位置如何组合成面。每种类型的拓扑都使用索引数组中…...

【python爬虫】14.Scrapy框架讲解

文章目录 前言Scrapy是什么Scrapy的结构Scrapy的工作原理 Scrapy的用法明确目标与分析过程代码实现——创建项目代码实现——编辑爬虫代码实现——定义数据代码实操——设置代码实操——运行 复习 前言 前两关&#xff0c;我们学习了能提升爬虫速度的进阶知识——协程&#xf…...

功率放大器主要作用是什么呢

功率放大器是一种电子设备&#xff0c;主要作用是将输入信号的功率增加到更高的水平&#xff0c;以便能够驱动高功率负载。在许多应用中&#xff0c;信号源产生的信号往往具有较低的功率&#xff0c;无法直接满足一些要求较高的设备或系统的需求。而功率放大器则可以增强信号的…...

SpringBoot ApplicationEvent详解

ApplicationStartingEvent 阶段 LoggingApplicationListener#onApplicationStartingEvent 初始化日志工厂,LoggingSystemFactory接口&#xff0c;可以通过spring.factories进行定制 可以通过System.setProperty("org.springframework.boot.logging.LoggingSystem",&q…...

WebSocket 报java.io.IOException: 远程主机强迫关闭了一个现有的连接。

在客户端强制关闭时&#xff0c;或者窗口强制关闭时&#xff0c;后端session没有关闭。 有时还会报&#xff1a;java.io.EOFException: 这个异常 前端心跳没有收到信息&#xff0c;还在心跳。 CloseReason close new CloseReason(CloseReason.CloseCodes.NORMAL_CLOSURE, &…...

关于git约定式提交IDEA

背景 因为git提交的消息不规范导致被乱喷&#xff0c;所以领导统一规定了约定式提交 官话 约定式提交官网地址 约定式提交规范是一种基于提交信息的轻量级约定。 它提供了一组简单规则来创建清晰的提交历史&#xff1b; 这更有利于编写自动化工具。 通过在提交信息中描述功能…...

【计算机网络】http协议

目录 前言 认识URL URLEncode和URLDecode http协议格式 http方法 GET POST GET与POST的区别 http状态码 http常见header 简易的http服务器 前言 我们在序列化和反序列化这一章中&#xff0c;实现了一个网络版的计算器。这个里面设计到了对协议的分析与处…...

仓库太大,clone 后,git pull 老分支成功,最新分支失败

由于 git 仓库太大&#xff0c;新加入的小伙伴在拉取时&#xff0c;无法切换到最新的分支&#xff0c;报错如下&#xff1a; fetch-pack: unexpected disconnect while reading sideband packet fatal: early EOF fatal: fetch-pack: invalid index-pack output在此记录解决步…...

javafx Dialog无法关闭

// 生成二维码图片String qrCodeText "https://example.com";DialogPane grid new DialogPane();grid.setPadding(new Insets(5));VBox vBox new VBox();vBox.setAlignment(Pos.CENTER);Image qrCodeImage generateQRCodeImage(qrCodeText);ImageView customImag…...

vue3中TCplayer应用

环境win10:vitevue3elementUI 1 安装 npm install tcplayer.js2 使用 <template><div><video id"player-container-id" width"414" height"270" preload"auto" playsinline webkit-playsinline></video>&l…...

算法通关村14关 | 数据流中位数问题

1. 数据流中位数问题 题目 LeetCode295: 中位数是有序列表中间的数&#xff0c;如果列表长度是偶数&#xff0c;中位数是中间两个数的平均值&#xff0c; 例如:[2,3,4]的中位数是3&#xff0c; [2,3]中位数是&#xff08;23&#xff09;/ 2 2.5 设计一个数据结构&#xff1a; …...

工厂模式 与 抽象工厂模式 的区别

工厂模式&#xff1a; // 抽象产品接口 interface Product {void showInfo(); }// 具体产品A class ConcreteProductA implements Product {Overridepublic void showInfo() {System.out.println("This is Product A");} }// 具体产品B class ConcreteProductB impl…...

安装虚拟机+安装/删除镜像

安装虚拟机 注意&#xff0c;官网可能无法登录&#xff0c;导致无法从官网下载&#xff0c;就自己去网上搜靠谱的下载&#xff0c;我用的16.2.3 删除镜像 Vm虚拟机怎么删除已经创建的系统&#xff1f;Vm虚拟机创建好之后iso删除方法 - 系统之家 (xitongzhijia.net) 安装镜像…...

MySQL的内置函数复合查询内外连接

文章目录 内置函数时间函数字符串函数数学函数其他函数 复合查询多表笛卡尔积自连接在where中使用子查询多列子查询在from中使用子查询 内连接外连接左外连接右外连接 内置函数 时间函数 函数描述current_date()当前日期current_time()当前时间current_timestamp()当前时间戳…...

操作系统(OS)与系统进程

操作系统&#xff08;OS&#xff09;与系统进程 冯诺依曼体系结构操作系统(Operator System)进程基本概念进程的描述&#xff08;PCB&#xff09;查看进程通过系统调用获取进程标示符&#xff08;PID&#xff09;通过系统调用创建进程&#xff08;fork&#xff09;进程状态&…...

防重复提交:自定义注解 + 拦截器(HandlerInterceptor)

防重复提交&#xff1a;自定义注解 拦截器&#xff08;HandlerInterceptor&#xff09; 一、思路&#xff1a; 1、首先自定义注解&#xff1b; 2、创建拦截器实现类&#xff08;自定义类名称&#xff09;&#xff0c;拦截器&#xff08;HandlerInterceptor&#xff09;; 3…...

Excel中将文本格式的数值转换为数字

在使用excel时&#xff0c;有时需要对数字列进行各种计算&#xff0c;比如求平均值&#xff0c;我们都知道应该使用AVERAGE()函数&#xff0c;但是很多时候结果却“不尽如人意”。 1 问题&#xff1a; 使用AVERAGE函数&#xff1a; 结果&#xff1a; 可以看到单元格左上角有个…...

uni-app开发小程序中遇到的map地图的点聚合以及polygon划分区域问题

写一篇文章来记录以下我在开发小程序地图过程中遇到的两个小坑吧&#xff0c;一个是点聚合&#xff0c;用的是joinCluster这个指令&#xff0c;另一个是polygon在地图上划分多边形的问题&#xff1a; 1.首先说一下点聚合问题&#xff0c;由于之前没有做过小程序地图问题&#…...

【笔记】软件测试的艺术

软件测试的心理学和经济学 测试是为发现错误而执行程序的过程&#xff0c;所以它是一个破坏性的过程&#xff0c;测试是一个“施虐”的过程。 软件测试的10大原则 1、测试用例需要对预期输出的结果有明确的定义 做这件事的前提是能够提前知晓需求和效果图&#xff0c;如果不…...

Docker 离线安装指南

参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性&#xff0c;不同版本的Docker对内核版本有不同要求。例如&#xff0c;Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本&#xff0c;Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...

零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?

一、核心优势&#xff1a;专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发&#xff0c;是一款收费低廉但功能全面的Windows NAS工具&#xff0c;主打“无学习成本部署” 。与其他NAS软件相比&#xff0c;其优势在于&#xff1a; 无需硬件改造&#xff1a;将任意W…...

基于Flask实现的医疗保险欺诈识别监测模型

基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施&#xff0c;由雇主和个人按一定比例缴纳保险费&#xff0c;建立社会医疗保险基金&#xff0c;支付雇员医疗费用的一种医疗保险制度&#xff0c; 它是促进社会文明和进步的…...

学习STC51单片机31(芯片为STC89C52RCRC)OLED显示屏1

每日一言 生活的美好&#xff0c;总是藏在那些你咬牙坚持的日子里。 硬件&#xff1a;OLED 以后要用到OLED的时候找到这个文件 OLED的设备地址 SSD1306"SSD" 是品牌缩写&#xff0c;"1306" 是产品编号。 驱动 OLED 屏幕的 IIC 总线数据传输格式 示意图 …...

鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个生活电费的缴纳和查询小程序

一、项目初始化与配置 1. 创建项目 ohpm init harmony/utility-payment-app 2. 配置权限 // module.json5 {"requestPermissions": [{"name": "ohos.permission.INTERNET"},{"name": "ohos.permission.GET_NETWORK_INFO"…...

【Java学习笔记】BigInteger 和 BigDecimal 类

BigInteger 和 BigDecimal 类 二者共有的常见方法 方法功能add加subtract减multiply乘divide除 注意点&#xff1a;传参类型必须是类对象 一、BigInteger 1. 作用&#xff1a;适合保存比较大的整型数 2. 使用说明 创建BigInteger对象 传入字符串 3. 代码示例 import j…...

Linux 内存管理实战精讲:核心原理与面试常考点全解析

Linux 内存管理实战精讲&#xff1a;核心原理与面试常考点全解析 Linux 内核内存管理是系统设计中最复杂但也最核心的模块之一。它不仅支撑着虚拟内存机制、物理内存分配、进程隔离与资源复用&#xff0c;还直接决定系统运行的性能与稳定性。无论你是嵌入式开发者、内核调试工…...

Netty从入门到进阶(二)

二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架&#xff0c;用于…...

Webpack性能优化:构建速度与体积优化策略

一、构建速度优化 1、​​升级Webpack和Node.js​​ ​​优化效果​​&#xff1a;Webpack 4比Webpack 3构建时间降低60%-98%。​​原因​​&#xff1a; V8引擎优化&#xff08;for of替代forEach、Map/Set替代Object&#xff09;。默认使用更快的md4哈希算法。AST直接从Loa…...

区块链技术概述

区块链技术是一种去中心化、分布式账本技术&#xff0c;通过密码学、共识机制和智能合约等核心组件&#xff0c;实现数据不可篡改、透明可追溯的系统。 一、核心技术 1. 去中心化 特点&#xff1a;数据存储在网络中的多个节点&#xff08;计算机&#xff09;&#xff0c;而非…...