当前位置: 首页 > news >正文

python切片赋值_Python中实现:变量和变量赋值的几种形式

c7cf883d18877cb4596d7a2f6f6f448b.png

动态类型的语言

python是动态类型的语言,不需要声明变量的类型。

实际上,python中的变量仅仅只是用来保存一个数据对象的地址。无论是什么数据对象,在内存中创建好数据对象之后,都只是把它的地址保存到变量名中。所以变量名是类型无关的,但它指向的值是类型相关的,可以是数值、字符串、列表、函数、类、对象等等。这些内存对象中都至少包含3部分:对象类型、对象的引用计数(用来判断改对象是否可被垃圾回收器回收)、对象的值。

因此,a = 3中,变量名a保存的是数据对象3的地址,之后可以为其赋值一个字符串a = “hello”,这时a保存的是"hello"字符串的地址。在这个类型改变的过程中,a仅仅只是修改了一下地址而已。

变量的命名风格

python中的变量命名时只能包含数字、大小写字母、下划线这三种类型的字符,并且数字不能是首字符。 还有一些有特殊意义的变量命名方式(目前这些内容了解即可):

  1. 前缀并后缀双下划线的变量,如__ name__,这种类型的变量在python中有特殊意义,属于对象的内置属性,以后学了类和对象就知道了
  2. 单下划线前缀的变量,如_x,这类变量不会被from ModuleName import *的方式导入
  3. 双下划线前缀的变量,如__x,这类变量是类的本地变量或称为类的私有变量,它会扩展成__classname_x

除此之外,还有约定俗成的命名方式:

  1. 常量以全大写字符表示
  2. 普通变量、函数名、方法名都以小写字母开头命名
  3. 模块名、包名以全小写字母命名
  4. 类名以大写字母开头

因为只是约定俗成,所以没有强制限制。

变量赋值的几种形式细节

本文解释python中变量赋值的形式,并解释一些细节。后面还有一篇文章解释python中按引用赋值的文章。 python中变量赋值的几种形式。

6773fd1286f55e9b2cdd60cf159e4741.png

注意:python的数值是不可变对象,无法在原处修改数据,所以不支持自增、自减。

703052edc4ca1bde91e81d4283c66097.png

其中(1)-(3)无需过多解释,唯一需要注意的是,当使用逗号的时候,python总会临时或永久地建立成tuple来保存元素,所以x, y = “long”, "shuai"在内部完全等价于(x, y) = (“long”, “shuai”)。

实际上,列表元素也可以赋值给元组,或者元组赋值给列表,只要两边的序列元素个数能对应,无所谓左右两边的序列类型:

9cfcf35bb727f7ff12e670375051d3bb.png

对于(4)赋值方式,是序列赋值的行为,在python中,只要是序列,都可以这样赋值。正如这里的变量赋值情况等价于:

f8cf25a926bfc6de5bfb233d60e015d9.png

如果换成其它的序列也一样。例如:

63bdeaace2225207f3e2febd2ece6840.png

但是变量和序列中的元素必须一一对应。如果变量名与元素个数不同,则会报错,除非只有一个变量名,这表示将整个序列赋值给这个变量。

如果想要将序列中的元素赋值给不等的变量,可以考虑先将序列进行切片。

3d588ba8659c59f22b7c3469e200567f.png

(5)的赋值方式则正好是让变量名少于元素个数的方式。这种赋值形式称为序列解包(下文会专门解释这种赋值方式),多出来的元素会全部以列表的方式赋值给最后一个变量名。正如这里等价于:

d6578d1eb75286233901b6be2441173c.png

下面两种赋值方式得到的结果是一样的,a是字符串,b是列表,b都包含3个元素:

2efa3c13753d7b86570b51c2366247a1.png

赋值的结果:

ca5ac080933442b0371ccc086901c54f.png

(6)的赋值方式等价于:

43ef608528f9cb3af4376812aa9a00d9.png

python赋值时,总是先计算"=“右边的结果,然后将结果按照赋值方式赋值给”="左边的变量。所以,这里的过程是先将"long"赋值给变量b,再将b赋值给变量a。

因为总是先计算右边,所以交换变量非常的方便。

9853a12cefb39a755bf240494fe9a18e.png

(7)的赋值方式a += 3在结果上等价于a = a + 3,在其它语言中这两种赋值方式是完全等价的,但在python中这种增强赋值的方式要比后者更高效率些,为什么效率要高一些,下文会稍作解释。在很大程度上来说,Python中只要是简化的形式,基本上都比更复杂的等价形式效率更高。

(8)的赋值方式((a, b), c) = (‘lo’, ‘ng’)是将序列进行嵌套序列赋值,将’lo’赋值给元组(a, b),'ng’赋值给c,元组又进一步赋值a=‘l’, b=‘o’。这种赋值方式在python中很好用,特别是在表达式中赋值的时候,比如for循环和函数参数:

b10f2d630a969f9da4fe820dd9945a1d.png

关于序列解包

在前面简单介绍了一下序列解包:

c9f6c2d52cc171aae53e85cac5119420.png

当使用一个前缀变量的时候,表示将序列中对应的元素全部收集到一个列表中(注意,总是一个列表),这个列表名为开头的那个变量名。*号可以出现在任意位置处,只要赋值的时候能前后对应位置关系即可。 注意其中的几个关键字:序列、对应的元素、列表

  • 序列意味着可以是列表、元组、字符串等等
  • 列表意味着只要收集不报错,赋值给解包变量的一定是一个列表
  • 对应的元素意味着可能收集到0或任意个元素到列表。

不管如何,收集的结果总是列表,只不过可能是空列表或者只有一个元素的列表。 例如:

0b47273ddabaf9ec22a1099f94d1a72a.png

两个注意事项:

  1. 因为序列解包是根据元素位置来进行赋值的,所以不能出现多个解包变量
  2. 如果将序列直接赋值给单个解包变量时(即没有普通变量),这个解包变量必须放在列表或元组中
37036918eb3c893876ec8bb5d328427c.png

之所以单个解包变量时必须放在元组或变量中,看下面两个等价的例子就很容易理解了:

c2e2cb0a636033293a52ccb356d392eb.png

最后,序列解包是切片的便捷替代方式。能用序列解包的,都能用切片来实现,但切片要输入额外的各种字符。例如:

f4141b2fe0705f9a8a64e2af71ab72d5.png

需要注意,解包返回的一定是列表,但序列切片返回的内容则取决于序列的类型。例如下面元组的切片返回的是元组,而不是列表:

8544f313b3986c9fd27e67efd047d780.png

二元赋值表达式

python支持类似于a += 3这种二元表达式。比如:

545f68d727f52b495ade7a8812c13bf4.png

在python中的某些情况下,这种二元赋值表达式可能比普通的赋值方式效率更高些。原因有二:

  1. 二元赋值表达式中,a可能会是一个表达式,它只需计算评估一次,而a = a + 3中,a要计算两次。
  2. 对于可变对象,可以直接在原处修改得到修改后的值,而普通的一元赋值表达式必须在内存中新创建一个修改后的数据对象,并赋值给变量

第一点无需解释。关于第二点,看下面的例子:

96a35407f25de4984c63a256cc883b53.png

对于上面(1)和(4)的一元赋值表达式,先取得L,然后创建一个新的列表对象,将L拷贝到新列表对象中,并将4或5,6放进新列表对象,最后赋值给L。这个过程中涉及到了几个步骤:新分配内存、内存中列表拷贝、放入新数据。

而(2)(3)是等价的,(5)(6)也是等价的,它们都是直接在内存中的原始列表处修改,不会有拷贝操作,新建的数据对象仅仅只是一个元素。

按照理论上来说,确实二元赋值方式要效率高一些,但要注意的是,列表中保存的只是各元素的引用,所以拷贝列表也仅仅只是拷贝一点引用,这是微乎其微的开销。所以一元赋值和二元赋值的差距在这一点的性能上基本没差距,主要的差距还在于一元、二元赋值方式可能存在的表达式不同评估次数。

总的来说,使用二元赋值表达式通常可以作为可变对象赋值的一种优化手段。

相关文章:

  • linux shell mysql 遍历_shell 脚本中获取mysql多个字段的值
  • mysql current read_浅谈MySQL之 REPEATABLE-READ.
  • mysql 索引pk_细说mysql索引
  • mysql 6.0 jar_maven搭建ssm框架是使用最新mysql 6.0jar遇到的问题
  • rhel7最小化安装 mysql_安装完最小化 RHEL/CentOS 7 后需要做的 30 件事情(二)
  • 链接索引 mysql_mysql索引
  • linux c epoll mysql_epoll简介和使用
  • python可以不对变量初始化_Python的诡异陷阱
  • python怎么绘制图例_用Python绘制图例
  • centos7安装mysql5.7.23_centos7.1上安装Mysql5.7.23
  • java客户端_大家都是怎么发布Java客户端程序的?难道让用户自己装JRE?
  • java 加载jar_java – 在运行时加载jar
  • java ant linux_linux下ant的安装
  • java获取数据库信息_java获取数据库的库、表、字段信息
  • java jpanel 布局管理器,JPanel和Java布局管理器
  • [LeetCode] Wiggle Sort
  • dva中组件的懒加载
  • jquery ajax学习笔记
  • ubuntu 下nginx安装 并支持https协议
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 简析gRPC client 连接管理
  • 视频flv转mp4最快的几种方法(就是不用格式工厂)
  • ​iOS安全加固方法及实现
  • ​iOS实时查看App运行日志
  • #Java第九次作业--输入输出流和文件操作
  • #pragma 指令
  • (C语言)二分查找 超详细
  • (ZT)一个美国文科博士的YardLife
  • (二十一)devops持续集成开发——使用jenkins的Docker Pipeline插件完成docker项目的pipeline流水线发布
  • (附源码)springboot 智能停车场系统 毕业设计065415
  • (附源码)ssm高校志愿者服务系统 毕业设计 011648
  • (三十五)大数据实战——Superset可视化平台搭建
  • (四)图像的%2线性拉伸
  • (一)u-boot-nand.bin的下载
  • ******IT公司面试题汇总+优秀技术博客汇总
  • ./include/caffe/util/cudnn.hpp: In function ‘const char* cudnnGetErrorString(cudnnStatus_t)’: ./incl
  • .bat批处理(六):替换字符串中匹配的子串
  • .NET BackgroundWorker
  • .NET Core 控制台程序读 appsettings.json 、注依赖、配日志、设 IOptions
  • .NET 分布式技术比较
  • .NET 将混合了多个不同平台(Windows Mac Linux)的文件 目录的路径格式化成同一个平台下的路径
  • .NET 中 GetHashCode 的哈希值有多大概率会相同(哈希碰撞)
  • .NET面试题解析(11)-SQL语言基础及数据库基本原理
  • .NET微信公众号开发-2.0创建自定义菜单
  • @DataRedisTest测试redis从未如此丝滑
  • @FeignClient 调用另一个服务的test环境,实际上却调用了另一个环境testone的接口,这其中牵扯到k8s容器外容器内的问题,注册到eureka上的是容器外的旧版本...
  • @vue/cli 3.x+引入jQuery
  • [ C++ ] STL_list 使用及其模拟实现
  • [ 数据结构 - C++] AVL树原理及实现
  • [1181]linux两台服务器之间传输文件和文件夹
  • [2016.7.Test1] T1 三进制异或
  • [2019.3.20]BZOJ4573 [Zjoi2016]大森林
  • [ajaxupload] - 上传文件同时附件参数值
  • [BT]BUUCTF刷题第9天(3.27)
  • [BZOJ2281][SDOI2011]黑白棋(K-Nim博弈)