当前位置:   article > 正文

数据结构与算法--数组_数据结构和算法 数组

数据结构和算法 数组


前言

每一种编程语言中,基本都有数组这种数据类型。它不仅仅是一种数据类型,还是一种最基础的数据结构。


一、什么是数组

1.1 定义

数组(Array)是一种线性表数据结构。它用一组连续的内存空间,来存储一组具有相同类型的数据。
在这里插入图片描述

1.2 数组特性

  1. 支持随机访问,根据下标随机访问的时间复杂度为 O(1)。
  2. 连续的内存空间、相同类型的数据:因此可以随机访问,但为了保证内存的连续性,需要做大量的数据搬移工作,使增删等操作变得非常低效。
  3. 线性存储,程序执行效率高。

1.3 时间复杂度

  • 插入元素:O(n),所插入位置后面的元素都要向后移动。
  • 删除元素:O(n),所删除位置后面的元素都要向前移动。
  • 访问元素:O(1),使用索引访问。
  • 查找元素:与使用查找算法有关,如顺序查找、二分查找、插值查找等。

1.4 空间复杂度

数组只用来存储指定类型的数据,所以存储n个数据的空间复杂度为O(n)。

二、基本操作

2.1 插入数据

数组插入数据,特定场景,特殊对待,如下:

  • 数组需要保持有序:将插入位置k后面的数据元素依次后移,空出要插入的位置,再将新元素插入。O(n)
  • 数组不需要保持有序:将插入位置的旧元素移到最后,再在k处插入新元素。或直接将新元素放在最后。O(1)

2.2 删除数据

数组删除数据,特定场景,特殊对待,如下:

  • 考虑内存连续性:

    数组需要保持有序:删除k位置元素后,将后面的元素依次前移一个位置。O(n)

    数组不需要保持有序:用最后一个元素覆盖要删除的元素。O(1)

  • 不考虑内存连续:

    标记删除法,将要删除的元素标记(没有真正的删除),当数组空间不够时,一次性删除。但会造成内存不连续,如下图:
    在这里插入图片描述

数组空间:数组所占内存空间,不随插入或删除元素而改变,除非申请空间
数组长度:数组所存储的数据元素个数,随插入或删除元素而改变

2.3 随机访问

以存储整型数据为例
在这里插入图片描述
对于数组int a[],其地址为a,第一个数组元素的地址和数组的地址相等,第二个数据元素的地址为a+4,依次类推,每往后移动一个数据元素,地址加4,可见只要知道了数组的首地址,就知道了数组中每一个元素的地址,所以利用数组下标可以实现数组元素的随机访问。

随机访问寻址方法:a[i]_address = base_address + i * data_type_size

其中,base_address 为数组首地址,data_type_size 为每个元素的大小,依数据类型确定。

#include <stdio.h>

int main()
{
    int a[] = {0, 1, 2, 3};

    printf("   a.address: %p\n", a);
    for (int i = 0; i < 4; i++)
    { 
	printf("a[%d].address: %p\n", i, &a[i]);
    }
}

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

打印结果为:

 a.address: 0x7ffd46f51c50
a[0].address: 0x7ffd46f51c50
a[1].address: 0x7ffd46f51c54
a[2].address: 0x7ffd46f51c58
a[3].address: 0x7ffd46f51c5c
  • 1
  • 2
  • 3
  • 4
  • 5

三、关于数组的笔试题

3.1 字符串逆序

给定一个含有n个元素的字符数组a,将其原地逆序

// 字符串逆序
void Reverse(char*a, int n)
{
     int left =0; 
     int right = n -1;

     while (left < right)
     {
         char temp = a[left] ;
         a[left++] = a[right] ;
         a[right--] = temp ;
     }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

3.2 组合问题

给定一个含有n个元素的整型数组a,从中任取m个元素,求所有组合。比如下面的例子
a = 1, 2, 3, 4, 5
m = 3
输出
1 2 3, 1 2 4, 1 2 5, 1 3 4, 1 3 5, 1 4 5
2 3 4, 2 3 5, 2 4 5
3 4 5

// n选m的所有组合
int buffer[100] ;

void PrintArray(int *a, int n)
{
    for (int i = 0; i < n; ++i)
        cout << a[i] << "";
    cout << endl ;
}

bool IsValid(int lastIndex, int value)
{
    for (int i = 0; i < lastIndex; i++)
    {
        if (buffer[i] >= value)
            return false;
    }
    return true;
}

void Select(int t, int n, int m)
{
    if (t == m)
        PrintArray(buffer, m);
    else
    {
        for (int i = 1; i <= n; i++)
        {
            buffer[t] = i;
            if (IsValid(t, i))
                Select(t + 1, n, m);
        }
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34

3.3 合并两个数组

给定含有n个元素的两个有序(非降序)整型数组a和b。合并两个数组中的元素到整型数组c,要求去除重复元素并保持c有序(非降序)。例子如下
a = 1, 2, 4, 8
b = 1, 3, 5, 8
c = 1, 2, 3, 4, 5, 8
分析
利用合并排序的思想,两个指针i,j和k分别指向数组a和b,然后比较两个指针对应元素的大小,有以下三种情况

  1. a[i] < b[j],则c[k] = a[i]。

  2. a[i] == b[j],则c[k]等于a[i]或b[j]皆可。

  3. a[i] > b[j],则c[k] = b[j]。

重复以上过程,直到i或者j到达数组末尾,然后将剩下的元素直接copy到数组c中即可

// 合并两个有序数组
void Merge(int *a, int *b, int *c, int n)
{
    int i = 0 ;
    int j = 0 ;
    int k = 0 ;

    while (i < n && j < n)
    {
        if (a[i] < b[j])// 如果a的元素小,则插入a中元素到c
        {
            c[k++] = a[i] ;
            ++i ;
        }
        else if (a[i] == b[j])// 如果a和b元素相等,则插入二者皆可,这里插入a
        {
            c[k++] = a[i] ;
            ++i ;
            ++j ;
        }
        else // a[i] > b[j] // 如果b中元素小,则插入b中元素到c
        {
            c[k++] = b[j] ;
            ++j ;
        }
    }

    if (i == n) // 若a遍历完毕,处理b中剩下的元素
    {
        for (int m = j; m < n; ++m)
            c[k++] = b[m] ;
    }
    else//j == n, 若b遍历完毕,处理a中剩下的元素
    {
        for (int m = i; m < n; ++m)
            c[k++] = a[m] ;
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38

3.4 重排问题

给定含有n个元素的整型数组a,其中包括0元素和非0元素,对数组进行排序,要求:

  1. 排序后所有0元素在前,所有非零元素在后,且非零元素排序前后相对位置不变

  2. 不能使用额外存储空间

例子如下

输入 0, 3, 0, 2, 1, 0, 0

输出 0, 0, 0, 0, 3, 2, 1

void Arrange(int* a, int n)
{
    int k = n -1 ;
    for (int i = n -1; i >=0; --i)
    {
        if (a[i] !=0)
        {
            if (a[k] ==0)
            {
                a[k] = a[i] ;
                a[i] =0 ;
            }
            --k ;
        }
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/900634
推荐阅读
相关标签
  

闽ICP备14008679号