C语言线性表中顺序表超详细理解

目录
  • 一、本章重点
  • 二、线性表
  • 三、顺序表
  • 四、静态顺序表接口实现
    • 4.1顺序表初始化
    • 4.2顺序表打印
    • 4.3顺序表尾插
    • 4.4顺序表尾删
    • 4.5顺序表头插
    • 4.6顺序表头删
    • 4.7顺序表任意位置插入
    • 4.8顺序表任意位置删除
  • 五、动态顺序表接口实现
    • 5.1顺序表的初始化
    • 5.2顺序表打印
    • 5.3顺序表尾插
    • 5.4顺序表尾删
    • 5.5顺序表头插
    • 5.6顺序表头删
    • 5.7顺序表任意位置插入
    • 5.8顺序表任意位置删除
  • 六、在线0j练习
    • 一、移除元素(力扣)
    • 二、合并两个有序数组(力扣)

一、本章重点

1.线性表和顺序表的概念

2.动态和静态顺序表接口实现

3.在线0j训练

二、线性表

满足下列条件的即为线性表:

  • 线性表(linear list)是n个具有相同特性的数据元素的有限序列。
  • 线性表在逻辑上是线性结构,但是在物理结构上并不一定是连续的。(这里的物理结构一般指物理地址空间)。

三、顺序表

满足下列条件的即为顺序表:

  • 是线性表
  • 物理结构上是连续的

顺序表一般可以分为:

  • 静态顺序表:使用定长数组存储。
  • 动态顺序表:使用动态开辟的数组存储。

四、静态顺序表接口实现

4.1顺序表初始化

void SeqListInint(SeqList* s)
{
	assert(s);
	memset(s->a, 0, sizeof(SeqListDataType) * MAXSIZE);
	s->size = 0;
}

还有一种简单初始化的方式:

在创建顺序表s的时候直接赋值0,即SeqList s = { 0 };

4.2顺序表打印

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

传顺序表的地址,使用for循环语句,逐步打印数组元素。

4.3顺序表尾插

void SeqListPushBack(SeqList* s, int x)
{
	assert(s);
	if (s->size == MAXSIZE)
	{
		printf("当前空间已满,无法继续添加\n");
		exit(1);
	}
	s->a[s->size] = x;
	s->size++;
}

先检查s是否位空,如果为空则报错,再检查是否满了,如果满了,则提示已满并结束程序。

4.4顺序表尾删

void SeqListPopBack(SeqList* s)
{
	assert(s);
	if (s->size == 0)
	{
		printf("当前顺序表为空,无法删除\n");
		exit(1);
	}
	s->size--;
}

直接s->size--即可,不需要把最后的元素置为0.

4.5顺序表头插

void SeqListPushFront(SeqList* s, int x)
{
	if (s->size == MAXSIZE)
	{
		printf("空间已满,无法继续添加\n");
		exit(1);
	}
	if (s->size == 0)
	{
		s->a[s->size] = x;
		s->size++;
		return;
	}
	else
	{
		int j = 0;
		for (j = s->size - 1; j >= 0; j--)
		{
			s->a[j + 1] = s->a[j];
		}
		s->a[0] = x;
		s->size++;
	}
}

先将元素往后移动,移动完之后再放入要插入的元素。

4.6顺序表头删

void SeqListPopFront(SeqList* s)
{
	if (s->size == 0)
	{
		printf("当前顺序表为空,无法删除\n");
		exit(1);
	}
	int j = 0;
	for (j = 1; j <s->size; j++)
	{
		s->a[j - 1] = s->a[j];
	}
	s->size--;
}

使用移动元素的方式,覆盖前面的内容,达到删除的目的。

4.7顺序表任意位置插入

void SeqListInsert(SeqList* s, int pos, int x)
{
	if (s->size == MAXSIZE)
	{
		printf("当前空间已满,无法继续添加\n");
		exit(1);
	}
	if (pos < 0||pos>s->size)
	{
		printf("插入位置有误,无法插入\n");
		exit(1);
	}
	if (pos == s->size)
	{
		s->a[s->size] = x;
		s->size++;
		return;
	}
	for (int j = s->size - 1; j >= pos; j--)
	{
		s->a[j + 1] = s->a[j];
	}
	s->a[pos] = x;
	s->size++;
}

找到元素位置,移动元素,再将要插入的元素放入。

4.8顺序表任意位置删除

void SeqListErase(SeqList* s, int pos)
{
	assert(s);
	if (s->size == 0)
	{
		printf("顺序表为空,删除失败\n");
		exit(1);
	}
	if (pos >= s->size || pos < 0)
	{
		printf("删除位置不存在\n");
		exit(1);
	}
	int j = 0;
	for (j = pos; j < s->size-1; j++)
	{
		s->a[j] = s->a[j + 1];
	}
	s->size--;
}

找到要删除的位置,通过移动覆盖要删除的元素。

五、动态顺序表接口实现

5.1顺序表的初始化

void SeqListInint(SeqList* s)
{
	assert(s);
	s->a = (DataType*)malloc(10 * sizeof(DataType));
	s->size = 0;
	s->capacity = 10;
}

将元素个数size置为0

开辟a的空间

初始容量设置为10

5.2顺序表打印

void SeqListPrint(SeqList* s)
{
	assert(s);
	int i = 0;
	for (i = 0; i < s->size; i++)
	{
		printf("%d ", s->a[i]);
	}
	printf("\n");
}

5.3顺序表尾插

void SeqListPushBack(SeqList* s, DataType x)
{
	assert(s);
	SeqListCheckCapacity(s);
	s->a[s->size] = x;
	s->size++;
}

5.4顺序表尾删

void SeqListPopBack(SeqList* s)
{
	assert(s);
	if (s->size == 0)
	{
		printf("当前顺序表为空,删除失败\n");
		exit(1);
	}
	s->size--;
}

5.5顺序表头插

void SeqListPushFront(SeqList* s, DataType x)
{
	assert(s);
	SeqListCheckCapacity(s);
	if (s->size == 0)
	{
		s->a[0] = x;
		s->size++;
	}
	else
	{
		int end = s->size - 1;
		while (end >= 0)
		{
			s->a[end + 1] = s->a[end];
			end--;
		}
		s->a[0] = x;
		s->size++;
	}
}

5.6顺序表头删

void SeqListPopFront(SeqList* s)
{
	assert(s);
	if (s->size == 0)
	{
		printf("当前顺序表为空,无法删除\n");
		exit(1);
	}
	if (s->size == 1)
	{
		s->size--;
		return;
	}
	else
	{
		int i = 0;
		for (i = 0; i <=s->size-2 ; i++)
		{
			s->a[i] = s->a[i + 1];
		}
		s->size--;
	}
}

5.7顺序表任意位置插入

void SeqListInsert(SeqList* s, int pos, DataType x)
{
	assert(s);
	SeqListCheckCapacity(s);
	if (pos<0 || pos>s->size)
	{
		printf("插入位置不存在\n");
		exit(1);
	}
	else if(pos==s->size)
	{
		s->a[s->size] = x;
		s->size++;
	}
	else
	{
		int i = 0;
		for (i = s->size - 1; i >= pos; i--)
		{
			s->a[i + 1] = s->a[i];
		}
		s->a[pos] = x;
		s->size++;
	}
}

5.8顺序表任意位置删除

void SeqListErase(SeqList* s, int pos)
{
	assert(s);
	if (s->size == 0)
	{
		printf("当前顺序表为空,删除失败\n");
		exit(1);
	}
	if (pos<0||pos>s->size-1)
	{
		printf("要删除的位置不存在\n");
		exit(1);
	}
	else
	{
		int i = 0;
		for (i = pos; i <= s->size - 2; i++)
		{
			s->a[i] = s->a[i + 1];
		}
		s->size--;
	}
}

六、在线0j练习

一、移除元素(力扣)

给你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素,并返回移除后数组的新长度。

不要使用额外的数组空间,你必须仅使用 O(1) 额外空间并 原地 修改输入数组。

元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。

例一:

输入:nums = [0,1,2,2,3,0,4,2], val = 2 输出:5, nums = [0,1,4,0,3] 解释:函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4。注意这五个元素可为任意顺序。你不需要考虑数组中超出新长度后面的元素。

思路:用两个指针,一个用来遍历数组,另一个指向你要存数据的地方。

如果可以申请额外的空间的话,一般来说,我们可以这样做:申请一个新的数组空间,用来存放非val值的数据。其实这个新的空间我们可以直接把nums数组原空间直接当做新空间使用,我们只需遍历一遍nums数组即可。

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

二、合并两个有序数组(力扣)

给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。

请你 合并 nums2 到 nums1 中,使合并后的数组同样按 非递减顺序 排列。

注意:最终,合并后数组不应由函数返回,而是存储在数组 nums1 中。为了应对这种情况,nums1 的初始长度为 m + n,其中前 m 个元素表示应合并的元素,后 n 个元素为 0 ,应忽略。nums2 的长度为 n

例一

输入:nums1 = [1,2,3,0,0,0], m = 3, nums2 = [2,5,6], n = 3 输出:[1,2,2,3,5,6] 解释:需要合并 [1,2,3] 和 [2,5,6] 。 合并结果是 [1,2,2,3,5,6] ,其中斜体加粗标注的为 nums1 中的元素。

思路:从后往前放,nums1和nums2中较大的数。(参考一)

void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n)
{
    int end1 = m-1;
    int end2 = n-1;
    int k = m + n -1;
    while(end1>=0 && end2>=0)
    {
        if(nums1[end1] >= nums2[end2])
        {
            nums1[k]=nums1[end1];
            k--;
            end1--;
        }
        else
        {
            nums1[k]=nums2[end2];
            k--;
            end2--;
        }
    }
    while(end2>=0)
    {
        nums1[k]=nums2[end2];
        k--;
        end2--;
    }
}

到此这篇关于C语言线性表中顺序表超详细理解的文章就介绍到这了,更多相关C语言 顺序表内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言的线性表之顺序表你了解吗

    目录 线性表 —— 顺序表 (C语言) 1. 顺序表的储存结构 2. 顺序表的基本操作 2.1 顺序表的插入 2.2 顺序表的查找 2.3 顺序表的删除 总结 线性表 —— 顺序表 (C语言) 概念 线性表的顺序表示指的是用一组地址连续的存储单元依次存储线性表中的数据元素,这种表示也称做线性表的顺序储存结构或顺序映像.通常,称这种存储结构的线性表为顺序表 (Sequential List) .其特点是,逻辑上相邻的数据元素,其物理次序也是相邻的. 1. 顺序表的储存结构 #include <st

  • C语言数据结构深入探索顺序表

    目录 1.线性表 2.顺序表 2.1概念及结构 2.2 接口实现 2.2.1初始化 2.2.2 检查容量 2.2.3 顺序表打印 2.2.4 顺序表尾插 2.2.5 顺序表尾删 2.2.6 顺序表头插 2.2.7 顺序表头删 2.2.8 顺序表在pos位置插入x 2.2.9 顺序表删除pos位置的值 2.2.10 尾插.尾删.头插.头删的改进 2.2.11 顺序表查找 2.2.12 顺序表销毁 2.3 数组相关面试题 2.4 顺序表的问题及思考 1.线性表 线性表(linear list)是n个

  • 详解C语言之顺序表

    目录 一.思维导图 二.步骤 1.初始化 2.求表长 3.插入数据元素 4.删除数据元素 5.取出数据元素 按位查找 按位查找 所有代码 总结 一.思维导图 二.步骤 1.初始化 代码如下: void ListInit(SeqList *L) { L->size = 0; } 2.求表长 代码如下: int ListLength(SeqList L) { return L.size; } 3.插入数据元素 代码如下: int ListInsert(SeqList *L, int i, DataT

  • C语言 数据结构之数组模拟实现顺序表流程详解

    目录 线性表和顺序表 线性表 顺序表 静态顺序表 动态顺序表 代码已经放在Gitee上,需要可以小伙伴可以去看看 用C语言数组模拟实现顺序表 Gitee 线性表和顺序表 线性表 线性表(linear list)是n个具有相同特性的数据元素的有限序列,这是我们广泛使用的数据结构. 线性表在逻辑上是线性结构,也就说是连续的一条直线.但是在物理结构上并不一定是连续的,线性表在物理上存储时,通常以数组和链式结构的形式存储. 常见的线性表:顺序表.链表.栈.队列.字符串- 顺序表 顺序表是用一段物理地址连

  • C语言数据结构顺序表中的增删改(头插头删)教程示例详解

    目录 头插操作 头删操作 小结 头插操作 继上一章内容(C语言数据结构顺序表中的增删改教程示例详解),继续讲讲顺序表的基础操作. 和尾插不一样,尾插出手阔绰直接的开空间,咱头插能开吗?好像没听说过哪个接口可以在数据前面开一片空间吧,那我们思路就只有一个了——挪数据.那应该从第一位开始挪吗?注意,这和 memcpy 函数机制是一样的,并不意味着后面数据一起挪动,也不会彼此独立,而是相互影响,挪动的数据会对后面数据进行覆盖. 那我们的逻辑就应该是从后往前挪,那我们就直接定一个下标,指向这段空间的最后

  • C语言数据结构顺序表中的增删改(尾插尾删)教程示例详解

    目录 初始化 尾插 格局打开 尾删 初始化 在初步认识顺序表这一结构后,我们就可以继续深入探究这是我之前在.h文件中创建的结构体 typedef int type; typedef struct list { type* a; int size; int capacity; }st; 在处理顺序表结构时我们会用到的一些接口,处理其中的关系,其实本质上就是函数,这里我用复杂英文对应出来方便形成记忆. void init(st *s); //插入 void pushback( st* p, type

  • C语言动态顺序表实例代码

    目录 顺序表概念: 一.准备工作 二.顺序表的基本操作  1.顺序表的初始化函数 2.尾插函数(在尾部插入数据) 3.头插函数(在数组头部插入数据)  4.尾删函数 5.头删函数 6.在第pos的位置插入数据 7.删除第pos个位置的数据 8.修改第pos个位置的数据 9.查找函数. 10.销毁函数 11.打印函数 三.总代码: 顺序表概念:         顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构.一般情况下用数组存储.在数组上完成数据的增删查改. 代码解析: 一.准备工

  • C语言 超详细顺序表的模拟实现实例建议收藏

    目录 概念及结构 接口实现 1 顺序表的动态存储 2 顺序表初始化 3 顺序表的销毁 4 顺序表的尾插 5 顺序表的尾删 6 顺序表的头插 7 顺序表的头删 8 顺序表容量的检查与扩容 9 顺序表任意位置的插入 10 顺序表任意位置的删除 11 顺序表的打印 12 顺序表元素的查找 13 顺序表元素的修改 概念及结构 顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储.在数组 上完成数据的增删查改. 顺序表一般可以分为: 静态顺序表:使用定长数组存储元素,元素

  • C语言线性表中顺序表超详细理解

    目录 一.本章重点 二.线性表 三.顺序表 四.静态顺序表接口实现 4.1顺序表初始化 4.2顺序表打印 4.3顺序表尾插 4.4顺序表尾删 4.5顺序表头插 4.6顺序表头删 4.7顺序表任意位置插入 4.8顺序表任意位置删除 五.动态顺序表接口实现 5.1顺序表的初始化 5.2顺序表打印 5.3顺序表尾插 5.4顺序表尾删 5.5顺序表头插 5.6顺序表头删 5.7顺序表任意位置插入 5.8顺序表任意位置删除 六.在线0j练习 一.移除元素(力扣) 二.合并两个有序数组(力扣) 一.本章重点

  • C语言编程数据结构线性表之顺序表和链表原理分析

    目录 线性表的定义和特点 线性结构的特点 线性表 顺序存储 顺序表的元素类型定义 顺序表的增删查改 初始化顺序表 扩容顺序表 尾插法增加元素 头插法 任意位置删除 任意位置添加 线性表的链式存储 数据域与指针域 初始化链表 尾插法增加链表结点 头插法添加链表结点 打印链表 任意位置的删除 双向链表 测试双向链表(主函数) 初始化双向链表 头插法插入元素 尾插法插入元素 尾删法删除结点 头删法删除结点 doubly-Linked list.c文件 doubly-Linkedlist.h 线性表的定

  • Python中顺序表原理与实现方法详解

    本文实例讲述了Python中顺序表原理与实现方法.分享给大家供大家参考,具体如下: Python中的顺序表 Python中的list和tuple两种类型采用了顺序表的实现技术,具有顺序表的所有性质. tuple是不可变类型,即不变的顺序表,因此不支持改变其内部状态的任何操作,而其他方面,则与list的性质类似. list的基本实现技术 Python标准类型list就是一种元素个数可变的线性表,可以加入和删除元素,并在各种操作中维持已有元素的顺序(即保序),而且还具有以下行为特征: 基于下标(位置

  • Python中顺序表的实现简单代码分享

    顺序表python版的实现(部分功能未实现) 结果展示: 代码示例: #!/usr/bin/env python # -*- coding:utf-8 -*- class SeqList(object): def __init__(self, max=8): self.max = max #创建默认为8 self.num = 0 self.date = [None] * self.max #list()会默认创建八个元素大小的列表,num=0,并有链接关系 #用list实现list有些荒谬,全当

  • C语言函数声明以及函数原型超详细讲解示例

    C语言代码由上到下依次执行,原则上函数定义要出现在函数调用之前,否则就会报错.但在实际开发中,经常会在函数定义之前使用它们,这个时候就需要提前声明. 所谓声明(Declaration),就是告诉编译器我要使用这个函数,你现在没有找到它的定义不要紧,请不要报错,稍后我会把定义补上. 函数声明的格式非常简单,相当于去掉函数定义中的函数体,并在最后加上分号;,如下所示: dataType functionName( dataType1 param1, dataType2 param2 ... ); 也

  • C语言可变参数与内存管理超详细讲解

    目录 概述 动态分配内存 重新调整内存的大小和释放内存 概述 有时,您可能会碰到这样的情况,您希望函数带有可变数量的参数,而不是预定义数量的参数.C 语言为这种情况提供了一个解决方案,它允许您定义一个函数,能根据具体的需求接受可变数量的参数.下面的实例演示了这种函数的定义. int func(int, ... ) { . . . } int main() { func(2, 2, 3); func(3, 2, 3, 4); } 请注意,函数func()最后一个参数写成省略号,即三个点号(...)

  • C语言结构体中内存对齐的问题理解

    目录 前言 思考 结构体在内存中开辟空间时内存对齐的规则 为什么存在内存对齐 1.平台的原因 2.性能的原因 前言 学C的同学应该知道~ 想精通C语言就不得不面对—指针与内存 续上次指针的进阶,这一章我来聊一聊C语言内存对齐的问题 学习结构体的你有没有注意过结构体向系统申请的内存为多少呢的 思考 #include<stdio.h> typedef struct s1 { char a; char b; int c; }s1; typedef struct s2 { char a; int c;

  • Python中Parser的超详细用法实例

    目录 1 前言 2.使用方法 2.1 实例化ArgumentParser 2.2 使用add_argument函数添加参数 2.3 add_argument() 方法定义如何解析命令行参数 2.4 使用parse_args解析参数 3 案例实践:action的可选参数store_true的作用 附:python-Parser使用步骤记忆 总结 这次主要记录python-Parser的用法,以及可能遇到的系列操作. 1 前言 if __name__ == "__main__": #Add

  • IntelliJ IDEA中配置Tomcat超详细教程

    目录 在IntelliJ IDEA中配置Tomcat 一.下载及安装Tomcat 二.配置Tomcat环境变量 三.在IntelliJ IDEA中配置Tomcat 在IntelliJ IDEA中配置Tomcat 一.下载及安装Tomcat 1.首先进入Tomcat官网:http://tomcat.apache.org/,在Download中选择需要下载的版本,然后根据电脑系统选择64位/32位的zip,开始下载(要记住安装路径!). ps:有zip和exe两种格式的,zip(64-bit Win

随机推荐