【初阶数据结构】- 顺序表

1. 顺序表的概念及结构

1.1 线性表

顺序表是一种线性表,线性表是n个具有相同特性的数据元素的有限序列,常见的线性表有:顺序表,链表,栈,队列
👉线性表的逻辑结构:在逻辑上是一条连续的直线。
👉线性表的物理结构:线性表在物理结构上并不是连续的,线性表在物理上存储时通常以数组链式的结构存储的。

2. 顺序表的分类

顺序表与数组的关系

      ◦ 顺序表的底层结构就是数组,通过对数组的封装,实现增删查改等功能。

• 顺序表的分类:
      ◦静态顺序表

//使用定长数组存储数据
typedef int SLDataType;
#define N 10
typedef struct SeqList
{
	SLDataType a[N];//定长数组
	int size;//有效元素的个数
}SL;

❗️缺点:空间是死的,短了不够用,长了会造成空间的浪费。
      ◦动态顺序表

//按需申请
typedef int SLDataType;
typedef struct SeqList
{
	SLDataType* arr;//动态数组。动态内存开辟,确定大小之后再去动态申请
	int size;//顺序表当前有效元素的个数
	int capacity; //顺序表的空间大小
}SL;

3. 动态顺序表的实现

3.1 头文件.h

顺序表结构的声明

#include<stdio.h>
#include<stdlib.h>//需要动态申请内存的头文件
#include<assert.h>
typedef int SLDataType;
// 动态顺序表 -- 按需申请 
typedef struct SeqList
{
 SLDataType* a;
 int size; // 有效数据个数 
 int capacity; // 空间容量 
}SL;
//初始化和销毁 
void SLInit(SL* ps);
void SLDestroy(SL* ps);
void SLPrint(SL* ps);
//扩容 
void SLCheckCapacity(SL* ps);
//头部插⼊删除 / 尾部插⼊删除 
void SLPushBack(SL* ps, SLDataType x);
void SLPopBack(SL* ps);
void SLPushFront(SL* ps, SLDataType x);
void SLPopFront(SL* ps);
//指定位置之前插⼊/删除数据 
void SLInsert(SL* ps, int pos, SLDataType x);
void SLErase(SL* ps, int pos);
int SLFind(SL* ps, SLDataType x);

3.2 .C文件

实现顺序表的方法

//顺序表的初始化
void SLInit(SL* ps)
{
	assert(ps);
	ps->arr = NULL;
	ps->size = ps->capacity = 0;
}
//顺序表的销毁
void SLDestroy(SL* ps)
{
	assert(ps);
	if (ps->arr)//如果arr不等于空
	{
		free(ps->arr);
	}
	ps->arr = NULL;
	ps->size = ps->capacity = 0;
}
//打印顺序表中的数据
void SLPrint(SL* ps)
{
	assert(ps);
	for (int i = 0; i < ps->size; i++)
	{
		printf("%d ",ps->arr[i]);
	}
	
}
//扩容 
void SLCheckCapacity(SL* ps)
{
	assert(ps);
	if (ps->size == ps->capacity)
	{
		//申请空间
		//malloc calloc realloc 因为要涉及增容,所以用realloc
		int newcapacity = ps->capacity == 0 ? 4 : 2* ps->capacity ;
		SLDataType* tmp = (SLDataType*)realloc(ps->arr, newcapacity * sizeof(SLDataType));//realloc申请的是字节大小,*(SLDtaType)整形大小
		if (tmp == NULL)
		{
			perror("realloc fail");//空间申请失败,报错
			exit(1);//直接退出程序,不再继续执行
		}
		//申请空间成功
		ps->arr = tmp;
		ps->capacity = newcapacity;
	}

}
//顺序表的尾部插入
void SLPushBack(SL* ps, SLDataType x)
{
	//温柔的方法
	/*if (ps == NULL)
	{
		return;
	}*/
	assert(ps);//暴力方法
	if(ps->size==ps->capacity)
	SLCheckCapacity(ps);
	//ps->arr[ps->size] = x;
	//ps->size++;
	ps->arr[ps->size++] = x;
}
//顺序表的尾删
void SLPopBack(SL* ps)
{
	assert(ps);
	assert(ps->size);
	--ps->size;
}
//顺序表的头插
void SLPushFront(SL* ps, SLDataType x)
{
	assert(ps);
	SLCheckCapacity(ps);
	for (int i = ps->size; i > 0; i--)
	{
		ps->arr[i] = ps->arr[i-1];
	}
	ps->arr[0] = x;
	ps->size++;
}
//头删
void SLPopFront(SL* ps)
{
	assert(ps);
	assert(ps->size);
	for (int i = 0; i<ps->size-1; i++)
	{
		ps->arr[i] = ps->arr[i+1];
	}
	ps->size--;
}
//在指定位置之前插入数据
void SLInsert(SL* ps, int pos, SLDataType x)
{
	assert(ps);
	assert(pos>=0 && pos<=ps->size);
	//插入数据之前看数据够不够
	SLCheckCapacity(ps);
	for (int i = ps->size; i > pos; i--)
	{
		ps->arr[i] = ps->arr[i-1];
	}
	ps->arr[pos] = x;
	ps->size++;

}
//在指定位置删除数据
void SLErase(SL* ps, int pos)
{
	assert(ps);
	assert(pos >= 0 && pos < ps->size);
	for (int i = pos; i < ps->size-1; i++)
	{
		ps->arr[i] = ps ->arr[i + 1];
	}
	ps->size--;
}
//查找顺序表中的数据
int SLFind(SL* ps, SLDataType x)
{
	assert(ps);
	for (int i = 0; i < ps->size; i++)
	{
		if (ps->arr[i] == x)
		{
			//找到了
			return i;
		}
	}
	return -1;
}

根据提供的引用内容,我们可以看出这是一个关于堆的实现和应用的问题。引用\[1\]提到了一种使用顺序表存储的方式来实现堆,但是这种方式存在空间浪费的问题。引用\[2\]列举了堆的接口函数和堆排序的过程。引用\[3\]介绍了一种常用且优化的表示方法,即左孩子右兄弟表示法。 根据问题描述,警告C6386是指在写入"popk"时发生了缓冲区溢出。根据提供的代码,问题出现在源文件的第64行。具体原因可能是在该行代码中,将数据写入了名为"popk"的缓冲区,但是该缓冲区的大小不足以容纳写入的数据,导致溢出。 为了解决这个问题,我们需要检查源文件中的相关代码,确保在写入缓冲区时不会超出其大小限制。可能需要调整缓冲区的大小或者使用更安全的写入方式来避免缓冲区溢出的问题。 #### 引用[.reference_title] - *1* *3* [二叉树第一弹之树和堆的概念和结构、基础堆接口函数的实现(编写思路加逻辑分析加代码实操,一应俱全的汇总...](https://blog.csdn.net/AMor_05/article/details/127175020)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [第九章:C语言数据结构与算法初阶之堆](https://blog.csdn.net/yanyongfu523/article/details/129582526)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值