顺序表是一种常见的数据结构,今天就让我来带领大家一起学习一下吧!
不会再休息,一分一毫了,OK,let’s go!
顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存
储。在数组上完成数据的增删查改。
顺序表一般可以分为:
//顺序表的静态存储
#define N 7
typedef int SLDataType;
typedef struct SeqList
{
SLDataType array[N];//定长数组
size_t size;//有效数据的个数
}SeqList;
typedef struct SeqList
{
SLDataType* array;//指向动态开辟的数组
size_t size;//有效数据的个数
size_t capacity;//容量
}SeqList;
(1)动态增容有性能消耗。
(2)当头部插入数据时,需要挪动数据
#pragma once
#include
#include
#include
typedef int SLDataType;
typedef struct SeqList
{
SLDataType* s;//顺序表的名称(头指针!)
int size;//储存的有效个数!
int capacity;//整块空间的大小!
}SL;
//初始化
void SLInit(SL* ps);
//销毁
void SLDestory(SL* ps);
//打印
void SLPrint(SL* ps);
//管理数据:增删查改
//尾插
void PushBack(SL* ps, SLDataType x);
//头插
void PushFront(SL* ps, SLDataType x);
//尾删
void PopBack(SL* ps);
//头删
void PopFront(SL* ps);
//判断是否扩容
void SLCheckCapacity(SL* ps);
//在pos位置之前插入数据
void SLInsert(SL* ps, int pos, SLDataType x);
#define _CRT_SECURE_NO_WARNINGS 1
#include
#include "SeqList.h"
//初始化函数
void SLInit(SL* ps)
{
assert(ps);
//创建空间
ps->s = (SLDataType*)malloc(sizeof(SLDataType)*4);
if (ps->s == NULL)
{
perror("malloc fail");
exit(-1);
}
ps->size = 0;
ps->capacity = 4;
}
//销毁函数
void SLDestory(SL* ps)
{
free(ps);
ps->s = NULL;
ps->size = ps->capacity = 0;
}
//打印函数
void SLPrint(SL* ps)
{
int i = 0;
for (i = 0; i < ps->size; i++)
{
printf("%d ", ps->s[i]);
}
printf("\n");
}
//判断是否扩容
void SLCheckCapacity(SL* ps)
{
assert(ps);
if (ps->size == ps->capacity)
{
//需要扩容
SLDataType* tmp = (SLDataType*)realloc(ps->s, sizeof(SLDataType) * ps->capacity * 2);//扩大了原来容量的二倍。
//SLDataType* tmp = (SLDataType*)realloc(ps->s, 2 * ps->capacity);标准的错误写法!
//如果空间不够用,要对一些元素进行扩容。我们扩容的标准:就是为这些元素申请它 自身大小 整数倍 的空间!所以说为什么要sizeof(数据类型),然后再乘以扩大的容量的倍数
if (tmp == NULL)
{
perror("realloc fail");
exit(-1);
}
ps->s = tmp;
ps->capacity *= 2;
}
}
//尾插
void PushBack(SL* ps, SLDataType x)
{
//检查容量
SLCheckCapacity(ps);
ps->s[ps->size] = x;
ps->size++;
}
//尾删
void PopBack(SL* ps)
{
assert(ps);
ps->size--;
}
//头插(利用一个end指针从后往前拷贝!)
void PushFront(SL* ps, SLDataType x)
{
assert(ps);
//检查容量
SLCheckCapacity(ps);
int end = ps->size - 1;
while (end >= 0)
{
ps->s[end+1] = ps->s[end];
end--;
}
ps->s[0] = x;
ps->size++;
}
//头删
void PopFront(SL* ps)
{
assert(ps);
int begin = 0;
while (begin < ps->size-1)
{
ps->s[begin] = ps->s[begin + 1];
begin++;
}
ps->size--;
}
//在pos位置之前插入数据
void SLInsert(SL* ps, int pos, SLDataType x)
{
assert(ps);
assert(pos >= 0 && pos < ps->size);
SLCheckCapacity( ps);
int end = ps->size - 1;
while (end >= pos)
{
ps->s[end+1] = ps->s[end];
end--;
}
ps->s[pos] = x;
ps->size++;
}
#define _CRT_SECURE_NO_WARNINGS 1
#include
#include "SeqList.h"
void test1()
{
SL s1;
SLInit(&s1);
PushFront(&s1, 1);
PushFront(&s1, 2);
PushFront(&s1, 3);
PushFront(&s1, 4);
PushFront(&s1, 5);
SLPrint(&s1);
PopFront(&s1);
PopFront(&s1);
PopFront(&s1);
SLPrint(&s1);
}
void test2()
{
SL s2;
SLInit(&s2);
PushBack(&s2,1);
PushBack(&s2,2);
PushBack(&s2,3);
PushBack(&s2,4);
PushBack(&s2,5);
SLPrint(&s2);
PopBack(&s2);
PopBack(&s2);
PopBack(&s2);
SLPrint(&s2);
}
void test3()
{
SL s2;
SLInit(&s2);
PushBack(&s2, 1);
PushBack(&s2, 2);
PushBack(&s2, 3);
PushBack(&s2, 4);
PushBack(&s2, 5);
SLPrint(&s2);
SLInsert(&s2, 3, 6);//在下标为3的数据之前插入一个6
SLPrint(&s2);
}
int main()
{
//test1();//测试头插,头删
//test2();//测试尾插 尾删
test3();//测试在pos位置之前插入数据!
return 0;
}
1.原地移除数组中所有的元素val,要求时间复杂度为O(N),空间复杂度为O(1)。OJ链接:OJ链接
int removeElement(int* nums, int numsSize, int val)
{
int src=0;
int dst=0;
while(src<numsSize)
{
if(nums[src]!=val)
{
nums[dst++]=nums[src++];
}
else
{
src++;
}
}
return dst;//返回的是:新数组的长度,因为最后一步出循环的时候,dst已经++了,所以说直接返回dst就行了
}
int removeDuplicates(int* nums, int numsSize)
{
int dst=1;
int src=0;
while(dst<numsSize)
{
if(nums[dst]!=nums[src])
{
//nums[++src]=nums[dst++];//这里的src一定要是前置++,先++,然后再赋值。
src++;
nums[src]=nums[dst];
dst++;
}
else
{
dst++;
}
}
return src+1;
}
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n)
{
int end1=m-1;
int end2=n-1;
int end=m+n-1;
while(end1>=0&&end2>=0)
{
if(nums1[end1]>nums2[end2])
{
nums1[end--]=nums1[end1--];
}
else
{
nums1[end--]=nums2[end2--];
}
}
//因为用nums1的初始长度是m+n,所以不会担心数组大小不够用。
//下面这个循环是针对:比如说nums1中的所有数字都插到自己数组后面了,但是因为两个数组都是有序的,所以我只需要把nums2中的全部数字依次放到nums1前面就行了。
while(end2>=0)
{
nums1[end--]=nums2[end2--];
}
}
好了,今天的分享就到这里了
如果对你有帮助,记得点赞+关注哦!
我的主页还有其他文章,欢迎学习指点。关注我,让我们一起学习,一起成长吧!