逻辑结构上呈线性分布的数据元素在实际的物理存储结构中也同样相互之间紧挨着,这种存储结构称为线性表的顺序存储结构。

也就是说,逻辑上具有线性关系的数据按照前后的次序全部存储在一整块连续的内存空间中,之间不存在空隙,这样的存储结构称为顺序存储结构。

使用顺序存储结构存储的数据,第一个元素所在的地址就是这块存储空间的首地址。通过首地址,可以轻松访问到存储的所有的数据,只要首地址不丢,数据永远都能找着(一根绳上的蚂蚱,要有就都有)。

使用线性表的顺序存储结构生成的表,称为顺序表。

                             
图1 顺序表结构示意图

顺序表的实现方法

顺序表中存放数据的特点和数组这种数据类型完全吻合,所以顺序表的实现使用的是数组。

数组实现顺序表的存储结构时,一定要注意预先申请足够大的内存空间,避免因存储空间不足,造成数据溢出,导致不必要的程序错误甚至崩溃。

顺序表的存储结构

在建立顺序表时,除了预先申请内存空间,还需要实时记录顺序表的长度和顺序表本身申请的内存大小,便于后期对顺序表中的数据元素进行调取。

所以,要自定义顺序表的结构:

typedef struct Table{
int * head;//声明了一个名为head的长度不确定的数组,也叫“动态数组”
int length;//记录当前顺序表的长度
int size;//记录顺序表分配的存储容量
}table;

顺序表的创建

顺序表的建立,也就是顺序表进行初始化,在预先申请内存空间的同时,给变量size和length赋初值:

table initTable()
{
  table t;
  t.head=(int*)malloc(Size*sizeof(int));  //构造一个空的顺序表,动态申请存储空间
  if (!t.head)   //如果申请失败,作出提示并直接退出程序
  {  
    printf("初始化失败");
    exit();
  }
  t.length=;     //空表的长度初始化为0
  t.size=Size;    //空表的初始存储空间为Size
  return t;
}

顺序表查找元素

在数组中查找某个数据元素时,可以采取多种查找算法,例如二分查找、插值查找、斐波那契查找算法等。

具体的查找算法以及各自的时间复杂度后续章节会介绍。

根据顺序表中存储的数据的特点,选择合适的算法。这里,采用顺序查找算法(普通的遍历算法)。

实现代码:

//查找函数,其中,elem表示要查找的数据元素的值
int selectTable(table t, int elem)
{
  for (int i=; i<t.length; i++)
  {
    if (t.head[i] == elem)
    {
      return i+;
    }
  }
  return -;//如果查找失败,返回-1
}

顺序表中更改元素

顺序表中更改数据元素,最简单直接的方式就是:调用查找算法找到该数据元素的位置,直接在该位置上更改。

实现代码:

//更改函数,其中,elem为要更改的元素,newElem为新的数据元素
table amendTable(table t, int elem, int newElem)
{
  int add = selectTable(t, elem);
  t.head[add-] = newElem;  //由于返回的是元素在顺序表中的位置,所以-1就是该元素在数组中的下标
  return t;
}

顺序表插入元素

插入数据元素,无非三种情况:

  1. 在表头插入
  2. 在表的中间某个位置插入
  3. 直接尾随顺序表,作为表的最后一个元素

无论在顺序表的什么位置插入数据元素,解决办法都是:找到要插入的位置,将后续数据元素整体向后移动一个位置,最后直接在腾出来的位置上插入数据元素。

实现代码:

//插入函数,其中,elem为插入的元素,add为插入到顺序表的位置
table addTable(table t, int elem, int add)
{
  //判断插入本身是否存在问题(如果插入元素位置比整张表的长度+1还大(如果相等,是尾随的情况),
  //或者插入的位置本身不存在,程序作为提示并自动退出)
  if (add>t.length+1 || add<)
  {
    printf("插入位置有问题");
    return t;
  }
  //做插入操作时,首先需要看顺序表是否有多余的存储空间提供给插入的元素,如果没有,需要申请
  if (t.length == t.size)
  {
    t.head=(int *)realloc(t.head, (t.size+)*sizeof(int));
    if (!t.head)
    {
      printf("存储分配失败");
      return t;
    }
    t.size += ;
  }
  //插入操作,需要将从插入位置开始的后续元素,逐个后移
  for (int i=t.length-; i>=add-; i--)
  {
    t.head[i+] = t.head[i];
  }
  //后移完成后,直接将所需插入元素,添加到顺序表的相应位置
  t.head[add-] = elem;
  //由于添加了元素,所以长度+1
  t.length++;
  return t;
}

注意:在此程序中,当数组存储空间不足时,使用realloc函数每次额外多申请 1 个int型的存储空间,这么做还不是最优。最好的办法就是每次发现空间不够时,多申请几个内存空间,这么做的好处是:在后续做插入操作过程中不需要每次都运行realloc函数,提高了程序的运行效率。

顺序表删除元素

在数组中删除元素时,只需将该元素所在位置后的所有数据元素整体前移 1 个位置即可。

前移的过程中被删除元素被后一个元素覆盖掉,间接实现了删除操作。

实现代码:

table delTable(table t,int add)
{
  if (add>t.length || add<)
  {
    printf("被删除元素的位置有误");
    exit();
  }
  //删除操作
  for (int i=add; i<t.length; i++)
  {
    t.head[i-]=t.head[i];
  }
  t.length--;
  return t;
}

完整的程序

#include <stdio.h>
#include <stdlib.h>
#define Size 4
typedef struct Table
{
  int * head;
  int length;
  int size;
}table;
table initTable()
{
  table t;
  t.head=(int*)malloc(Size*sizeof(int));
  if (!t.head)
  {
    printf("初始化失败");
    exit();
  }
  t.length = ;
  t.size = Size;
  return t;
}
table addTable(table t,int elem,int add)
{
  if (add>t.length+1 || add<)
  {
    printf("插入位置有问题");
    return t;
  }
  if (t.length >= t.size)
  {
    t.head = (int *)realloc(t.head, (t.size+)*sizeof(int));
    if (!t.head)
    {
      printf("存储分配失败");
    }
    t.size += ;
  }
  for (int i=t.length-; i>=add-; i--)
  {
    t.head[i+]=t.head[i];
  }
  t.head[add-] = elem;
  t.length++;
  return t;
}
table delTable(table t,int add)
{
  if (add>t.length || add<)
  {
    printf("被删除元素的位置有误");
    exit();
  }
  for (int i=add; i<t.length; i++)
  {
    t.head[i-] = t.head[i];
  }
  t.length--;
  return t;
}
int selectTable(table t,int elem)
{
  for (int i=; i<t.length; i++)
  {
    if (t.head[i]==elem)
    {
      return i+;
    }
  }
  return -;
}
table amendTable(table t, int elem, int newElem)
{
  int add = selectTable(t, elem);
  t.head[add-] = newElem;
  return t;
}
void displayTable(table t)
{
  for (int i=; i<t.length; i++)
  {
    printf("%d",t.head[i]);
  }
  printf("\n");
}
int main()
{
  table t1 = initTable();
  for (int i=; i<=Size; i++)
  {
    t1.head[i-] = i;
    t1.length++;
  }
  printf("原顺序表:\n");
  displayTable(t1);
  printf("删除元素1:\n");
  t1=delTable(t1, );
  displayTable(t1);
  printf("在第2的位置插入元素5:\n");
  t1 = addTable(t1, , );
  displayTable(t1);
  printf("查找元素3的位置:\n");
  int add = selectTable(t1, );
  printf("%d\n",add);
  printf("将元素3改为6:\n");
  t1 = amendTable(t1, , );
  displayTable(t1);
  return ;
}
输出结果: 原顺序表: 删除元素1: 在第2的位置插入元素5: 查找元素3的位置: 将元素3改为6:

顺序表的优缺点

顺序表实现的基础,完全借用了数组这一数据类型,优点是在对数据进行遍历时,数据在连续的物理空间中存放,查找的速度比较快。

但是由于数组本身的限制,在向顺序表中新增或者删除数据元素时,如果被操作位置后续有很多数据元素,后续所有的数据元素都需要前移,最后虽然实现了功能,但是程序总体效率不高。

数据结构4:顺序表(线性表的顺序存储结构)及C语言实现的更多相关文章

  1. YTU 2987: 调整表中元素顺序(线性表)

    2987: 调整表中元素顺序(线性表) 时间限制: 1 Sec  内存限制: 2 MB 提交: 1  解决: 1 题目描述 若一个线性表L采用顺序存储结构存储,其中所有元素都为整数.设计一个算法,将所 ...

  2. 完成代码将x插入到该顺序有序线性表中,要求该线性表依然有序

    #include <stdio.h> #include <malloc.h> int main(void) { int i, n; double s = 1.3; double ...

  3. 数据结构线性表的动态分配顺序存储结构算法c语言具体实现和算法时间复杂度分析

    #include<stdio.h>#include<stdlib.h>//线性表的动态分配顺序存储结构#define LIST_INIT_SIZE 100//线性表存储空间的初 ...

  4. 数据结构(Java描述)之线性表

    基础概念 数据结构:是相互之间存在一种或多种关系的数据元素的集合. 逻辑结构和物理结构 关于数据结构,我们可以从逻辑结构和物理结构这两个维度去描述 逻辑结构是数据对象中数据元素之间的关系,是从逻辑意义 ...

  5. 数据结构与算法分析java——线性表1

    说到线性结构的话,我们可以根据其实现方式分为三类: 1)顺序结构的线性表 2)链式结构的线性表 3)栈和队列的线性表   应用程序后在那个的数据大致有四种基本的逻辑结构: 集合:数据元素之间只有&qu ...

  6. 线性表之顺序存储结构(C语言动态数组实现)

    线性表的定义:N个数据元素的有限序列 线性表从存储结构上分为:顺序存储结构(数组)和 链式存储结构(链表) 顺序存储结构:是用一段连续的内存空间存储表中的数据 L=(a1,a2,a3....an) 链 ...

  7. 数据结构代码整理(线性表,栈,队列,串,二叉树,图的建立和遍历stl,最小生成树prim算法)。。持续更新中。。。

    //归并排序递归方法实现 #include <iostream> #include <cstdio> using namespace std; #define maxn 100 ...

  8. 《数据结构》C++代码 线性表

    线性表,分数组和链表两种(官方名称记得是叫顺序存储和链式存储).代码里天天用,简单写写. 首先是数组,分静态.动态两种,没什么可说的,注意动态的要手动释放内存就好了. 其次是链表,依旧分静态.动态.课 ...

  9. [数据结构 - 第3章] 线性表之单链表(C++实现)

    一.类定义 单链表类的定义如下: #ifndef SIGNALLIST_H #define SIGNALLIST_H typedef int ElemType; /* "ElemType类型 ...

随机推荐

  1. vue-cli脚手架build目录中的dev-server.js配置文件

    本文系统讲解vue-cli脚手架build目录中的dev-server.js配置文件 这个配置文件是命令npm run dev 和 npm run start 的入口配置文件,主要用于开发环境 由于这 ...

  2. python中引号中有双引号

    #/usr/bin/python import os name = "ABC" #ABC是具体的模块名,产品经理每一次给的模块名字都不一样 os.environ['name'] = ...

  3. 类型:.net;问题:asp.net window验证;结果:细说ASP.NET Windows身份认证

    细说ASP.NET Windows身份认证 阅读目录 开始 认识ASP.NET Windows身份认证 访问 Active Directory 在ASP.NET中访问Active Directory ...

  4. leetcode230

    /** * Definition for a binary tree node. * public class TreeNode { * public int val; * public TreeNo ...

  5. x264的一些参数设置对编码效率的影响

    i_luma_deadzone[0]和i_luma_deadzone[1]分别对应inter和intra, 取值范围1~32,测试可以得知,这连个参数的调整可以对数据量有很大影响,值越大数据量相应越少 ...

  6. 【277】◀▶ Python 列表/元组/字典说明

    目录: 前言 一.访问列表中的值 二.更新列表 三.删除列表元素 四.Python 列表脚本操作符 五.Python 列表函数 & 方法 参考:Python 列表(List)使用说明 列表截取 ...

  7. adb pull 和 adb push

    1. 操作单个文件 通过adb push,则可将文件添加到SD卡中.如果想在push的时候修改文件名称的话,只需要修改push的第二个参数改成完整路径(目录+文件名),如/sdcard/test-0. ...

  8. 每天一道算法题(12)——和为n的连续正数序列或者随机数

    题目:输入一个正数n,输出所有和为n 连续正数序列.例如输入15,由于1+2+3+4+5=4+5+6=7+8=15,所以输出3 个连续序列1-5.4-6 和7-8. 1.思路 尊崇以下策略: (1)对 ...

  9. 关于TortoiseGit设置代理服务器会自动取消的问题解决方法

    在前些日子,我使用TortoiseGit的并给其设置代理服务器的时候,每次设置好之后,关闭打开总是会自动取消勾选,通过不断的尝试和分析之后,得出了解决方案. 首先,这个配置项是保存在HOME环境变量对 ...

  10. elasticsearch配置文件里的一些坑 [Failed to load settings from [elasticsearch.yml]]

    这里整理几个空格引起的问题. 版本是elasticsearch-2.3.0 或者elasticsearch-rtf-master Exception in thread "main" ...