Python如何实现动态数组


Posted in Python onNovember 02, 2019

Python序列类型

在本博客中,我们将学习探讨Python的各种“序列”类,内置的三大常用数据结构——列表类(list)、元组类(tuple)和字符串类(str)。

不知道你发现没有,这些类都有一个很明显的共性,都可以用来保存多个数据元素,最主要的功能是:每个类都支持下标(索引)访问该序列的元素,比如使用语法 Seq[i]。其实上面每个类都是使用 数组 这种简单的数据结构表示。

但是熟悉Python的读者可能知道这3种数据结构又有一些不同:比如元组和字符串是不能修改的,列表可以修改。

计算机内存中的数组结构

计算机体系结构中,我们知道计算机主存由位信息组成,这些位通常被归类成更大的单元,这些单元则取决于精准的系统架构。一个典型的单元就是一个字节,相当于8位。

计算机系统拥有庞大数量的存储字节,那么如何才能找到我们的信息存在哪个字节呢?答案就是大家平时熟知的 存储地址 。基于存储地址,主存中的任何字节都能被有效的访问。实际上,每个存储字节都和一个作为其地址的唯一二进制数字相关联。如下图中,每个字节均被指定了存储地址:

Python如何实现动态数组

一般来说,编程语言记录标识符和其关联值所存储的地址之间的关系。比如,当我们声明标识符 xx 就有可能和存储器中的某一值相关联,而标识符 yy就可能和其他的值相关联。一组相关的变量能够一个接一个地存储在计算机存储器的一块连续区域内。我们将这种方式称为 数组。

我们来看Python中的例子,一个文本字符串 HELLO 是以一列有序字符的形式存储的,假定该字符串的每个Unicode字符需要两个字节的存储空间。最下面的数字就是该字符串的索引值。

Python如何实现动态数组

我们可以看到,数组可以存储多个值而无需构造具有特定索引的多个变量来指定其中的每个项目,并且几乎在所有编程语言(例如C、Java、C#、C++)中使用,但是Python更具有优势。Python在构建列表时,熟悉的读者可能知道,不需要预先定义数组或列表的大小,相反,在Python中,列表具有动态性质,我们可以不断的往列表中添加我们想要的数据元素。接下来,让我们看看Python列表的知识(已经熟悉的读者可以快速浏览或者跳过)。

Python列表

Python列表的操作

创建列表的语法格式:

[ele1, ele2, ele3, ele4, ...]

创建元组的语法格式:

(ele1, ele2, ele3, ele4, ...)

元组比列表的内存空间利用率更高,因为元组是固定不变的,所以没有必要创建拥有剩余空间的动态数组。

我们先在Python的IDE中创建一个列表,然后大致了解一下列表部分内置操作,我们先创建了一个名为test_list的列表,然后修改(插入或删除)元素,反转或清空列表,具体如下:

>>> test_list = [] # 创建名为test_list的空列表
>>> test_list.append("Hello")
>>> test_list.append("World")
>>> test_list
['Hello', 'World']
>>> test_list = ["Hello", "Array", 2019, "easy learning", "DataStructure"] # 重新给test_list赋值
>>> len(test_list) # 求列表的长度
5
>>> test_list[2] = 1024 # 修改列表元素
>>> test_list
['Hello', 'Array', 1024, 'easy learning', 'DataStructure']
>>>
>>> test_list.insert(1, "I love")  # 向列表中指定位置中插入一个元素
>>> test_list
['Hello', 'I love', 'Array', 1024, 'easy learning', 'DataStructure']
>>> test_list.append(2020) # 向列表末尾增加一个元素
>>> test_list
['Hello', 'I love', 'Array', 1024, 'easy learning', 'DataStructure', 2020]
>>>
>>> test_list.pop(1)  # 删除指定位置的元素
'I love'
>>> test_list.remove(2020) # 删除指定元素
>>> 
>>> test_list.index('Hello')  # 查找某个元素的索引值
0
>>> test_list.index('hello')  # 如果查找某个元素不在列表中,返回ValueError错误
Traceback (most recent call last):
 File "<pyshell#11>", line 1, in <module>
  test_list.index('hello')
ValueError: 'hello' is not in list
>>> 
>>> test_list.reverse() # 反转整个列表
>>> test_list
['DataStructure', 'easy learning', 2019, 'Array', 'Hello']
>>> test_list.clear()  # 清空列表
>>> test_list
[]

我们看上面的代码,可以看到list的相关操作——增删改查,已经很强大了,还有一些内置方法这里并没有做展示,留给读者自己去发现并体验。

那么Python内置的list类是如何被实现的呢?

好吧,答案是动态数组。说到这里,不知道大家学Python列表的时候是不是这样想的——列表很简单嘛,就是list()类、用中括号[]括起来,然后指导书籍或文档上的各类方法append、insert、pop...在IDE或者Pycharm一顿操作过后,是的我学会了。

但其实真的很不简单,比如我举个例子:A[-1]这个操作怎么实现?列表切片功能怎么实现?如何自己写pop()默认删除列表最右边的元素(popleft删除最左边简单)?...这些功能用起来爽,但真的自己实现太难了(我也还在学习中,大佬们请轻喷!)如果我们能学习并理解,肯定可以加强我们对数组这一结构的理解。

动态数组

什么是动态数组

动态数组是内存的连续区域,其大小随着插入新数据而动态增长。在静态数组中,我们需要在分配时指定大小。在定义数组的时候,其实计算机已经帮我们分配好了内存来存储,实际上我们不能扩展数组,因为它的大小是固定的。比如:我们分配一个大小为10的数组,则不能插入超过10个项目。

但是动态数组会在需要的时候自动调整其大小。这一点有点像我们使用的Python列表,可以存储任意数量的项目,而无需在分配时指定大小。

所以实现一个动态数组的实现的关键是——如何扩展数组?当列表list1的大小已满时,而此时有新的元素要添加进列表,我们会执行一下步骤来克服其大小限制的缺点:

  • 分配具有更大容量的新数组 list2
  • 设置 list2[i] = list1[i] (i=0,1,2,...,n-1),其中n是该项目的当前编号
  • 设置list1 = list2,也就是说,list2正在作为新的数组来引用我们的新列表。
  • 然后,只要将新的元素插入(添加)到我们的列表list1即可。

Python如何实现动态数组

接下来要思考的问题是,新数组应该多大?通常我们得做法是:新数组的大小是已满的旧数组的2倍。我们将在Python中编程实现动态数组的概念,并创建一个简单的代码,很多功能不及Python强大。

实现动态数组Python代码

在Python中,我们利用ctypes的内置库来创建自己的动态数组类,因为ctypes模块提供对原始数组的支持,为了更快的对数组进行学习,所以对ctypes的知识可以查看官方文档进行学习。关于Python的公有方法与私有方法,我们在方法名称前使用双下划线**__**使其保持隐藏状态,代码如下:

# -*- coding: utf-8 -*-
# @Time   : 2019-11-01 17:10
# @Author  : yuzhou_1su
# @ContactMe : https://blog.csdn.net/yuzhou_1shu
# @File   : DynamicArray.py
# @Software : PyCharm

import ctypes


class DynamicArray:
  """A dynamic array class akin to a simplified Python list."""

  def __init__(self):
    """Create an empty array."""
    self.n = 0       # count actual elements
    self.capacity = 1   # default array capacity
    self.A = self._make_array(self.capacity)   # low-level array

  def is_empty(self):
    """ Return True if array is empty"""
    return self.n == 0

  def __len__(self):
    """Return numbers of elements stored in the array."""
    return self.n

  def __getitem__(self, i):
    """Return element at index i."""
    if not 0 <= i < self.n:
      # Check it i index is in bounds of array
      raise ValueError('invalid index')
    return self.A[i]

  def append(self, obj):
    """Add object to end of the array."""
    if self.n == self.capacity:
      # Double capacity if not enough room
      self._resize(2 * self.capacity)
    self.A[self.n] = obj  # Set self.n index to obj
    self.n += 1

  def _resize(self, c):
    """Resize internal array to capacity c."""
    B = self._make_array(c)   # New bigger array
    for k in range(self.n):  # Reference all existing values
      B[k] = self.A[k]
    self.A = B     # Call A the new bigger array
    self.capacity = c  # Reset the capacity

  @staticmethod
  def _make_array(c):
    """Return new array with capacity c."""
    return (c * ctypes.py_object)()

  def insert(self, k, value):
    """Insert value at position k."""
    if self.n == self.capacity:
      self._resize(2 * self.capacity)
    for j in range(self.n, k, -1):
      self.A[j] = self.A[j-1]
    self.A[k] = value
    self.n += 1

  def pop(self, index=0):
    """Remove item at index (default first)."""
    if index >= self.n or index < 0:
      raise ValueError('invalid index')
    for i in range(index, self.n-1):
      self.A[i] = self.A[i+1]
    self.A[self.n - 1] = None
    self.n -= 1

  def remove(self, value):
    """Remove the first occurrence of a value in the array."""
    for k in range(self.n):
      if self.A[k] == value:
        for j in range(k, self.n - 1):
          self.A[j] = self.A[j+1]
        self.A[self.n - 1] = None
        self.n -= 1
        return
    raise ValueError('value not found')

  def _print(self):
    """Print the array."""
    for i in range(self.n):
      print(self.A[i], end=' ')
    print()

测试动态数组Python代码

上面我们已经实现了一个动态数组的类,相信都很激动,接下来让我们来测试一下,看能不能成功呢?在同一个文件下,写的测试代码如下:

def main():
  # Instantiate
  mylist = DynamicArray()

  # Append new element
  mylist.append(10)
  mylist.append(9)
  mylist.append(8)
  # Insert new element in given position
  mylist.insert(1, 1024)
  mylist.insert(2, 2019)
  # Check length
  print('The array length is: ', mylist.__len__())
  # Print the array
  print('Print the array:')
  mylist._print()
  # Index
  print('The element at index 1 is :', mylist[1])
  # Remove element
  print('Remove 2019 in array:')
  mylist.remove(2019)
  mylist._print()
  # Pop element in given position
  print('Pop pos 2 in array:')
  # mylist.pop()
  mylist.pop(2)
  mylist._print()
if __name__ == '__main__':
  main()

测试结果

激动人心的时刻揭晓,测试结果如下。请结合测试代码和数组的结构进行理解,如果由疏漏,欢迎大家指出。

The array length is: 5
Print the array:
10 1024 2019 9 8 
The element at index 1 is : 1024
Remove 2019 in array:
10 1024 9 8 
Pop pos 2 in array:
10 1024 8

总结

通过以上的介绍,我们知道了数组存在静态和动态类型。而在本博客中,我们着重介绍了什么是动态数组,并通过Python代码进行实现。希望你能从此以复杂的方式学会数组。

总结发言,其实越是简单的操作,背后实现原理可能很复杂。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python 输出一个两行字符的变量
Feb 05 Python
python批量下载图片的三种方法
Apr 22 Python
详解 Python中LEGB和闭包及装饰器
Aug 03 Python
使用python读取txt文件的内容,并删除重复的行数方法
Apr 18 Python
Python中pandas dataframe删除一行或一列:drop函数详解
Jul 03 Python
django 类视图的使用方法详解
Jul 24 Python
Win10下python 2.7与python 3.7双环境安装教程图解
Oct 12 Python
Pytorch之finetune使用详解
Jan 18 Python
已安装tensorflow-gpu,但keras无法使用GPU加速的解决
Feb 07 Python
使用jupyter Nodebook查看函数或方法的参数以及使用情况
Apr 14 Python
Python常用类型转换实现代码实例
Jul 28 Python
pytorch 如何使用float64训练
May 24 Python
python基于gevent实现并发下载器代码实例
Nov 01 #Python
python进程间通信Queue工作过程详解
Nov 01 #Python
通过实例了解python property属性
Nov 01 #Python
python装饰器练习题及答案
Nov 01 #Python
Django框架HttpRequest对象用法实例分析
Nov 01 #Python
Django框架HttpResponse对象用法实例分析
Nov 01 #Python
Django框架序列化与反序列化操作详解
Nov 01 #Python
You might like
怎样才能成为PHP高手?学会“懒惰”的编程
2006/12/05 PHP
php中session过期时间设置及session回收机制介绍
2014/05/05 PHP
PHPExcel实现表格导出功能示例【带有多个工作sheet】
2018/06/13 PHP
phpstorm 配置xdebug的示例代码
2019/03/31 PHP
PHP使用递归按层级查找数据的方法
2019/11/10 PHP
让GoogleCode的SVN下的HTML文件在FireFox下正常显示.
2009/05/25 Javascript
jQuery-Tools-overlay 使用介绍
2012/07/14 Javascript
JavaScript快速检测浏览器对CSS3特性的支持情况
2012/09/26 Javascript
JQuery之focus函数使用介绍
2013/08/20 Javascript
jQuery封装的获取Url中的Get参数示例
2013/11/26 Javascript
Javascript中的几种URL编码方法比较
2015/01/23 Javascript
javascript中字体浮动效果的简单实例演示
2015/11/18 Javascript
JS实现单击输入框弹出选择框效果完整实例
2015/12/14 Javascript
快速解决jquery.touchSwipe左右滑动和垂直滚动条冲突
2016/04/15 Javascript
bootstrap可编辑下拉框jquery.editable-select
2017/10/12 jQuery
浅谈es6语法 (Proxy和Reflect的对比)
2017/10/24 Javascript
Node层模拟实现multipart表单的文件上传示例
2018/01/02 Javascript
Vue自定义指令实现checkbox全选功能的方法
2018/02/28 Javascript
npm全局模块卸载及默认安装目录修改方法
2018/05/15 Javascript
vue-router 路由传参用法实例分析
2020/03/06 Javascript
基于javascript实现移动端轮播图效果
2020/12/21 Javascript
python判断端口是否打开的实现代码
2013/02/10 Python
python解析json实例方法
2013/11/19 Python
玩转python爬虫之URLError异常处理
2016/02/17 Python
python命名空间(namespace)简单介绍
2019/08/10 Python
Python使用微信itchat接口实现查看自己微信的信息功能详解
2019/08/22 Python
Python数据可视化:箱线图多种库画法
2019/11/06 Python
HTML5之HTML元素扩展(上)—新增加的元素及使用概述
2013/01/31 HTML / CSS
SHEIN香港:价格实惠的女性时尚服装
2018/08/14 全球购物
英国打印机墨盒销售网站:Ink Factory
2019/10/07 全球购物
加拿大领先家居家具网上购物:Aosom.ca
2020/05/27 全球购物
护理专科毕业推荐信
2013/11/10 职场文书
社区义诊活动总结
2014/04/30 职场文书
2016年读书月活动总结范文
2016/04/06 职场文书
微软Win11有哪些隐藏功能? windows11多个功能汇总
2021/11/21 数码科技
通过T-SQL语句创建游标与实现数据库加解密功能
2022/03/16 SQL Server