一、概述

一般的高阶语言都有反射的功能特性,python也不例外,网上资料显示,python支持类反射和模块反射,今天就先学习一下类反射的相关知识,模块反射后续再展开把。Python的类反射用于把字符串(对应于属性或方法)反射成内存中的地址对象,以便按需调用,实现某些特性的动态装配,它主要通过hasattr()、getattr()、setattr()、和delattr()来实现,类似于数据库的增删改查操作把。

二、反射函数

以下内容主要思想转自师兄张其高的博客https://www.cnblogs.com/zhangqigao/articles/6947023.html

2.1 hasattr(obj, name_str)

作用:判断一个对象obj中是否有对应的name_str字符串所代表的属性或者方法,返回布尔值。注意这里的对象可以是类,也可以是实例化的对象。

  1 class Dog(object):
2
3 def __init__(self, name):
4 self.name = name
5
6 def eat(self, food):
7 print('%s is eating %s' % (self.name, food))
8
9 obj1 = Dog('XiaoHuang')
10
11 print('obj1:\t %s ' % hasattr(obj1, 'name')) #判断属性
12 print('class Dog:\t %s' % hasattr(Dog, 'eat')) #判断方法
13
14 input = input(">>>:").strip() #动态输入字符串,动态判断
15 print(hasattr(obj1, input))
16
17 输出:
18 obj1: True
19 class Dog: True
20 >>>:food
21 False

2.2 getattr(obj, name_str)

作用:根据字符串name_str获取obj对象中的对应方法的内存地址或者对应属性的值

  1 class Dog(object):
2
3 def __init__(self, name):
4 self.name = name
5
6 def eat(self, food):
7 print('%s is eating %s' % (self.name, food))
8
9 obj1 = Dog('XiaoHuang')
10 obj2 = Dog('Bark')
11
12 print('obj1:\t %s ' % getattr(obj1, 'name')) #返回属性地址
13 print('class Dog:\t %s' % getattr(Dog, 'eat')) #返回方法地址
14 print('obj1:\t %s' % getattr(obj1, 'eat')) #返回方法地址
15 print('obj2:\t %s' % getattr(obj2, 'eat'))
16
17 结果输出:
18 obj1: XiaoHuang
19 class Dog: <function Dog.eat at 0x00000000021CEEA0>
20 obj1: <bound method Dog.eat of <__main__.Dog object at 0x00000000021CCCC0>>
21 obj2: <bound method Dog.eat of <__main__.Dog object at 0x00000000021CCCF8>>

这里有一个小疑问,既然类中的方法都保存在类中,并且只保存一份,这里通过obj1和obj2去获取的类方法的地址显示是不同的,这是为什么呢?原因待脑补后补充。

2.3 setattr(obj, name_str, value)

作用:给obj对象添加一个新属性或者新方法,setattr(x, 'y', v) is equivalent to ``x.y = v''

  1. 给对象新增一个方法

      1 class Dog(object):
    2
    3 def __init__(self, name):
    4 self.name = name
    5
    6 def eat(self, food):
    7 print('%s is eating %s' % (self.name, food))
    8
    9 def bulk(self): #预期新增的方法
    10 print('%s is yelling...' % self.name)
    11
    12 obj1 = Dog('XiaoHuang')
    13
    14 str_input = input(">>>:").strip()
    15 setattr(obj1, str_input, bulk)
    16 func = getattr(obj1, str_input)
    17 func(obj1)
    18
    19 print('obj1:\t %s ' % getattr(obj1, str_input))
    20 print('class Dog:\t %s' % getattr(Dog, str_input))
    21
    22 结果输出:
    23 >>>:hehe
    24 XiaoHuang is yelling...
    25 obj1: <function bulk at 0x0000000001CF3E18>
    26 Traceback (most recent call last):
    27 File "D:/python/S13/Day6/testclass.py", line 28, in <module>
    28 print('class Dog:\t %s' % getattr(Dog, str_input))
    29 AttributeError: type object 'Dog' has no attribute 'hehe'
    30
    31 Process finished with exit code 1

    注意:
    (1) 这里新增的方法只是对实例化的对象有效,对类无效
    (2) 最后在获取func的内存地址后,也就是bulk的内存地址后,调用时需要传入实例化对象方法,因为定义的新增bulk方法有参数self,而这种动态新增的方法,参数无论是否为self,都需要手动传入

  2. 给对象新增一个属性

      1 # !/usr/bin/env python
    2 # -*- coding:utf-8 -*-
    3
    4 class Dog(object):
    5
    6 def __init__(self, name):
    7 self.name = name
    8
    9 def eat(self, food):
    10 print('%s is eating %s' % (self.name, food))
    11
    12
    13 obj1 = Dog('XiaoHuang')
    14
    15 str_input = input(">>>:").strip()
    16 setattr(obj1, str_input, 3)
    17 print(getattr(obj1, str_input))
    18 print('obj1:\t %s ' % getattr(obj1, str_input))
    19 print('----')
    20 print(obj1.__dict__)
    21
    22 输出:
    23 >>>:age
    24 3
    25 obj1: 3
    26 ----
    27 {'name': 'XiaoHuang', 'age': 3}

上述示例程序显示,对象新增的属性可以通过__dict__来查看到,新增成功。

2.4 delattr(obj, name_str)

作用:删除obj对象中的属性或者方法,delattr(x, 'y') is equivalent to ``del x.y''

  1 class Dog(object):
2
3 def __init__(self, name):
4 self.name = name
5
6 def eat(self, food):
7 print('%s is eating %s' % (self.name, food))
8
9
10 obj1 = Dog('XiaoHuang')
11
12 str_input = input(">>>:").strip()
13 delattr(obj1, str_input)
14 print(Dog.__dict__)
15 print('----')
16 print(obj1.__dict__)
17 print(getattr(obj1, str_input))
18
19 输出:
20 #删除属性
21 >>>:name
22 Traceback (most recent call last):
23 File "D:/python/S13/Day6/testclass.py", line 24, in <module>
24 print(getattr(obj1, str_input))
25 AttributeError: 'Dog' object has no attribute 'name' #访问被删除的name属性报错
26 {'__module__': '__main__', '__init__': <function Dog.__init__ at 0x000000000219EEA0>, 'eat': <function Dog.eat at 0x000000000219EF28>, '__dict__': <attribute '__dict__' of 'Dog' objects>, '__weakref__': <attribute '__weakref__' of 'Dog' objects>, '__doc__': None}
27 ----
28 {} #实例的唯一属性name不见了
29
30 #删除方法
31 >>>:eat
32 Traceback (most recent call last):
33 File "D:/python/S13/Day6/testclass.py", line 20, in <module>
34 delattr(obj1, str_input)
35 AttributeError: eat #方法删除后立马报错了
36

三、综合运用

上面主要讲述的属性或方法的动态增、删、改、查的基本用法,实际上删除和增加用的很少,这里结合查、删、改来简述一下综合运用。

  1 # !/usr/bin/env python
2 # -*- coding:utf-8 -*-
3
4 class Dog(object):
5
6 def __init__(self, name):
7 self.name = name
8
9 def eat(self, food):
10 print('%s is eating %s' % (self.name, food))
11
12
13 obj1 = Dog('XiaoHuang')
14
15 str_input = input(">>>:").strip()
16 if hasattr(obj1, str_input):
17 print('Before change')
18 print('obj1\t%s:%s' % (str_input, getattr(obj1, str_input)))
19 print('--------')
20 setattr(obj1, str_input, 'Jinba')
21 print('After change')
22 print('obj1\t%s:%s' %(str_input, getattr(obj1, str_input)))
23 else:
24 setattr(obj1, str_input, None) #对于不存在的属性设置为None,感觉意义不大?
25 print(getattr(obj1, str_input))
26
27 输出:
28 >>>:name
29 Before change
30 obj1 name:XiaoHuang
31 --------
32 After change
33 obj1 name:Jinba
34
35

day7-python类反射的更多相关文章

  1. Python类总结-反射及getattr,setattr

    类反射的四个基本函数 hasattr getattr setattr delattr #反射 class BlackMedium: feature = 'Ugly' def __init__(self ...

  2. python 类知识点总结

    python 类知识点总结 面向对象思想: 1.设计的时候,一定要明确应用场景 2.由对象分析定义类的时候,找不到共同特征和技能不用强求 1.简述类.对象.实例化.实例这些名词的含义: 类:从一组对象 ...

  3. python类:magic魔术方法

    http://blog.csdn.net/pipisorry/article/details/50708812 魔术方法是面向对象Python语言中的一切.它们是你可以自定义并添加"魔法&q ...

  4. (转)python类:magic魔术方法

    原文:https://blog.csdn.net/pipisorry/article/details/50708812 版权声明:本文为博主皮皮http://blog.csdn.net/pipisor ...

  5. 如何访问python类中的私有方法

    在python中,不像c#/java类语言,支持类的私有方法,这点有点像objc,虽然objc可以通过扩展extension来实现,但源于objc的运行时特性,我们还是可以通过非常手段来进行访问的.不 ...

  6. python类的相关知识第二部分

    类的继承.多态.封装 一.类的继承 1.应用场景: 类大部分功能相同,大类包含小类的情况 例如: 动物类 共性:都要吃喝拉撒.都有头有脚 特性: 猫类.走了很轻,叫声特别,喜欢白天睡觉 狗类.的叫声很 ...

  7. Python进阶----反射(四个方法),函数vs方法(模块types 与 instance()方法校验 ),双下方法的研究

    Python进阶----反射(四个方法),函数vs方法(模块types 与 instance()方法校验 ),双下方法的研究 一丶反射 什么是反射: ​ 反射的概念是由Smith在1982年首次提出的 ...

  8. Python类中super()和__init__()的关系

    Python类中super()和__init__()的关系 1.单继承时super()和__init__()实现的功能是类似的 class Base(object): def __init__(sel ...

  9. LightMysql:为方便操作MySQL而封装的Python类

    原文链接:http://www.danfengcao.info/python/2015/12/26/lightweight-python-mysql-class.html mysqldb是Python ...

随机推荐

  1. 软工网络15团队作业1——团队组队&展示

    一.团队展示 1.队名:想不出队名 2.队员学号(标记组长) 201521123064 郭炜埕 201521123066 郑晓丽 201521123067 廖怡洁 201521123068 包梦榕 2 ...

  2. DevOps实践之持续集成Jenkins(最新版本测试)

    一.安装Jenkins (1)下载并安装配置Java Development Kit 8 [root@localhost ~]# rpm -ivh jdk-8u161-linux-x64.rpm Pr ...

  3. CSS 布局 - 水平 & 垂直对齐

    CSS 布局 - 水平 & 垂直对齐 一.元素居中对齐 要水平居中对齐一个元素(如 <div>), 可以使用 margin: auto;. 设置到元素的宽度将防止它溢出到容器的边缘 ...

  4. 利用python获取nginx服务的ip以及流量统计信息

    #!/usr/bin/python #coding=utf8 log_file = "/usr/local/nginx/logs/access.log" with open(log ...

  5. 20145230熊佳炜《网络对抗》实验九:web安全基础实践

    20145230熊佳炜<网络对抗>实验九:web安全基础实践 webgoat webgoat的中文是代罪羔羊的意思,而它是一个有很多漏洞的web应用程序,我们可以利用它来研究关于web应用 ...

  6. 谈谈let与const

    let 命令 let命令用于声明变量,但是与传统var命令的不同之处在于拥有以下特性: 使用let命令声明的变量只在let命令所在的代码块内有效(我将之称为变量绑定): 不存在变量提升: 存在暂时性死 ...

  7. MR案例:小文件处理方案

    HDFS被设计来存储大文件,而有时候会有大量的小文件生成,造成NameNode资源的浪费,同时也影响MapReduce的处理效率.有哪些方案可以合并这些小文件,或者提高处理小文件的效率呢? 1). 所 ...

  8. ssh-keygen 的 详解

    为了让两个Linux机器之间使用ssh不需要用户名和密码.所以采用了数字签名RSA或者DSA来完成这个操作. 模型分析 假设 A (192.168.20.59)为客户机器,B(192.168.20.6 ...

  9. struts2中各个jar包作用 (转)

    Struts2.3.4 所需的Jar包及介绍 Jar包的分类 jar包名称 jar包版本 jar包 文件名 jar包 的作用 jar包内包含的主要包路径及主要类 依赖的自有jar包名称 依赖的第三方j ...

  10. tornado之WebSocket

    WebSocket WebSocket是HTML5规范中新提出的客户端-服务器通讯协议,协议本身使用新的ws://URL格式. WebSocket 是独立的.创建在 TCP 上的协议,和 HTTP 的 ...