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

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

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

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

                             
图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. 杀死tomcat进程

    由于tomcat运行时eclipse非法关闭,导致tomcat进程没有关闭,再次启动eclipse,启动tomcat会报tomcat不能启动,且指出端口被占用.笔者解决方案如下: 方案一:重启电脑,简 ...

  2. Tiny4412 u-boot分析(1)u-boot配置流程分析

    参考Friendlyarm的文档,编译uboot的流程为 make tiny4412_config make 这个过程主要涉及到两个文件,顶层的Makefile文件和mkconfig文件,makeco ...

  3. java获取Linux持续运行时间及友好显示

    一.uptime命令 uptime命令可以查看系统的运行时间和负载 终端输入uptime 04:03:58 up 10 days, 13:19, 1 user, load average: 0.54, ...

  4. android键盘的Done按钮

    在EditText中,可以使用setImeOptions()方法来来开启软键盘的"Done"按钮. 示例代码如下:editText.setImeOptions(EditorInfo ...

  5. sqlplus--spool基础运用

    set heading offset feedback offset echo offset newp noneset termout offspool /home/orarun/scripts/da ...

  6. str_place()替换函数

    str_replace() 函数使用一个字符串替换字符串中的另一些字符. 注释:该函数对大小写敏感.请使用 str_ireplace() 执行对大小写不敏感的搜索. echo str_replace( ...

  7. WOJ 18 动态无向图

    一开始我是不会写的,后来点开了题解: 无话可说……那就写吧……然而第一发跑成暴力分,后来加了一个优化:就是在询问里面提到过的边都不用再加了. 然后……然后就过了呀…… 其实还有面向数据的编程的骚操作… ...

  8. python-ASCII与Unicode

    # Auther: Aaron Fan'''ASCII:不支持中文,1个英文占1个字节Unicode(万国码,支持所有国家的文字显示):支持中文,但是每个英文和中文都占2个字节UTF-8(是一种针对U ...

  9. SDUT 3398 数据结构实验之排序一:一趟快排

    数据结构实验之排序一:一趟快排 Time Limit: 1000MS Memory Limit: 65536KB Submit Statistic Problem Description 给定N个长整 ...

  10. python连接数据库--查询数据

    #!/usr/bin/python # -*- coding: utf-8 -*- import pymysql def fileDB(): # 打开数据库连接(ip/数据库用户名/登录密码/数据库名 ...