嵌套函数
函数内部定义的函数
【操作】定义
#嵌套内部函数的定义
def outer():
print("outer running")
def inner01():
print("inner running")
inner01()
outer()
【操作】使用嵌套函数避免重复代码
def printName(isChinese,name,familyName):
def inner_print(a,b):
print("{0} {1}".format(a,b))
if isChinese:
inner_print(familyName,name)
else:
inner_print(name,familyName)
printName(True,"小七","高")
printName(False,"George","Bush")
nonlocal 关键字
nonlocal 用来声明外层的局部变量。
global 用来声明全局变量。
【测试】 nonlocal、global 关键字的用法
a = 100
def outer ():
b = 10
def inner():
nonlocal b #生命外部函数的局部变量
print("inner",b)
b = 20
global a
a = 1000
inner()
print("outer b:",b)
outer()
print("a: ",a)
LEGB 规则
Local–>Enclosed–>Global–>Built in
Local 指的就是函数或者类的方法内部
Enclosed 指的是嵌套函数(一个函数包裹另一个函数,闭包)
Global 指的是模块中的全局变量
Built in 指的是 Python 为自己保留的特殊名称。
【测试】 LEGB
str = "global"
def outer():
str ="outer"
def inner():
str = "inner"
print(str)
inner()
outer()
面向对象编程
面向过程
执行者思维,简单,小型程序 ,微观细节,动词
面向对象
设计者思维,大规模程序,协作。宏观面向对象,顶层思维,小步骤仍然需要面向过程。名词
类的定义
我们通过类定义数据类型的属性(数据)和方法(行为),也就是说,“类将行为和状态打 包在一起”。
从一个类创建对象时,每个对象会共享这个类的行为(类中定义的方法),但会有自己的属 性值(不共享状态)。更具体一点:“方法代码是共享的,属性数据不共享”。
要点如下:
- 类名必须符合“标识符”的规则;一般规定,首字母大写,多个单词使用“驼峰原则”。
- 类体中我们可以定义属性和方法。
- 属性用来描述数据,方法(即函数)用来描述这些数据相关的操作。
【操作】一个典型的类的定义
class Student: #类名首字母大写,多个单词驼峰原则
def __init__(self,name,score):#self必须第一个
self .name ="gaoqo"
self.score = score
def say_score(self):
print("{0}分数:{1}".format(self .name,self .score ) )
s1 = Student ("高琪",18)
s1.say_score()
构造函数__init__()
Python 对象包含如下部分:
- id(identity 识别码)
- type(对象类型)
- value(对象的值)
(1) 属性(attribute)
(2) 方法(method)
创建对象,我们需要定义构造函数__init__()方法。构造方法用于执行“实例对象的初始化工 作”,即对象创建后,初始化当前对象的相关属性,无返回值。
init()的要点如下:
- 名称固定,必须为:init()
- 第一个参数固定,必须为:self。 self 指的就是刚刚创建好的实例对象。
- 构造函数通常用来初始化实例对象的实例属性,如下代码就是初始化实例属性:name 和 score
def __init__(self,name,score):
self.name = name #实例属性
self.score = score
- 通过“类名(参数列表)”来调用构造函数。调用后,将创建好的对象返回给相应的变量。 比如:s1 = Student(‘张三’, 80)
- init()方法:初始化创建好的对象,初始化指的是:“给实例属性赋值”
- new()方法: 用于创建对象,但我们一般无需重定义该方法。
- 如果我们不定义__init__方法,系统会提供一个默认的__init__方法。如果我们定义了带参 的__init__方法,系统不创建默认的__init__方法。
实例属性和实例方法
实例属性是从属于实例对象的属性,也称为“实例变量”。他的使用有如下几个要点:
- 实例属性一般在__init__()方法中通过如下代码定义:
self.实例属性名 = 初始值 - 在本类的其他实例方法中,也是通过 self 进行访问:
self.实例属性名 - 创建实例对象后,通过实例对象访问:
obj01 = 类名() #创建对象,调用__init__()初始化属性
obj01.实例属性名 = 值 #可以给已有属性赋值,也可以新加属性
实例方法是从属于实例对象的方法。
实例方法的定义格式如下:
def 方法名(self [, 形参列表]):
函数体
方法的调用格式如下:
对象.方法名([实参列表])
要点:
- 定义实例方法时,第一个参数必须为 self。和前面一样,self 指当前的实例对象。
- 调用实例方法时,不需要也不能给 self 传参。self 由解释器自动传参。
·
函数和方法的区别
1.都是用来完成一个功能的语句块,本质一样。
2.方法调用时,通过对象来调用。方法从属于特定实例对象,普通函数没有这个特点。
3.直观上看,方法定义时需要传递 self,函数不需要。
实例对象的方法调用本质
· 其他操作:
- dir(obj)可以获得对象的所有属性、方法
- obj.dict 对象的属性字典
- pass 空语句
- isinstance(对象,类型) 判断“对象”是不是“指定类型”
类对象、类属性、类方法、静态方法
类对象、类属性的理解
如下图,类对象与实例对象的位置不同,实例方法与类属性指向类对象,实例属性在实例对象内。
类方法与静态方法
类方法
类方法是从属于“类对象”的方法。
格式如下:
@classmethod
def 类方法名(cls [,形参列表]) :
函数体
要点如下:
- @classmethod 必须位于方法上面一行
- 第一个 cls 必须有;cls 指的就是“类对象”本身;
- 调用类方法格式:“类名.类方法名(参数列表)”。 参数列表中,不需要也不能给 cls 传值。
- 类方法中访问实例属性和实例方法会导致错误 5. 子类继承父类方法时,传入 cls 是子类对象,而非父类对象
静态方法
与“类对象”无关的方法。
和在模块中定义普通函数没有区别,只不过放到了“类的名字空间里”,需要通过“类调用”。
格式如下:
@staticmethod
def 静态方法名([形参列表]) :
函数体
要点如下:
- @staticmethod 必须位于方法上面一行
- 调用静态方法格式:“类名.静态方法名(参数列表)”。
- 静态方法中访问实例属性和实例方法会导致错误
__del__方法(析构函数)和垃圾回收机制
用于实现对象被销毁时所需的操作。
当对象没有被引用时(引用计数为 0),由垃圾回收器调用__del__方法。
我们也可以通过 del 语句删除对象,调用__del__方法。
class Person:
def __del__(self):
print("销毁对象{0}".format(self ) )
p1 = Person ()
p2 = Person ()
del p2
print("程序结束")
运算结果:
销毁对象:<main.Person object at 0x02175610>
程序结束
销毁对象:<main.Person object at 0x021755D0>
__call__方法和可调用对象
定义了__call__方法的对象,称为“可调用对象”,即该对象可以像函数一样被调用。
#测试可调用方法
class SalaryAccount:
'''工资计算类'''
def __call__(self,salary):
print("算工资啦...")
yearSalary = salary *12
daySalary = salary//22.5
hourSalary = daySalary //8
return dict(yearSalary =yearSalary,monthSalary=salary,daySalary =daySalary ,hourSalary =hourSalary )
s = SalaryAccount ()
print(s(3000))
运行结果:
{‘monthSalary’: 5000, ‘yearSalary’: 60000, ‘daySalary’: 166,‘hourSalary’: 20}