快乐学Python,Python基础之组织代码「类与对象」
在上一篇文章中,我们了解了函数。这一篇文章我们来了解一下Python中另外一个重要的概念:类与对象。
1、类与对象
(1)类与对象有什么关系?
你可能会奇怪,为什么要叫类与对象呢?是两个不同的东西吗?简单来说,类代表一个类别,而对象则代表类的一个实例。比如我们在变量与数据类型中学习的整型变量为例。
a = 3print(type(a))
输出结果为:
<class 'int'>
我们创建了一个变量 a,并赋值为 3,然后打印了它的类型。可以看到,输出 int,代表整型变量。
细心的你也发现了,输出里面有一个单词 class,它就是类别的意思。难道说?所谓的整型,是一个类?你猜得没错。这里的整型:int,就是一个类。而我们创建的整型变量 a,就是他的一个实例。
我们再举一些生活中的例子:
- “人”是一个类,小明则是“人”这个类的一个实例,所以小明是一个对象;
- “打车公司”是一个类,滴滴则是这个类的一个实例对象(也可以隐藏 实例 这个称呼);
- “酱油”是一个类,海天酱油 则是一个对象;
- “车”是一个类,奔驰车则是一个对象。
细心的你不难发现,类与对象本质上是抽象与具象的关系,对象在类的基础上进行了适当的具象。所以在某个抽象关系中的对象也可能会成为另一个抽象关系中的类。比如上面奔驰车是车的一个对象,那同样可能存在,奔驰车是一个类,而 S350L 是一个对象。
(2)Python 中的类
理解了类与对象,现在我们来看一下 Python 中的类,我们在开头的例子中提到,类是用来把有联系的数据和函数给组织起来的一种方式。
在类中,数据被称为属性,而函数则被称为方法。每个类都可以有零个或者多个属性,也可以有零个或者多个方法。
类的属性和方法的概念也很适合用来描述现实世界,比如“人”这个类,身高、体重、年龄等就是属性,而走路、吃饭、打球 等动作就是方法。
接下来,我们通过一个例子来演示怎么定义类,怎么使用类。
(3)定义一个“人”的类
现在我们尝试定义一个人的类。需要有年龄、性别、姓名三个属性,并提供两个方法:introduce 方法,打印一句介绍自己的话。get_age 方法,返回当前对象的 age。
# 类的定义环节class Person:def __init__(self):self.name = ""self.age = 0self.gender = ""def introduce(self):print("Hello, 我是" + self.name)print("我今年 " + str(self.age) + " 岁")print("另外, 我是" + self.gender)def get_age(self):return self.age# 使用上面定义的类xm = Person()xm.name = "小明"xm.age = "25"xm.gender = "男生"print(xm.get_age())xm.introduce()
简单说一下上面的代码的主要逻辑:
- 在定义环节:
- 定义了一个类:Person(类名首字母用大写),并用 init 函数中初始化了三个属性:name,age,gender。(init 函数,前后都有两个下划线,代表初始化函数。简单来说就是,当我们创建这个类的对象的时候,这个函数会被自动执行,不用你再手动执行一下)
- 定义了 introduce 方法,会把几个属性的值组合成一段自我介绍,这里我们通过方法参数 self.属性名 的形式来拿到类的属性的值。这种形式叫作**点语法。**点语法的本质就是找点前面的对象拿点后面的属性的值。比如 A.B 代表 A对象的B属性的值。
- 定义了 get_age 方法,把当前对象的 age 对象的值返回。可以看到,无论是 introduce 方法 还是 get_age 方法,都有一个 self 参数。这是 Python 的语法规定,类的方法的第一个参数都必须是 self。这样方法内部就能通过对 self 使用点语法来获取属性的值。
- 使用环节
- 注意从我们使用的类的代码开始,我们这里就没有缩进。因为有缩进就会被认为还是在类的内部,但这里的代码是在外部的,所以不用缩进;
- 我们创建了一个 Person 对象,存在变量 xm 中,之后逐一给它的属性赋值
- 打印 xm 对象的 get_age 方法返回的值
- 调用 xm 对象的 introduce
运行后,输出如下所示:
25Hello, 我是小明我今年 25 岁另外, 我是男生
可以看到,通过类的机制,我们成功地把数据和函数绑定在了一起。比如上面例子的数据:name、age、gender 和函数:get_age、introduce。并且类提供了一种机制(self 参数),能够让类的方法可以访问类的属性。
(4)给属性赋值
回看刚刚的例子,我们创建了 Person 类的对象之后,逐一对他的属性赋值,每一次属性赋值都需要针对 xm 对象使用点语法,比较麻烦,有没有更好的方式呢? 答案是肯定的。
我们稍微改造一下刚才的例子,改造后的代码如下所示:
class Person:def __init__(self, name, age, gender):self.name = nameself.age = ageself.gender = genderdef introduce(self):print("Hello, 我是" + self.name)print("我今年 " + str(self.age) + " 岁")print("另外, 我是" + self.gender)def get_age(self):return self.agexm = Person("小明", 25, "男生")print(xm.get_age())xm.introduce()
执行之后,输出和刚才是一致的:
25Hello, 我是小明我今年 25 岁另外, 我是男生
不同在哪儿呢?是这里。
现在我们不需要进行逐个属性的赋值,而是在构造对象的阶段就完成了几个属性的初始化。
这其中的奥妙就在 init 方法里,我们给 init 函数加了参数,然后用这些参数直接在 __init__的函数体中给我们的属性赋值。这样直接在创建对象的时候,把对应的值依次放在类名后的括号中,用逗号分隔。就能实现一次性地给属性都赋值,精简了代码。
2、常见的系统类
类和函数一样,类同样也有 Python 的开发者们提前写好提供给我们使用的类,一般称为系统类。上面举的例子都是我们自己实现的类,下面来看一下常用的系统类。
(1)列表(list)
列表相信大家都不陌生,在变量与数据类型文章中已经介绍过,它是 Python 非常常用的数据类型。
这里介绍一下列表的基础使用,代码如下:
# 创建列表a = [90,1,23, 15]# 访问某个元素(下表从 0 开始)print("Third number is ",a[2])# 使用 append 方法添加元素a.append(-1)print("After append: ", a)# 删除某个位置的元素del a[1]print("After delete second number: ", a)# 删除具体某个元素a.remove(15)print("After delete 15:", a)# 排序,默认从小到大a.sort()print("After sort", a)# 逆排序,从大到小a.sort(reverse = True)print("After reverse sort", a)# 求长度print("Length: ", len(a))
上面主要方法都有注释,这里不再展开,你可以结合以下输出和代码,加深理解。重要的是你要知道,如果之后让你做某个事(删除、排序等),你能直接使用 list 类对应的方法。而不需要自己写循环完成。
Third number is 23After append: [90, 1, 23, 15, -1]After delete second number: [90, 23, 15, -1]After delete 15: [90, 23, -1]After sort [-1, 23, 90]After reverse sort [90, 23, -1]Length: 3
(2)字符串(string)
字符串我们用的也是非常多的,在之前也介绍过一些基础用法,比如用 + 号来连接两个字符串,这里我们再介绍额外的一些用法。
a = "Hello,this is my home,welcome"# 添加字符a = a + ",pp"print("\nAfter append pp:\n", a)# 删除前 x 个字符,比如 3a = a[3:]print("\nAfter remove first 3 characters:\n", a)# 删除固定内容的字符a = a.replace("my","")print("\nAfter remove 'my':\n", a)# 替换固定内容成另外一个字符串a = a.replace("home", "company")print("\nAfter replace 'home' to 'company':\n",a)# 用某个字符分割字符串,返回一个数组,比如逗号str_list = a.split(",")print("\nList split from string by comma:\n", str_list)
输出:
After append pp:Hello,this is my home,welcome,ppAfter remove first 3 characters:lo,this is my home,welcome,ppAfter remove 'my':lo,this is home,welcome,ppAfter replace 'home' to 'company':lo,this is company,welcome,ppList split from string by comma:['lo', 'this is company', 'welcome', 'pp']
3、字典(dict)
字典和列表、字符串一样,也是 Python 中相对常用的数据类型,同样也是系统类。因为复杂一些,所以在变量与数据类型文章中没有介绍。
字典和列表类似,也是存储多个变量的容器。但与列表不同的是,字典存储的不仅仅只有变量,还有变量之间的映射关系。
我们举个例子,假设我们要存储班级里三位同学的年龄,可以用列表来完成,比如 a = [17, 18, 15], 但如果我们不仅要存储年龄,还得存储这三位同学的名字。那用列表就无能为力了。
简单来说,我们希望存储一个名字-年龄的映射关系。在这个例子上就是要存储这种形式的数据:小明:17,小红:18,小江:15。在 Python 中,这种有对应关系的两个变量我们称之为键值对。比如小明就是 键(key),而 17 就是值(value),一个键会唯一对应到一个值。
这里我们希望存储的就是三个键值对。而字典,就是专门用来存储键值对的容器。
# 创建一个空字典,用 花括号 {}d = {}print("Empty dict:", d)# 用键值对创建字典,键和值中间用冒号隔开,不同的键值对用逗号隔开d = {"xiaoming" : 3.5 , "xiaohong": 4}print("Two key-value pair dict:", d)# 访问字典的元素,和列表一样用中括号,但传入 key,查询 value。比如查询 xiaoming 的年龄print("Xiaoming's company age:", d["xiaoming"])# 添加新的键值对,直接对 key 对应的 value 赋值即可d["xiaogang"] = 5.5print("After append xiaogang:", d)# 删除键值对,类似列表删除,只是中括号内写 key,而不是序号del d["xiaohong"]print("After remove xiaohong:", d)# 修改键值对,同添加一样,直接对某个已有的 key 重新赋值即可d["xiaoming"] = 9.3print("After change xiaoming's value:", d)
输出:
Empty dict: {}Two key-value pair dict: {'xiaoming': 3.5, 'xiaohong': 4}Xiaoming's company age: 3.5After append xiaogang: {'xiaoming': 3.5, 'xiaohong': 4, 'xiaogang': 5.5}After remove xiaohong: {'xiaoming': 3.5, 'xiaogang': 5.5}After change xiaoming's value: {'xiaoming': 9.3, 'xiaogang': 5.5}
3、实战
现在有个任务,记录部门信息,每个部门都需要统计部门名称,员工列表,部门主管等信息。另外,还有两个要求:1、部门员工入职和离职都能方便的更新信息;2、可以方便查看某个部门的汇总信息。
问题分析:
- 目前我们需要记录的信息都是围绕部门这个实体的,所以我们可以用一个部门的类来记录,部门名称、员工列表和部门主管则都是这个类的属性。
- 另外的要求是有人员变动的时候方便更新信息,那其实就是部门类需要提供增加人员和减少人员的方法。
- 最后一个要求是方便查看部门汇总,那其实就是需要一个打印方法,来把部门的属性都打印出来。
我们接下来一步步来实现它。
(1)创建部门类
我们首先创建部门类和它的属性,部门名称和主管姓名,是字符串类型的变量,这两个属性我们通过初始化函数的参数来初始化。而员工列表是一个列表类型,我们先把它初始化成一个空列表。
class Department:def __init__(self, dep_name, boss_name):self.dep_name = dep_nameself.boss_name = boss_nameself.stuff_list = []
(2)实现员工的新增方法
目前员工列表,也就是 stuff_list 属性还是空列表,为了实现往这个列表增加员工的名字,我们需要在部门类增加一个添加员工的方法,我们就命名为 add_stuff。这个方法除了类方法必须带的 self 参数之外,只有一个参数:要添加的员工的姓名。
然后方法里面只需要把这个姓名添加到 stuff_list 即可,添加数据到列表,只需要调用列表对象的 append 方法即可。
class Department:def __init__(self,dep_name, boss_name):self.dep_name = dep_nameself.boss_name = boss_nameself.stuff_list = []# 新增代码def add_stuff(self, name):self.stuff_list.append(name)
(3)实现删除员工的方法
部门会新增员工,比如员工入职,或者调入。也可能会减少员工,比如员工离职,比如转岗去其他部门,所以我们同样需要一个删除员工的方法。
类似添加员工的设计,删除员工的方法同样需要员工姓名的参数,在方法内部调用 stuff_list 对象的 remove 方法来将员工从列表中移除。
class Department:def __init__(self,dep_name, boss_name):self.dep_name = dep_nameself.boss_name = boss_nameself.stuff_list = []def add_stuff(self, name):self.stuff_list.append(name)# 新增代码def remove_stuff(self, name):self.stuff_list.remove(name)
(4)添加打印部门信息的方法
打印部门信息,就是将部门的三个属性直接打印出来,比较简单,这里我们直接实现:
class Department:def __init__(self,dep_name, boss_name):self.dep_name = dep_nameself.boss_name = boss_nameself.stuff_list = []def add_stuff(self, name):self.stuff_list.append(name)def remove_stuff(self, name):self.stuff_list.remove(name)# 打印部门信息def print_dep_info(self):print("部门名称:",self.dep_name);print("主管名称:",self.boss_name);#这里使用上一篇文章中说到的len()函数来计算部门人数print("员工共个:",len(self.stuff_list));print("分别是:",self.stuff_list);
(5)使用部门类
现在,我们的部门类已经开发完毕了,现在让我们来用一用它,看看它是不是好使。
首先是记录部门信息,假设现在先记录2个部门:技术部和财务部。首先要分别创建两个部门的对象:
#创建技术部和财务部
it_dep = Department("技术部","小江");
finance_dep = Department("财务部","小红");#给技术部添加两个员工
it_dep.add_stuff("小江1");
it_dep.add_stuff("小江2");#给财务添加两个员工
finance_dep.add_stuff("小红1");
finance_dep.add_stuff("小红2");
#打印技术部和财务部信息
it_dep.print_dep_info();
finance_dep.print_dep_info();
执行后输出:
部门名称: 技术部
主管名称: 小江
员工共个: 2
分别是: ['小江1', '小江2']
部门名称: 财务部
主管名称: 小红
员工共个: 2
分别是: ['小红1', '小红2']
最后,当有人需要离职时,只需要调用 remove_stuff 方法即可,假设 技术部的小江2决定去创业,要离职,我们需要将他从部门列表中移除。
it_dep.remove_stuff("小江2")it_dep.print_dep_info()
输出
部门: 技术部主管: 小江员工共个:1员工: ['小江1']
可以看到,小江2已经不在员工列表中了。
至此,我们通过类与对象的方法,完成了部门信息统计的任务,并且可以非常方便地处理员工增加和减少的场景。
相关文章:

快乐学Python,Python基础之组织代码「类与对象」
在上一篇文章中,我们了解了函数。这一篇文章我们来了解一下Python中另外一个重要的概念:类与对象。 1、类与对象 (1)类与对象有什么关系? 你可能会奇怪,为什么要叫类与对象呢?是两个不同的东…...
H5的3D游戏开源框架
在H5的3D游戏框架中,Three.js、Babylon.js和Turbulenz是比较受欢迎的选择。 Three.js是一个广泛应用并且功能强大的JavaScript 3D库,可以创建简单的3D动画到创建交互的3D游戏。 Babylon.js是David Catuhe对3D游戏引擎热爱的结果,是最好的Ja…...
浅谈一些生命周期
vue2生命周期 beforeCreate :实例创建之初 created:组件已经创建完成 beforeMount:组件挂载之前 mounted:组件挂载之后 beforeUpdate:数据发生变化 更新之前 undated:数据发生之后 beforeDestroy :实…...

JavaScript基础(25)_dom查询练习(二)
<!DOCTYPE html> <html lang"zh"> <head><meta charset"UTF-8"><title>dom查询练习二</title><link rel"stylesheet" href"../browser_default_style/reset.css"><style>form {margi…...

【React系列】React生命周期、setState深入理解、 shouldComponentUpdate和PureComponent性能优化、脚手架
本文来自#React系列教程:https://mp.weixin.qq.com/mp/appmsgalbum?__bizMzg5MDAzNzkwNA&actiongetalbum&album_id1566025152667107329) 一. 生命周期 1.1. 认识生命周期 很多的事物都有从创建到销毁的整个过程,这个过程称之为是生命周期&…...

一文初步了解slam技术
本文初步介绍slam技术,主要是slam技术的概述,涉及技术原理、应用场景、分类、以及各自优缺点,和slam技术的未来展望。 🎬个人简介:一个全栈工程师的升级之路! 📋个人专栏:slam精进之…...

滑动窗口协议仿真(2024)
1.题目描述 滑动窗口协议以基于分组的数据传输协议为特征,该协议适用于在数据链路层以及传输层中对按 顺序传送分组的可靠性要求较高的环境。在长管道传输过程(特别是无线环境)中,相应的滑动窗口 协议可实现高效的重传恢复。附录 …...
uniapp上传文件时用到的api是什么?格式是什么?
在UniApp中,你可以使用uni.uploadFile()方法来上传文件。这是一个异步方法,用于将本地资源上传到服务器。 该方法的基本格式如下: uni.uploadFile({url: 上传接口地址,filePath: 要上传的文件路径,name: 后端接收的文件参数名,formData: {/…...

Java面试——框架篇
1、Spring框架中的单例bean是线程安全的吗? 所谓单例就是所有的请求都用一个对象来处理,而多例则指每个请求用一个新的对象来处理。 结论:线程不安全。 Spring框架中有一个Scope注解,默认的值就是singleton,单例的。一…...

GO语言笔记1-安装与hello world
SDK开发工具包下载 Go语言官网地址:golang.org,无法访问Golang中文社区:首页 - Go语言中文网 - Golang中文社区下载地址:Go下载 - Go语言中文网 - Golang中文社区 尽量去下载稳定版本,根据使用系统下载压缩包格式的安装…...

指针传参误区
C语言中指针作为形参传递时,func(*a, *b) 这种形式的话,是无法通过简单的 ab来修改的,在函数体内a的地址确实被修改成b的地址了,但是当函数执行结束时,a的地址会重新回到原本的地址里面…...
力扣-42.接雨水
题目: 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 示例 1: 输入:height [0,1,0,2,1,0,1,3,2,1,2,1] 输出:6 解释:上面是由数组[0,1,0,2…...

LeetCode-移动零(283)
题目描述: 给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。 请注意 ,必须在不复制数组的情况下原地对数组进行操作。 思路: 这里的思路跟以前做过的去重复数字的思路有点像&…...

文件系统与日志分析
一,文件系统 (一)inode 和block概述 1,文件数据包括元信息与实际数据 2,文件存储在硬盘上,硬盘最小存储单位是“扇区”,每个扇区存储512字节 3,block (块) 连续的八个扇区组成一…...

labview 与三菱FX 小型PLC通信(OPC)
NI OPC服务器与三菱FX3U PLC通讯方法 一、新建通道名称为:MIT 二、选择三菱FX系列 三、确认端口号相关的参数(COM端:7.波特率:9600,数据位:7,校验:奇校验,停止位…...
掌握Linux网络配置:价格亲民,操作简便!
前言 在Linux系统中,网络配置是实现连接、通信和安全的重要一环。无论你是初学者还是有经验的用户,掌握网络配置命令能帮助你轻松管理网络接口、设置IP地址以及查看连接状态。以下是一些关键命令和示例,让你快速掌握网络操作的精髓ÿ…...

郑州大学算法设计与分析实验2
判断题 1 #include<bits/stdc.h> using namespace std;const int N 50; int f[N], n;int main() { // freopen("1.in", "r", stdin);ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);cin >> n;f[1] 1; f[2] 1;for(int i 3; i &l…...

【CMake】1. VSCode 开发环境安装与运行
CMake 示例工程代码 https://github.com/LABELNET/cmake-simple 插件 使用 VSCode 开发C项目,安装 CMake 插件 CMakeCMake ToolsCMake Language Support (建议,语法提示) 1. 配置 CMake Language Support , Windows 配置 donet 环境 这…...
使用vue3+<script setup>+element-plus中el-table前端切片完成分页效果
<template><div><el-table :data"visibleData" :row-key"row > row.id"><el-table-column prop"name" label"姓名"></el-table-column><el-table-column prop"age" label"年龄&qu…...
vue 中 computed 和 watch 的区别
在Vue中,computed和watch都是用于监听数据的变化,并且根据变化做出相应的反应。 computed是一个计算属性,它会根据依赖的数据的变化自动计算得出一个新的值,并且具有缓存的特性。当依赖的数据发生变化时,computed属性…...
逻辑回归:给不确定性划界的分类大师
想象你是一名医生。面对患者的检查报告(肿瘤大小、血液指标),你需要做出一个**决定性判断**:恶性还是良性?这种“非黑即白”的抉择,正是**逻辑回归(Logistic Regression)** 的战场&a…...
FFmpeg 低延迟同屏方案
引言 在实时互动需求激增的当下,无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作,还是游戏直播的画面实时传输,低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架,凭借其灵活的编解码、数据…...

【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
ssc377d修改flash分区大小
1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...
MVC 数据库
MVC 数据库 引言 在软件开发领域,Model-View-Controller(MVC)是一种流行的软件架构模式,它将应用程序分为三个核心组件:模型(Model)、视图(View)和控制器(Controller)。这种模式有助于提高代码的可维护性和可扩展性。本文将深入探讨MVC架构与数据库之间的关系,以…...

P3 QT项目----记事本(3.8)
3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...

微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
Python 包管理器 uv 介绍
Python 包管理器 uv 全面介绍 uv 是由 Astral(热门工具 Ruff 的开发者)推出的下一代高性能 Python 包管理器和构建工具,用 Rust 编写。它旨在解决传统工具(如 pip、virtualenv、pip-tools)的性能瓶颈,同时…...