当我们拿到一个对象的引用时,如何知道这个对象是什么类型、有哪些方法呢?

使用type()

首先,我们来判断对象类型,使用type()函数:

基本类型都可以用type()判断:

  1. >>> type(123)
  2. <class 'int'>
  3. >>> type('str')
  4. <class 'str'>
  5. >>> type(None)
  6. <type(None) 'NoneType'>

如果一个变量指向函数或者类,也可以用type()判断:

  1. >>> type(abs)
  2. <class 'builtin_function_or_method'>
  3. >>> type(a)
  4. <class '__main__.Animal'>

但是type()函数返回的是什么类型呢?它返回对应的Class类型。如果我们要在if语句中判断,就需要比较两个变量的type类型是否相同:

  1. >>> type(123)==type(456)
  2. True
  3. >>> type(123)==int
  4. True
  5. >>> type('abc')==type('123')
  6. True
  7. >>> type('abc')==str
  8. True
  9. >>> type('abc')==type(123)
  10. False

判断基本数据类型可以直接写intstr等,但如果要判断一个对象是否是函数怎么办?可以使用types模块中定义的常量:

  1. >>> import types
  2. >>> def fn():
  3. ... pass
  4. ...
  5. >>> type(fn)==types.FunctionType
  6. True
  7. >>> type(abs)==types.BuiltinFunctionType
  8. True
  9. >>> type(lambda x: x)==types.LambdaType
  10. True
  11. >>> type((x for x in range(10)))==types.GeneratorType
  12. True

使用isinstance()

对于class的继承关系来说,使用type()就很不方便。我们要判断class的类型,可以使用isinstance()函数。

我们回顾上次的例子,如果继承关系是:

  1. object -> Animal -> Dog -> Husky

那么,isinstance()就可以告诉我们,一个对象是否是某种类型。先创建3种类型的对象:

  1. >>> a = Animal()
  2. >>> d = Dog()
  3. >>> h = Husky()

然后,判断:

  1. >>> isinstance(h, Husky)
  2. True

没有问题,因为h变量指向的就是Husky对象。

再判断:

  1. >>> isinstance(h, Dog)
  2. True

h虽然自身是Husky类型,但由于Husky是从Dog继承下来的,所以,h也还是Dog类型。换句话说,isinstance()判断的是一个对象是否是该类型本身,或者位于该类型的父继承链上。

因此,我们可以确信,h还是Animal类型:

  1. >>> isinstance(h, Animal)
  2. True

同理,实际类型是Dog的d也是Animal类型:

  1. >>> isinstance(d, Dog) and isinstance(d, Animal)
  2. True

但是,d不是Husky类型:

  1. >>> isinstance(d, Husky)
  2. False

能用type()判断的基本类型也可以用isinstance()判断:

  1. >>> isinstance('a', str)
  2. True
  3. >>> isinstance(123, int)
  4. True
  5. >>> isinstance(b'a', bytes)
  6. True

并且还可以判断一个变量是否是某些类型中的一种,比如下面的代码就可以判断是否是list或者tuple:

  1. >>> isinstance([1, 2, 3], (list, tuple))
  2. True
  3. >>> isinstance((1, 2, 3), (list, tuple))
  4. True
总是优先使用isinstance()判断类型,可以将指定类型及其子类“一网打尽”。

使用dir()

如果要获得一个对象的所有属性和方法,可以使用dir()函数,它返回一个包含字符串的list,比如,获得一个str对象的所有属性和方法:

  1. >>> dir('ABC')
  2. ['__add__', '__class__',..., '__subclasshook__', 'capitalize', 'casefold',..., 'zfill']

类似__xxx__的属性和方法在Python中都是有特殊用途的,比如__len__方法返回长度。在Python中,如果你调用len()函数试图获取一个对象的长度,实际上,在len()函数内部,它自动去调用该对象的__len__()方法,所以,下面的代码是等价的:

  1. >>> len('ABC')
  2. 3
  3. >>> 'ABC'.__len__()
  4. 3

我们自己写的类,如果也想用len(myObj)的话,就自己写一个__len__()方法:

  1. >>> class MyDog(object):
  2. ... def __len__(self):
  3. ... return 100
  4. ...
  5. >>> dog = MyDog()
  6. >>> len(dog)
  7. 100

剩下的都是普通属性或方法,比如lower()返回小写的字符串:

  1. >>> 'ABC'.lower()
  2. 'abc'

仅仅把属性和方法列出来是不够的,配合getattr()setattr()以及hasattr(),我们可以直接操作一个对象的状态:

  1. >>> class MyObject(object):
  2. ... def __init__(self):
  3. ... self.x = 9
  4. ... def power(self):
  5. ... return self.x * self.x
  6. ...
  7. >>> obj = MyObject()

紧接着,可以测试该对象的属性:

  1. >>> hasattr(obj, 'x') # 有属性'x'吗?
  2. True
  3. >>> obj.x
  4. 9
  5. >>> hasattr(obj, 'y') # 有属性'y'吗?
  6. False
  7. >>> setattr(obj, 'y', 19) # 设置一个属性'y'
  8. >>> hasattr(obj, 'y') # 有属性'y'吗?
  9. True
  10. >>> getattr(obj, 'y') # 获取属性'y'
  11. 19
  12. >>> obj.y # 获取属性'y'
  13. 19

如果试图获取不存在的属性,会抛出AttributeError的错误:

  1. >>> getattr(obj, 'z') # 获取属性'z'
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. AttributeError: 'MyObject' object has no attribute 'z'

可以传入一个default参数,如果属性不存在,就返回默认值:

  1. >>> getattr(obj, 'z', 404) # 获取属性'z',如果不存在,返回默认值404
  2. 404

也可以获得对象的方法:

  1. >>> hasattr(obj, 'power') # 有属性'power'吗?
  2. True
  3. >>> getattr(obj, 'power') # 获取属性'power'
  4. <bound method MyObject.power of <__main__.MyObject object at 0x10077a6a0>>
  5. >>> fn = getattr(obj, 'power') # 获取属性'power'并赋值到变量fn
  6. >>> fn # fn指向obj.power
  7. <bound method MyObject.power of <__main__.MyObject object at 0x10077a6a0>>
  8. >>> fn() # 调用fn()与调用obj.power()是一样的
  9. 81

简言之:判断某类有某方法么,获取方法给到某个指向/变量

Python实用笔记 (21)面向对象编程——获取对象信息的更多相关文章

  1. Python面向对象:获取对象信息

    学习笔记内容简介: 获取对象属性和方法的函数: type(): 判断对象类型 isinstance() : 判断class的类型 dir() : 获得一个对象的所有属性和方法 把属性和方法列出来是不够 ...

  2. 【Python】[面性对象编程] 获取对象信息,实例属性和类属性

    获取对象信息1.使用isinstance()判断class类型2.dir() 返回一个对象的所有属性和方法3.如果试图获取不存在的对象会抛出异常[AttributeError]4.正确利用对象内置函数 ...

  3. python 学习笔记7 面向对象编程

    一.概述 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发"更快更好更强..." ...

  4. python学习笔记(七):面向对象编程、类

    一.面向对象编程 面向对象--Object Oriented Programming,简称oop,是一种程序设计思想.在说面向对象之前,先说一下什么是编程范式,编程范式你按照什么方式来去编程,去实现一 ...

  5. python 面向对象五 获取对象信息 type isinstance getattr setattr hasattr

    一.type()函数 判断基本数据类型可以直接写int,str等: >>> class Animal(object): ... pass ... >>> type( ...

  6. Python学习笔记之面向对象编程(三)Python类的魔术方法

    python类中有一些方法前后都有两个下划线,这类函数统称为魔术方法.这些方法有特殊的用途,有的不需要我们自己定义,有的则通过一些简单的定义可以实现比较神奇的功能 我主要把它们分为三个部分,下文也是分 ...

  7. python学习笔记1 -- 面向对象编程类和实例

    由于之前有一定基础,所以python中的类的概接受的比较快,与其他语言一样, python也是通过类来进行事务的抽象,一切皆对象,要不然怎么说程序员不缺对象呢. 言归正传,python中类的声明是cl ...

  8. python学习笔记1 -- 面向对象编程高级编程1

    说起高级其实也就是些基础的东西,但是活用和熟用肯定会大幅度提升代码质量 首先要记录的是面向对象的灵活性,及如何去控制其灵活性,她允许你在实例中新增属性和方法,允许你给类新增属性和方法,也支持在定义类时 ...

  9. python 面向对象编程、获取对象信息

    面向对象与面向过程 参考链接:https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0 ...

随机推荐

  1. .net core 基于AspNetCore.Identity+Identityserver4用户的权限管理

    一般权限控制,是先给角色分配对应权限,然后再给用户分配角色:总权限应该是在代码编写的时候就已经固定了,例如有个用户更新的接口,这里就会诞生一个用户更新的权限,接口在权限就在,没有接口也就没有了这个权限 ...

  2. Rocket - tilelink - BankBinder

    https://mp.weixin.qq.com/s/oZCYBdy5glxJQmYKVWvpvA   简单介绍BankBinder的实现.   ​​   1. 基本介绍   A BankBinder ...

  3. 【JSP】el、jstl、MVC、三层架构

    JSP[Java Server Pages] [概念] java服务器页面 简化的servlet设计 html语法中扩展java 跨平台 JSP全名为Java Server Pages,中文名叫jav ...

  4. treegrid树形表格的完美运用

    一 问题描述: 树形表格TreeGrid在日常项目中还是运用的比较多的,哪我们在项目中,应该怎么引入和使用 TreeGrid呢? 二 使用步骤 1.首先我们需要在项目中,引入TreeGrid组件  需 ...

  5. Java实现 LeetCode 629 K个逆序对数组(动态规划+数学)

    629. K个逆序对数组 给出两个整数 n 和 k,找出所有包含从 1 到 n 的数字,且恰好拥有 k 个逆序对的不同的数组的个数. 逆序对的定义如下:对于数组的第i个和第 j个元素,如果满i < ...

  6. Java实现 蓝桥杯 算法训练 关联矩阵

    算法训练 关联矩阵 时间限制:1.0s 内存限制:512.0MB 提交此题 问题描述 有一个n个结点m条边的有向图,请输出他的关联矩阵. 输入格式 第一行两个整数n.m,表示图中结点和边的数目.n&l ...

  7. Java实现 蓝桥杯VIP 算法训练 一元三次方程

    问题描述 有形如:ax3+bx2+cx+d=0 这样的一个一元三次方程.给出该方程中各项的系数(a,b,c,d 均为实数),并约定该方程存在三个不同实根(根的范围在-100至100之间),且根与根之差 ...

  8. PAT 数字黑洞

    给定任一个各位数字不完全相同的 4 位正整数,如果我们先把 4 个数字按非递增排序,再按非递减排序,然后用第 1 个数字减第 2 个数字,将得到一个新的数字.一直重复这样做,我们很快会停在有“数字黑洞 ...

  9. CGLIB动态代理机制,各个方面都有写到

    CGLIB库介绍 代理提供了一个可扩展的机制来控制被代理对象的访问,其实说白了就是在对象访问的时候加了一层封装.JDK从1.3版本起就提供了一个动态代理,它使用起来非常简单,但是有个明显的缺点:需要目 ...

  10. 从源码研究如何不重启Springboot项目实现redis配置动态切换

    上一篇Websocket的续篇暂时还没有动手写,这篇算是插播吧.今天讲讲不重启项目动态切换redis服务. 背景 多个项目或微服务场景下,各个项目都需要配置redis数据源.但是,每当运维搞事时(修改 ...