1. 类(class)

下面的代码建立了一个Employee类:

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. class Employee(object):
  4. company = "IBM"
  5. def __init__(self, name, sex, age, salary):
  6. self.name = name
  7. self.sex = sex
  8. self.age = age
  9. self.__salary = salary
  10. def getSignature(self):
  11. signature = "My name is %s, I'm %d years old." % (self.name, self.age)
  12. return signature
  13. def getSalary(self):
  14. return self.__salary
  15. def setSalary(self, salary):
  16. if 0 < salary <= 10000:
  17. self.__salary = salary
  18. else:
  19. raise ValueError("Invalid Value")
  20. tom = Employee("tom", "male", 23, 3000)
  21. print(tom.getSignature())
  22. # My name is tom, I'm 23 years old.
  23. print(tom.age)
  24. # 23
  25. tom.setSalary(5000)
  26. tom.__salary = 9000 # 无效,其实是新增了一个名为"__salary"的变量
  27. print(tom.getSalary())
  28. # 5000

__init__方法相当于其它语言的“构造函数”,该方法的第一个参数必须为self,self代表创建的实例本身,因此在__init__方法内部可以把各种属性绑定到self;在实际调用时,self并不需要传递,Python解释器自己会把实例变量传进去。

以一个下划线开头的变量名,例如_company,这种变量外部是可以访问的,但是按照约定俗成的规定,这种变量应该视为私有变量,不要随意访问。

以两个下划线开头的变量名,例如__salary,是私有变量,外部不能直接访问,一般提供"get"和"set"方法去间接获取和修改。

开头与结尾都是两个下划线,例如__name__,是特殊变量,特殊变量是可以直接访问的。

需要注意的是,当在一个外部类尝试用下面的代码访问新建的Employee类时,是会报错的:

  1. import Employee
  2. tom = Employee("tom", "female", "23")

报错内容为TypeError: 'module' object is not callable,这个错误是指试图把模块作为一个函数来调用。产生错误的原因是,import Emplyee其实是导入了整个的Employee.py,而不是名为Employee的类。正确的做法有两种:

(1) 用“模块名.类名“来访问:

  1. import Employee
  2. tom = Employee.Employee("tom", "male", 23, 6000)

(2) 用"from...import..."的形式导入

  1. from Employee import *
  2. tom = Employee("tom", "male", 23, 6000)

2. 获取对象的类型

type(obj)函数返回参数的对象类型,基本类型如intstr也可以用它来判断:

  1. from Employee import *
  2. tom = Employee("tom", "male", 23, 6000)
  3. print(type(23))
  4. # <class 'int'>
  5. print(type("ABC"))
  6. # <class 'str'>
  7. print(type(tom))
  8. # <class 'Employee.Employee'>
  9. if type(123) == type(456):
  10. print("Equal")
  11. if type("ABC") == str:
  12. print("Equal")
  13. print(type(tom) == Employee)
  14. # True

可以使用types模块中定义的常量来判断一个对象是否是函数,lambda函数或generator:

  1. import types
  2. def myFun():
  3. pass
  4. # 是否是函数
  5. print(type(myFun) == types.FunctionType)
  6. # True
  7. # 是否是内置函数
  8. print(type(abs) == types.BuiltinFunctionType)
  9. # True
  10. # 是否是lambda函数
  11. print(type(lambda x: x)==types.LambdaType)
  12. # True
  13. # 是否是generator
  14. print(type((x for x in range(10)))==types.GeneratorType)
  15. # True

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

  1. print(isinstance(23,int))
  2. # True
  3. print(isinstance("ABC", str))
  4. # True
  5. print(isinstance(b"A", bytes))
  6. # True
  7. print(isinstance(tom, Employee))
  8. # True

isinstance()还可以用于判断一个对象是否是某些类型中的一种:

  1. print(isinstance("23", (str, int)))
  2. # True
  3. print(isinstance([1, 2, 3], (list, tuple)))
  4. # True

3. 获取对象的属性和方法

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

  1. print(dir("ABC"))
  2. # ['__add__', '__class__', ... , 'upper', 'zfill']

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

  1. print(len("ABC"))
  2. print("ABC".__len__())

下面的例子证明了len()会调用__len__()方法:

  1. class MyClass1(object):
  2. def __len__(self):
  3. return 100
  4. class MyClass2(object):
  5. pass
  6. myClass = MyClass1()
  7. print(len(myClass))
  8. # 100
  9. myClass = MyClass2()
  10. print(len(myClass))
  11. # TypeError: object of type 'MyClass2' has no len()

4. hasattr、getattr和setattr

利用这三个方法,可以判断对象是否有某属性/方法,获取指定名称的属性/方法,新增属性等操作:

  1. class Employee(object):
  2. def __init__(self, name, sex, age, salary):
  3. self.name = name
  4. self.sex = sex
  5. self.age = age
  6. self.__salary = salary
  7. def getSignature(self):
  8. signature = "My name is %s, I'm %d years old." % (self.name, self.age)
  9. return signature
  10. employee = Employee("tom", "male", 23, 3000)
  11. # 判断对象是否有"age"属性,有则打印并赋值
  12. if hasattr(employee, "age"):
  13. print(employee.age)
  14. employee.age = 18
  15. # 如果对象没有"hobby"属性,则新增该属性并赋值
  16. if not hasattr(employee, "hobby"):
  17. setattr(employee, "hobby", "music")
  18. # 通过getattr获取对象指定的属性值
  19. print(getattr(employee, "hobby"))
  20. # music
  21. # 如果试图获取不存在的属性,会抛出AttributeError的错误:
  22. # getattr(employee, "gold")
  23. # AttributeError: 'Employee' object has no attribute 'gold'
  24. # 利用getattr的第三个参数:如果属性不存在,就返回一个默认值
  25. print(getattr(employee, "gold", "not exist"))
  26. # not exist
  27. # 通过getattr获取方法,注意:如果方法不存在,会抛出AttributeError
  28. print(getattr(employee, "getSignature"))
  29. # <bound method Employee.getSalary of <__main__.Employee object at 0x10832a4a8>>
  30. # 判断是否存在指定名称的方法,如果存在,则执行该方法
  31. try:
  32. funcName = "getSignature"
  33. func = getattr(employee, funcName)
  34. if hasattr(func, "__call__"):
  35. print("存在方法", funcName)
  36. # 存在方法 getSignature
  37. print(func())
  38. # My name is tom, I'm 18 years old.
  39. except AttributeError as e:
  40. print("没有方法:", funcName)

Python基础笔记(五)的更多相关文章

  1. Python基础笔记系列十一:标准输入输出、文件读写和指针等操作

    本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! 标准输入输出一.输入 在sublime中这个时候需要安装SublimeRE ...

  2. Python基础笔记系列一:基本工具与表达式

    本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! 工具基础(Windows系统下)传送门:Python基础笔记系列四:工具的 ...

  3. Python基础学习五

    Python基础学习五 迭代 for x in 变量: 其中变量可以是字符串.列表.字典.集合. 当迭代字典时,通过字典的内置函数value()可以迭代出值:通过字典的内置函数items()可以迭代出 ...

  4. 我的Python基础笔记

    Python是从刚开始参加工作,就有听各方面的测试大牛推崇,但是刚开始做测试时还是把基础的测试方法放在第一位来学习的,直到半年多以后才开始接触Python. 我的Python基础主要是以廖雪峰老师的在 ...

  5. Python基础知识(五)------字典

    Python基础知识(四)------字典 字典 一丶什么是字典 ​ dict关键字 , 以 {} 表示, 以key:value形式保存数据 ,每个逗号分隔 ​ 键: 必须是可哈希,(不可变的数据类型 ...

  6. Python基础笔记1

    这篇笔记来自廖雪峰的Python教程. 一.Python基础 Python使用缩进来组织代码块,务必遵守约定俗成的习惯,坚持使用4个空格的缩进. 在文本编辑器中,需要设置把Tab自动转换为4个空格,确 ...

  7. python基础笔记-0

    python中数据结构,主要有列表.元组.字典.集合. python中最基本数据结构是序列(sequence).序列中每个元素被分配一个序号——即元素位置,也成为索引.第一个索引是0,第二个是1,以此 ...

  8. Python基础笔记系列五:元组

    本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! 元组 1)元组的结构和访问.使用方法和列表基本一致,区别主要有两点:1.使 ...

  9. python学习笔记五 模块下(基础篇)

    shevle 模块 扩展pickle模块... 1.潜在的陷进 >>> import shelve>>> s = shelve.open("nb" ...

随机推荐

  1. APICloud项目纪要

    一.页面之间的传递参数通过pageParam传递参数: api.openWin({ name: 'ware', url: './ware.html', pageParam: { wareId: 'w1 ...

  2. 由MQTT topic的正则表达式匹配引发的特殊字符"/"匹配思考

    正则表达式中的'/'替换 近期项目对接OneNET的MQTT物联网套件,需要完成命令下发流程. 流程要求: (1)设备在接收平台下发的命令(topic为$sys/{pid}/{device-name} ...

  3. oracle linux 7 yum报错解决:COULD NOT RESOLVE HOST: YUM.ORACLE.COM

    虚拟机中yum报错 [root@localhost ~]# yum -y install oracle-rdbms-server-11gR2-preinstall Loaded plugins: la ...

  4. 转载--从输入URL到页面展示到底发生了什么

    最近我也在看http协议, tcp相关知识, 在吃饭时无意看到来一篇文章讲解“从输入URL到页面展示到底发生了什么”, 细细看完, 很值得回味, 所以转载, 以供日后在温习. (PS, 作者这篇文章发 ...

  5. HDU5952 dfs+剪枝

    题目分析: 对于给出的n个点和m条边,求这个图的完全联通子图的数量(每次查询的子图的大小为s),对于本题而言,很容易想到的是dfs暴力和这个点相连的所有的点,并且判断这个图是否是度为s 的完全联通子图 ...

  6. java http get和post请求

    1.http工具类 package com.funshion.common.utils; import java.net.URI;import java.net.URL; import org.apa ...

  7. T4模板 简单使用

    原文:https://www.cnblogs.com/sanduo8899/p/3964563.html <#@ template debug="false" hostspe ...

  8. document.write和innerHTML的区别?

    document.write是直接重写整个页面,innerHTML针对所属DOM节点进行重写,效率优于document.write.

  9. 数学建模之Python操作csv文件

    1.用Python通过csv文件里面的某一列,形成键值,然后统计键在其他列出现的次数. import pandas as pd import numpy as np import csv import ...

  10. nuxtjs如何在单独的js文件中引入store和router

    nuxtjs里面集成vuex的创建方式改变了,并且官方不建议以导出Vuex实例的方式创建store,并且会在nuxt3里面删除.这样就会存在一个问题,我怎么像普通vue spa项目一样直接 impor ...