c语言 线性表

线性表的本质是数组,是在内存的连续开辟的一块空间。所以在物理上是连续的,逻辑上也是。而链表只是在逻辑上连续。

但不同于数组,线性表的空间是动态开辟的,能随时调整线性表的大小,而数组的大小则是固定的。

1.结构体定义与初始化

typedef int SListDataType;

typedef struct SeqList
{
	SListDataType* a;
	int sz;
	int capacity;
};

线性表结构中有一个指针,指向的是动态开辟的空间,sz表示当前指针指向的空间中数据的数量,capac表示动态开辟空间最大的储存数量。

 void SListCapacityCheck(SList* ps)
{
	assert(ps);

	if (ps->sz == ps->capacity)
	{
		ps->capacity *= 2;
		ps->a = (SListDataType*)realloc(ps->a, sizeof(SList) * ps->capacity);
	}
}

2.尾插与尾删

在插入数据前先检查sz释放等于capacity,如果相等则需要再次申请内存,每次申请的内存是之前的两倍。然后将ps指向的a数组下标为ps指向的sz的元素赋值。ps指向的sz要+1。

void SListPushBack(SList* ps, SListDataType x)
{
	assert(ps);

	SListCapacityCheck(ps);

	ps->a[ps->sz] = x;
	ps->sz++;
}

尾删只要将ps指向的sz-1就行了

void SListPopBack(SList* ps)
{
	assert(ps);
	assert(ps->sz >= 0);

	ps->sz--;
}

3.头插与头删

往数组下标为0处插入数据,需要将后面的数据移动。还要确保数组的空间是足够的。

void SListPushFront(SList* ps, SListDataType x)
{
	assert(ps);

	SListCapacityCheck(ps);

	int end = ps->sz - 1;
	while (end >= 0)
	{
		ps->a[end + 1] = ps->a[end];
		end--;
	}

	ps->a[0] = x;
	ps->sz++;
}

头删只要把后面的数据先前覆盖,sz–

void SListPopFront(SList* ps)
{
	assert(ps);
	
	if (ps->sz > 0)
	{
		int start = 1;
		while (start <= ps->sz - 1)
		{
			ps->a[start - 1] = ps->a[start];
			start++;
		}
		ps->sz--;
	}
}

4.随机位置的插入与删除

往表中添加数据首先要先检查容量是否足够,然后将数据往后移动,再将数据插入。

void SListInsert(SList* ps, size_t pos, SListDataType x)//在pos的后面插入数据
{
	assert(ps);
	assert(pos <= ps->sz);

	SListCapacityCheck(ps);
	size_t end = ps->sz;
	while (end > pos)
	{
		ps->a[end] = ps->a[end - 1];
		end--;
	}

	ps->a[pos] = x;
	ps->sz++;
}

当pos的类型是无符号时,不能用end >= pos这个判断条件。当end为-1时,判断条件仍成立,程序进入死循环。用end > pos,挪动数据用end - 1 = end

删除即将pos后的数据往前覆盖,sz–。这里数据的移动要将前面的数据往前移动。

void SListErase(SList* ps, int pos)
{
	assert(ps);
	assert(pos >= 0 && pos < ps->sz);

	int start = pos;
	while (start < ps->sz - 1)
	{
		ps->a[start] = ps->a[start + 1];
		start++;
	}
	ps->sz--;
}

5.查找元素,返回其下标

找到返回下标,否则返回-1

int SListFind(SList* ps, SListDataType x)
{
	assert(ps);

	for (int i = 0; i < ps->sz; i++)
	{
		if (ps->a[i] == x)
			return i;
	}
	return -1;
}

6.线性表的销毁

void SListDestory(SList* ps)
{
	assert(ps);
	free(ps);
}

7.线性表与链表优缺点

优点:线性表可以随机访问元素
缓存命中率高(cpu访问数据时不是只拿一个数据到缓存中,而是拿从这个数据开始的一段数据到缓存中,要拿下个数据时,会先从缓存中找这个数据,如果没有命中就从内存中拿。由于线性表的数据都是连续存放的,所以缓存的命中率很高,但链表的数据都是在内存中随机存放,出现数据连续存放的情况较少,缓存命中率低)

缺点:中间或头部的数据插入速度慢,时间复杂度是O(n)
空间不够时,增容造成一定的空间浪费

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值