线性表的本质是数组,是在内存的连续开辟的一块空间。所以在物理上是连续的,逻辑上也是。而链表只是在逻辑上连续。
但不同于数组,线性表的空间是动态开辟的,能随时调整线性表的大小,而数组的大小则是固定的。
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)
空间不够时,增容造成一定的空间浪费