【从0到1】数据结构线性表——顺序表

目录

1、线性表概念

2、顺序表的实现

2.1、概念及结构

静态顺序表

动态顺序表

3、Leetcode练习题

3.1、移除元素 

思路一:暴力方法

思路二:以时间换空间

思路三:双指针

 3.2、合并两个有序数组

思路一:合并的同时比较大小 

思路二:先合并后排序


1、线性表概念

        线性表是n个具有相同特性的数据元素的有限序列。线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串……

        线性表在逻辑上是线性结构,也就是说是连续的一条直线。但在物理结构上并不一定是连续的,线性表在物理上存储时,通常以数组和链表结构的形式存储。

2、顺序表的实现

2.1、概念及结构

        顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改

注:数据必须是从头开始,依次存储 

顺序表一般可以分为:

1、静态顺序表:使用定长数组存储(数组长度是确定的)

2、动态顺序表:使用动态开辟的数组存储

静态顺序表

简单代码实现如下:

头文件(SeqList.h)

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<assert.h>
//增强程序的可维护性
#define MAX_SIZE 10

typedef int SQDataType;
typedef struct SeqList
{
	SQDataType a[MAX_SIZE];
	int size;//记录实际长度
}SL;

//增删查改等接口函数的实现
//函数声明
void SeqListInit(SL* ps);//初始化
void SeqListPushBack(SL* ps,SQDataType x);//尾插
void SeqListPushFront(SL* ps, SQDataType x);//头插
void SeqListPopBack(SL* ps);//尾删
void SeqListPopFront(SL* ps);//头删

void SeqListPrint(SL* ps);//打印

测试文件(Test.c):

#define _CRT_SECURE_NO_WARNINGS
#include"SeqList.h"
//静态顺序表
//问题:给少了不够用,给多了用不完浪费,不能灵活控制

void menu()
{
	printf("************************************\n");
	printf("***1、尾插数据    2、头插数据******\n");
	printf("***3、尾删数据    4、头删数据******\n");
	printf("***5、打印数据    0、关闭系统******\n");
	printf("************************************\n");
	printf("请输入您要操作的选项:\n");
}
int main()
{
	int input = 0;
	int x = 0;
	int i = 0;
	SL s;
	SeqListInit(&s);
	do
	{
		menu();
		scanf("%d", &input);
		switch (input)
		{

		case 1:
			printf("请输入要插入的数据:");
			scanf("%d", &x);
			SeqListPushBack(&s, x);
			break;
		case 2:
			printf("请输入要插入的数据:");
			scanf("%d", &x);
			SeqListPushFront(&s, x);
			break;
		case 3:
			SeqListPopBack(&s);
			break;
		case 4:
			SeqListPopFront(&s);
			break;
		case 5:
			SeqListPrint(&s);
			break;
		case 0:
			printf("退出!");
			break;
		default:
			printf("输入有误,请重新输入\n");
			break;
		}
	} while (input);
	return 0;
}




具体函数实现文件(SeqList.c)

#define _CRT_SECURE_NO_WARNINGS
#include"SeqList.h"


void SeqListInit(SL* ps)
{
	assert(ps);
	memset(ps->a, 0, sizeof(SQDataType)*MAX_SIZE);
	ps->size = 0;
}

void SeqListPushBack(SL* ps, SQDataType x)//尾插
{
	assert(ps);
	if (ps->size == MAX_SIZE)
	{
		printf("SeqList is Full\n");
		return;
	}
	ps->a[ps->size] = x;
	ps->size++;
}
void SeqListPushFront(SL* ps, SQDataType x)//头插
{
	assert(ps);
	if (ps->size == MAX_SIZE)
	{
		printf("SeqList is Full\n");
		return;
	}
	int end = ps->size - 1;
	while (end > 0)
	{
		ps->a[end+1] = ps->a[end];
	}
	ps->a[0] = x;
	ps->size++;
}
void SeqListPopBack(SL* ps)//尾删
{
	assert(ps);
	ps->size--;
}
void SeqListPopFront(SL* ps)//头删
{
	assert(ps);
	int start = 1;
	while (start == ps->size - 1)
	{
		ps->a[start-1] = ps->a[start];
	}
	ps->size--;
}

void SeqListPrint(SL* ps)//打印
{
	int i = 0;
	for (i = 0; i < ps->size; i++)
	{
		printf("%d ", ps->a[i]);
	}
	printf("\n");
}

动态顺序表

简单代码实现如下:

头文件(SeqList.h)

#pragma once
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<assert.h>
//增强代码的可维护性
typedef int SQDataType;

typedef struct SeqList
{
	SQDataType* a;
	int size;//有效数据的个数
	int capacity;//容量
}SL;

//初始化
void SeqListInit(SL* ps);
void SeqListPushBack(SL* ps, SQDataType x);//尾插
void SeqListPushFront(SL* ps, SQDataType x);//头插
void SeqListPopBack(SL* ps);//尾删
void SeqListPopFront(SL* ps);//头删
//随机插入
void SeqListInsert(SL* ps, int pos, SQDataType x);
//随即删除
void SeqListEarse(SL* ps, int pos);
void SeqListDestory(SL* ps);//销毁

//查
 void SeqListFind(SL* ps,SQDataType x);

 //改
 void SeqListModify(SL* ps, int pos,SQDataType x);

 //打印
 void SeqListPrint(SL*ps);

测试文件(Test.c):

#define _CRT_SECURE_NO_WARNINGS
#include"SeqList.h"

void menu()
{
	printf("************************************\n");
	printf("***1、尾插数据    2、头插数据******\n");
	printf("***3、尾删数据    4、头删数据******\n");
	printf("***5、查找数据    6、修改数据******\n");
	printf("***7、打印数据    0、关闭系统******\n");
	printf("************************************\n");
	printf("请输入您要操作的选项:\n");
}
int main()
{
	int input = 0;
	int x = 0;
	int i = 0;
	SL s;
	SeqListInit(&s);
	do
	{
		menu();
		scanf("%d", &input);
		switch (input)
		{
			
		case 1:
			printf("请输入要插入的数据:");
			scanf("%d", &x);
			SeqListPushBack(&s,x);
			break;
		case 2:
			printf("请输入要插入的数据:");
			scanf("%d", &x);
			SeqListPushFront(&s,x);
			break;
		case 3:
			SeqListPopBack(&s,x);
			break;
		case 4:
			SeqListPopFront(&s);
			break;
		case 5:
			printf("请输入要查找的数据:");
			scanf("%d", &x);
			SeqListFind(&s,x);
			break;
		case 6:
			printf("请输入要修改的数据的位置:");
			scanf("%d", &i);
			printf("请输入要修改为:");
			scanf("%d", &x);
			SeqListModify(&s,i,x);
			break;
		case 7:
			SeqListPrint(&s);
			break;
		case 0:
			printf("退出!");
			break;
		default:
			printf("输入有误,请重新输入\n");
			break;
		}
	} while (input);
	SeqListDestory(&s);//销毁
	return 0;
}




 具体函数实现文件(SeqList.c)

#define _CRT_SECURE_NO_WARNINGS
#include"SeqList.h"


//初始化
void SeqListInit(SL* ps)
{
	ps->a = NULL;
	ps->size = 0;
	ps->capacity = 0;
}

//增容
void SeqListCheckCapacity(SL* ps)
{
	//满了,扩容
	if (ps->size == ps->capacity)
	{
		int newcapacity = ps->capacity == 0 ? 4 : ps->capacity;
		SQDataType* tmp = (SQDataType*)realloc(ps->a, newcapacity * sizeof(SQDataType));
		if (tmp == NULL)
		{
			printf("realloc fail\n");
			exit(-1);
		}
		else
		{
			ps->a = tmp;
			ps->capacity = newcapacity;
		}
	}
}

void SeqListPushBack(SL* ps, SQDataType x)//尾插
{
	assert(ps);
	//检查容量
	SeqListCheckCapacity(ps);
	//插入
	ps->a[ps->size] = x;
	ps->size++;
}

//销毁
void SeqListDestory(SL* ps)
{
	assert(ps);
	free(ps->a);
	ps->a = NULL;
	ps->capacity = 0;
	ps->size = 0;
}

void SeqListPushFront(SL* ps, SQDataType x)//头插
{
	assert(ps);
	//检查容量
	SeqListCheckCapacity(ps);
	//1、初始条件
	int end = ps->size - 1;
	while (end >= 0)//2、结束条件
	{
		//从后往前覆盖
		ps->a[end + 1] = ps->a[end];//3、迭代过程
		end--;
	}
	ps->a[0] = x;
	ps->size++;
}
void SeqListPopBack(SL* ps)//尾删
{
	assert(ps);
	assert(ps->size > 0);//粗暴
	ps->size--;

}
void SeqListPopFront(SL* ps)//头删
{
	assert(ps->size > 0);//粗暴
	//从前向后
	int start = 1;
	while (start < ps->size)
	{
		ps->a[start - 1] = ps->a[start];
		start++;
	}
	ps->size--;
}

//随机插入
void SeqListInsert(SL* ps, int pos, SQDataType x)
{
	assert(pos < ps->size);
	SeqListCheckCapacity(ps);
	int end = ps->size - 1;
	while (end >= pos)
	{
		ps->a[end + 1] = ps->a[end];
		end--;
	}
	ps->a[pos] = x;
	ps->size++;
}

//随机删除
void SeqListEarse(SL* ps, int pos)
{
	assert(pos < ps->size);
	int start = pos + 1;
	while (start < ps->size)
	{
		ps->a[start - 1] = ps->a[start];
		start++;
	}
	ps->size--;
}

//查找
void SeqListFind(SL* ps, SQDataType x)
{
	for (int i = 0; i < ps->size; i++)
	{
		if (ps->a[i] == x)
		{
			return i;
		}
	}
	return -1;
}

//改
void SeqListModify(SL* ps, int pos,SQDataType x)
{
	assert(ps);
	assert(pos < ps->size);
	ps->a[pos] = x;
}

//打印
void SeqListPrint(SL* ps)
{
	int i = 0;
	for (i = 0; i < ps->size; i++)
	{
		printf("%d ", ps->a[i]);
	}
	printf("\n");
}

顺序表的缺陷:

1、如果空间不够,增容。增容会付出一定性能的消耗,其次可能存在一定的空间浪费

2、头部或者中部的插入删除效率低 

3、Leetcode练习题

3.1、移除元素 

思路一:暴力方法

int removeElement1(int* nums, int numsSize, int val)
{
    int count = 0;//统计个数
    for (int i = 0; i < numsSize; i++)
    {
        if (nums[i] == val)
        {
            count++;
            for (int j = i; j < numsSize - 1; j++)
            {
                nums[j] = nums[j + 1];
            }
        }
    }
    return numsSize-count;
}

时间复杂度:O(N^2)

思路二:以时间换空间

把不是val的数据,放到新的数组中,再将新的数据拷贝到原来的数组中

int removeElement2(int* nums, int numsSize, int val)
{
    int arr[100] = { 0 };
    int j = 0;
    for (int i = 0; i < numsSize; i++)
    {
            if (val != nums[i])
            {
                arr[j++] = nums[i];
            }
    }
    for (int i = 0; i < j; i++)
    {
        nums[i] = arr[i];
    }
    return j;
}

时间复杂度:O(N)

空间复杂度:O(N)

思路三:双指针

在思路二上稍加改进,不放到新的数组中,而是放给自己原本的数组

典型的双指针

int removeElement(int* nums, int numsSize, int val){
    int dst=0;
    int src=0;
    while(src<numsSize)
    {
        if(nums[src]!=val)
        {
            nums[dst++]=nums[src++];
        }
        else
        {
            src++;
        }
    }
    return dst;
}

时间复杂度:O(N)

空间复杂度:O(1)

 3.2、合并两个有序数组

思路一:合并的同时比较大小 

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 if (nums1[end1] < nums2[end2])
        {
            nums1[end--] = nums2[end2--];
        }
    }
    while (end2 >= 0)
    {
        nums1[end--] = nums2[end2--];
    }
}

 注意:

        是从后往前放数据,为什么呢?两数组原本就是按照从小到大的顺序,从后往前放,防止数据被覆盖。

        第二个while是当nums1的数据被移完了,但是nums二中还有数据,并原本就是有序的,所以直接放到nums1中即可。 

 

思路二:先合并后排序

int cmp_int(int* e1,int* e2)
{
    return *e1-*e2;
}
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n) {
    int end=m;
    for(int i=0;i<n;i++)
    {
        nums1[end++]=nums2[i];
    }
    qsort(nums1,n+m,sizeof(nums1[0]),cmp_int);
}

对于qsort函数,http://t.csdn.cn/pXtFR 这一期有详细讲解。

下期见咯! 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

龙洋静

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值