飞翔飞翔
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
  • 学习路径
  • 第1章 认识Python

    • Python 历史与特点
    • Python 2 与 Python 3 的核心差异
    • 安装与运行 Python 2.7.18
    • 编码规范 PEP 8
  • 第2章 基础语法

    • 变量与对象
    • 数字类型
    • 字符串 str
    • Unicode 字符串
    • 运算符
    • 空值 None
  • 第3章 流程控制

    • if 条件语句
    • if-else 条件语句
    • if-elif-else 多分支
    • 条件表达式(三元运算符)
    • while 循环
    • for 循环
    • range 与 xrange
    • 循环控制:break、continue、pass
    • 循环 else 子句
  • 第4章 数据结构

    • 列表基础
    • 列表方法
    • 列表推导式
    • 元组
    • 字典基础
    • 字典方法
    • 字典循环技巧
    • 集合
    • 序列解包
    • 序列比较
  • 第5章 函数

    • 定义函数
    • 参数传递机制
    • 默认参数
    • 关键字参数
    • 可变参数
    • Lambda 表达式
    • 文档字符串
    • 函数对象
  • 第6章 模块与包

    • import 导入
    • 模块搜索路径
    • name 与主程序
    • 编译文件 .pyc 与 .pyo
    • 包结构
    • dir() 函数
  • 第7章 文件与IO

    • 打开与关闭文件
    • 文件读写方法
    • with 上下文管理器
    • 格式化输出:% 操作符
    • 格式化输出:str.format()
    • JSON 序列化
  • 第8章 面向对象

    • 类定义与实例化
    • init 构造方法
    • 类变量与实例变量
    • 方法调用与 self
    • 继承基础
    • 多重继承
    • 新式类与旧式类
    • 私有变量与名称改写
    • 属性装饰器 property
    • 类方法与静态方法
    • 魔术方法
    • 空类与数据记录
  • 第9章 异常处理

    • 异常类型
    • try-except
    • try-except-else-finally
    • 抛出异常 raise
    • 自定义异常
    • with 语句与上下文管理器
  • 第10章 迭代器与生成器

    • 迭代器协议
    • 生成器函数
    • 生成器表达式
    • itertools模块
  • 第11章 标准库精要

    • os模块
    • sys模块
    • datetime模块
    • re模块
    • json模块
    • collections模块
    • math与random模块
    • urllib2与网络请求
    • subprocess与命令执行
    • threading与并发
    • unittest与测试
    • 虚拟环境与包管理
  • 第12章 工程实践

    • 调试技巧
    • 性能分析
    • 文档与注释
    • 下一步学习

方法调用与 self

Python 类中的方法与普通函数只有一个区别:第一个参数是实例本身的引用。这个参数通常命名为 self,由调用方隐式传入。理解 self 的工作机制,是理解 Python 面向对象设计的关键。

方法的本质

类中定义的函数,在通过类访问时是未绑定方法(unbound method),在通过实例访问时是绑定方法(bound method):

class Dog(object):
    def bark(self):
        return "woof!"

print Dog.bark          # <unbound method Dog.bark>

d = Dog()
print d.bark            # <bound method Dog.bark of <__main__.Dog object>>

绑定方法已经把实例 d "绑定"到方法的 self 参数上。调用 d.bark() 时,不需要传 self:

print d.bark()          # woof!

# 等价于
print Dog.bark(d)       # woof!,显式传入 self

self 不是关键字

self 只是约定俗成的参数名,不是关键字。你可以用其他名字,但绝对不要这样做:

class Bad(object):
    def bark(this):         # 语法合法,但极其不推荐
        return "woof!"

用 self 是 Python 社区的铁律。IDE、文档生成器、其他程序员都依赖这个约定。

方法调用过程

当调用 d.bark() 时,Python 执行以下步骤:

  1. 在 d.__dict__ 中查找 bark —— 找不到
  2. 在 Dog.__dict__ 中查找 bark —— 找到函数对象
  3. 因为是通过实例访问,把 d 和函数对象打包成绑定方法
  4. 调用绑定方法,自动把 d 作为第一个参数传入

如果通过类访问(Dog.bark(d)),步骤 3 跳过,需要显式传入实例。

在方法中调用其他方法

通过 self 调用同一实例的其他方法:

class Calculator(object):
    def __init__(self):
        self.result = 0
    
    def add(self, x):
        self.result += x
        return self
    
    def multiply(self, x):
        self.result *= x
        return self
    
    def get_result(self):
        return self.result
    
    def calculate(self):
        self.add(5).multiply(2).add(3)
        return self.get_result()

calc = Calculator()
print calc.calculate()      # 13,(0 + 5) * 2 + 3

每个方法返回 self,支持方法链式调用(method chaining)。

方法外的函数

函数定义不必在类体内部。可以把外部函数赋值给类属性:

def external_bark(self):
    return "%s says woof!" % self.name

class Dog(object):
    bark = external_bark
    
    def __init__(self, name):
        self.name = name

d = Dog("Fido")
print d.bark()          # Fido says woof!

这种技巧在动态修改类行为时有用,但通常会让代码更难理解,谨慎使用。

方法对象的内省

绑定方法对象有额外属性:

d = Dog("Fido")
method = d.bark

print method.im_self        # <__main__.Dog object>,绑定的实例
print method.im_func        # <function external_bark>,底层函数
print method.im_class       # <class '__main__.Dog'>,定义方法的类

这些属性在元编程、调试、框架开发中很有用。

全局作用域中的方法

方法内部可以访问全局名称(模块级别的变量和函数),但类本身不会成为方法的全局作用域:

module_var = 42

class MyClass(object):
    class_var = 100
    
    def method(self):
        print module_var        # 42,可以访问模块全局变量
        print self.class_var    # 100,通过 self 访问类变量
        # print class_var       # NameError!类不是全局作用域

方法的全局作用域是定义它的模块,不是类。访问类变量必须通过 self 或类名。

上一页
类变量与实例变量
下一页
继承基础