Python学习--10 面向对象编程
面向对象编程——Object Oriented Programming,简称OOP,是一种程序设计思想。OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数。
本节对于面向对象的概念不做展开说明。本节主要内容是Python里如何使用面向对象编程。
分下面几部分:
1、类的格式
2、类的实例
3、类的封装
4、继承和多态
类的格式
下面是一个示例:
student.py
class Student(object):
def __init__(self, name, score):
self.name = name
self.score = score
def print_score(self):
print('%s: %s' % (self.name, self.score))
通过例子可以发现:
1、类由class开头,类名首字母大写,类名后面的括号表示继承自基类object,所有类都会继承这个类;
2、类的构造方法是__init__,第一个参数是固定的,永远是self,类里面通过self调用属性和方法,不同于JAVA里使用this。
现在来实例化类:
stu = Student('yjc', 22)
print(stu)
print(stu.name)
stu.print_score()
输出:
<__main__.Student object at 0x028811F0>
yjc
yjc: 22
Python里实例化类不用new关键字,而是像使用函数那样即可。调用类里的方法的时候不用传self参数。
类的封装
默认的,我们实例化了类后,可以直接访问对象里的属性,也可以去修改对象里的属性:
stu = Student('yjc', 22)
print(stu.name)
stu.print_score()
stu.name = 'Allen'
stu.print_score()
输出:
yjc
yjc: 22
Allen: 22
如果要让内部属性不被外部访问,可以把属性的名称前加上两个下划线__,在Python中,实例的变量名如果以__开头,就变成了一个私有变量(private),只有内部可以访问,外部不能访问(子类里也不能继承),所以,我们把Student类改一改:
class Student(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def print_score(self):
print('%s: %s' % (self.__name, self.__score))
stu = Student('yjc', 22)
print(stu.name)
输出:
Traceback (most recent call last):
File "/Projects/python/code/class/Student.py", line 12, in <module>
print(stu.__name)
AttributeError: 'Student' object has no attribute '__name'
这时候如果直接访问,就会报错。
继承和多态
在OOP程序设计中,当我们定义一个class的时候,可以从某个现有的class继承,新的class称为子类(Subclass),而被继承的class称为基类、父类或超类(Base class、Super class)。
继承示例:
class Animal(object):
def run(self):
print('runing')
class Dog(Animal):
pass
以上Dog类继承了Animal类,从而获得父类的方法run():
dog = Dog()
dog.run()
输出:
runing
这里子类虽然没有写run()方法,但由于父类已经拥有,所以可以直接继承过来。
需要注意的是父类的私有变量(__开头的)子类是不能访问,也不能继承的。
再看看类的多态:
class Dog(Animal):
def run(self):
print('Dog is runing')
class Cat(Animal):
def run(self):
print('Cat is runing')
def runClass(obj):
obj.run()
runClass(Animal())
runClass(Dog())
runClass(Cat())
输出:
runing
Dog is runing
Cat is runing
通过例子我们可以看到:
1、子类如果定义了和父类一样的方法,子类的会覆盖父类;
2、方法runClass接收一个Animal类,只要含有run()方法,均可正常运行,原因就在于多态。
类属性和实例属性
由于Python是动态语言,根据类创建的实例可以任意绑定属性。对于上面的Animal类,我们可以动态绑定新的属性:
a = Animal()
a.name = 'lala'
但这个绑定的属性属于实例,不属于类,其它Animal的实例是不能访问的。
在类里直接定义的属性,称为类属性,所有实例均可访问。
获取对象信息
type
最简单的,我们可以使用type()获取对象的类型:
>>> type(1)
<class 'int'>
>>> type('1')
<class 'str'>
>>> type(True)
<class 'bool'>
>>> type(1.2)
<class 'float'>
>>> type(None)
<class 'NoneType'>
>>> type(abs)
<class 'builtin_function_or_method'>
>>> type(lambda x:x*2)
<class 'function'>
>>> type([1,2])
<class 'list'>
>>> type((1,2))
<class 'tuple'>
>>> type({"name":"yjc"})
<class 'dict'>
>>> type((x for x in range(10)))
<class 'generator'>
还可以使用types模块进行判断:
>>> import types
>>> def f():pass
...
>>>
>>> type(f)==types.FunctionType
True
>>> type(f)==types.BuiltinFunctionType
False
>>> type(lambda x: x)==types.LambdaType
True
>>> type((x for x in range(10)))==types.GeneratorType
True
isinstance
对于类的继承关系,使用isinstance()会更方便:
class Animal(object):
def run(self):
print('runing')
class Dog(Animal):
pass
dog = Dog()
isinstance(dog, Dog)
isinstance(dog,object)
dir
如果要获得一个对象的所有属性和方法,可以使用dir()函数:
>>> import math
>>> dir(math)
['__doc__', '__loader__', '__name__', '__package__', '__spec__', 'acos', 'acosh', 'asin', 'asinh', 'atan', 'atan2', 'atanh', 'ceil', 'copysign', 'cos', 'cosh', 'degrees', 'e', 'erf', 'erfc', 'exp', 'expm1', 'fabs', 'factorial', 'floor', 'fmod', 'frexp', 'fsum', 'gamma', 'hypot', 'isfinite', 'isinf','isnan', 'ldexp', 'lgamma', 'log', 'log10', 'log1p', 'log2', 'modf', 'pi', 'pow', 'radians', 'sin','sinh', 'sqrt', 'tan', 'tanh', 'trunc']
类似__xxx__的属性和方法在Python中都是有特殊用途的,比如__len__方法返回长度。
getattr/setattr/hasattr
getattr()可以获取类的某个属性,setattr()可以设置类的某个属性,getattr()可以判断类是否拥有某个属性:
class Animal(object):
def __init__(self):
self.type = 'animal'
self.area = 'China'
def run(self):
print('runing')
class Dog(Animal):
pass
dog = Dog()
print(hasattr(dog,'type'))
print(hasattr(dog,'run'))
print(hasattr(dog,'name'))
print(getattr(dog,'area'))
# print(getattr(dog,'name'))
setattr(dog, 'name', 'Aia')
print(getattr(dog,'name'))
输出:
True
True
False
China
Aia
使用getattr()尝试获取对象的一个不存在属性会报错。可以通过设置默认值避免:
print(getattr(dog,'city', 'beijing')) #beijing
Python学习--10 面向对象编程的更多相关文章
- Python学习之==>面向对象编程(二)
一.类的特殊成员 我们在Python学习之==>面向对象编程(一)中已经介绍过了构造方法和析构方法,构造方法是在实例化时自动执行的方法,而析构方法是在实例被销毁的时候被执行,Python类成员中 ...
- 从0开始的Python学习014面向对象编程
简介 到目前为止,我们的编程都是根据数据的函数和语句块来设计的,面向过程的编程.还有一种我们将数据和功能结合起来使用对象的形式,使用它里面的数据和方法这种方法叫做面向对象的编程. 类和对象是面向对象 ...
- Python学习之==>面向对象编程(一)
一.面向对象与面向过程 面向对象与面向过程是两种不同的编程范式,范式指的是按照什么方式去编程.去实现一个功能.不同的编程范式本质上代表对各种不同类型的任务采取不同的解决问题的思路. 1.面向过程编程 ...
- Python开发——10.面向对象编程进阶
一.isinstance(obj,cls)和issubclass(sub,super) 1.isinstance(obj,cls) 判断obj是不是由cls产生的类 2.issubclass(sub, ...
- python学习_数据处理编程实例(二)
在上一节python学习_数据处理编程实例(二)的基础上数据发生了变化,文件中除了学生的成绩外,新增了学生姓名和出生年月的信息,因此将要成变成:分别根据姓名输出每个学生的无重复的前三个最好成绩和出生年 ...
- Python:笔记(3)——面向对象编程
Python:笔记(3)——面向对象编程 类和面向对象编程 1.类的创建 说明:和Java不同的是,我们不需要显示的说明类的字段属性,并且可以在后面动态的添加. 2.构造函数 构造函数的功能毋庸置疑, ...
- Python学习day40-并发编程(终)
figure:last-child { margin-bottom: 0.5rem; } #write ol, #write ul { position: relative; } img { max- ...
- Python学习day39-并发编程(各种锁)
figure:last-child { margin-bottom: 0.5rem; } #write ol, #write ul { position: relative; } img { max- ...
- Python学习day38-并发编程(线程)
figure:last-child { margin-bottom: 0.5rem; } #write ol, #write ul { position: relative; } img { max- ...
随机推荐
- 大批量烧写openwrt系统
http://wiki.openwrt.org/toh/tp-link/tl-wr1043nd OEM mass flashing Flashing hundreds of devices using ...
- Add GUI to connect to SQL
(*********************************************************************************) (* *) (* Below i ...
- jquery-validate的用法
默认校验规则 (1)required:true 必输字段(2)remote:"check.php" 使用ajax方法调用check.p ...
- Mysql基础之字符集与乱码
原文:Mysql基础之字符集与乱码 Mysql的字符集设置非常灵活 可以设置服务器默认字符集 数据库默认字符集 表默认字符集 列字符集 如果某一个级别没有指定字符集,则继承上一级. 以表声明utf8为 ...
- 在Android中自动实现横竖屏切换的问题
http://developer.android.com/training/basics/supporting-devices/screens.html参照Google推荐的做法 在你项目的res 文 ...
- 【C#版本详情回顾】C#3.0主要功能列表
隐式类型的本地变量和数组 在与本地变量一起使用时,var 关键字指示编译器根据初始化语句右侧的表达式推断变量或数组元素的类型 对象初始值设定项 支持无需显式调用构造函数即可进行对象初始化 集合初始值设 ...
- ASP.NET MVC + EF 利用存储过程读取大数据
ASP.NET MVC + EF 利用存储过程读取大数据,1亿数据测试很OK 看到本文的标题,相信你会忍不住进来看看! 没错,本文要讲的就是这个重量级的东西,这个不仅仅支持单表查询,更能支持连接查询, ...
- [转]SQL2005后的ROW_NUMBER()函数的应用
SQL Server 2005后之后,引入了row_number()函数,row_number()函数的分组排序功能使这种操作变得非常简单 分组取TOP数据是T-SQL中的常用查询, 如学生信息管理系 ...
- error: C1083: 无法打开包括文件:“QDomDocument”“QAxObject”
包含了头文件但是提示无法打开包括文件,是需要在项目的.pro里面手动加上一个变量 针对QAxObject是 QT += axcontainer 针对QDomDocument是 QT ...
- EntityFramework中支持BulkInsert扩展
EntityFramework中支持BulkInsert扩展 本文为 Dennis Gao 原创技术文章,发表于博客园博客,未经作者本人允许禁止任何形式的转载. 前言 很显然,你应该不至于使用 Ent ...