python3入门笔记六之面向对象编程

简介: 这是我参与更文挑战的第24天,活动详情查看: 更文挑战不知不觉也来到python的面向对象教程阶段了😄,看到这几个字立刻就想到这个面向对象的三大特征 1. 封装 2. 继承 3. 多态面向对象编程 Object Oriented Programming,简称OOP面向对象和面向过程的区别

这是我参与更文挑战的第24天,活动详情查看: 更文挑战


不知不觉也来到python的面向对象教程阶段了😄,看到这几个字立刻就想到这个面向对象的三大特征 1. 封装 2. 继承 3. 多态


面向对象编程 Object Oriented Programming,简称OOP


面向对象和面向过程的区别


区别就是有木有对象 哈哈,面向对象都是通过这个对象去传递消息的, 例如 写代码时,面向对象都是先考虑有没有这个对象,没有的话要去创建这个对象(Object),而对象是类(Class)里面的一个实例(Instance), 在这个类里会有该类对象的特有属性,还有行为(方法)。 而面向过程呢,没有这个类和对象的概念,代码里出现的都是封装好的函数和变量。


例如打印学生的成绩 面向过程:


std1 = { 'name': 'Michael', 'score': 98 }
std2 = { 'name': 'Bob', 'score': 81 }
def print_score(std):
    print('%s: %s' % (std['name'], std['score']))
复制代码


面向对象:


class Student(object):
    age=18 # 类属性
    def __init__(self, name, score):
        self.name = name # 实例属性
        self.score = score # 实例属性
    def print_score(self):
        print('%s: %s' % (self.name, self.score))
bart = Student('Bart Simpson', 59)
lisa = Student('Lisa Simpson', 87)
print(bart) # <__main__.Student object at 0x000001F7D6996DF0>
bart.print_score() # Bart Simpson: 59
lisa.print_score() # Lisa Simpson: 87
print(Student.age) # 18
print(bart.age)     # 18
复制代码


从上面这个面向对象的例子中,


  1. class Student(object): 表示一个Student类,继承object类,类名一般首字母大写,和java一样.


  1. print(bart) 可以看到该对象是属于Student类,内存地址为后面的0x000001F7D6996DF0


  1. def __init__(self, name, score):方法(注意是双下划线)


注意到__init__方法的第一个参数永远是self,表示创建的实例本身,因此,在__init__方法内部,就可以把各种属性绑定到self,因为self就指向创建的实例本身


  1. 有java构造器的feel,不同的是每个方法的第一个参数都是self


  1. 实例属性: 通过实例变量或者self进行绑定


  1. 类属性: 如上age=18


访问限制(封装)


private变量:双下划线开头并且不能以双下划线结尾的 (不能直接访问) 特殊变量:   双下划线开头并且以双下划线结尾 (能直接访问)


有些时候,你会看到以一个下划线开头的实例变量名,比如_name,这样的实例变量外部是可以访问的,但是,按照约定俗成的规定,当你看到这样的变量时,意思就是,“虽然我可以被访问,但是,请把我视为私有变量,不要随意访问”。 双下划线开头的实例变量是不是一定不能从外部访问呢?其实也不是。不能直接访问__name是因为Python解释器对外把__name变量改成了_Student__name,所以,仍然可以通过_Student__name来访问__name变量:


网络异常,图片无法展示
|


通过将变量私有化,提供get/set方法去获取/设置这个变量,来提高程序的健壮性(毕竟在方法中可以对参数进行处理)。


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))
    def get_score(self):
        return self.__score
    def set_score(self, score):
        self.__score = score
bart = Student('Bart Simpson', 59)
print(bart.__name) # 'Student' object has no attribute '__name'
print(bart._Student__name) # Bart Simpson
bart.set_score(88)
print(bart.get_score()) #88
复制代码


继承


被继承的class称为基类、父类或超类(Base class、Super class)。 例子:父类:Animal,子类Dog


class Animal(object):
    __age=1
    def run(self):
        print('Animal is running...')
    def set_age(self,age):
        self.__age=age
    def get_age(self):
        return self.__age
class Dog(Animal):
    def run(self):
        print('Dog is running...')
class Cat(Animal):
    def run(self):
        print('Cat is running...')
d=Dog()
d.run()
d.set_age(2)
print(d.get_age())
print(d.__age)
复制代码


额,这个python的写法确实好简洁呀!直接将括号中的类改掉即可。


继承和java的一样,继承就有了父类的公有属性和方法,毕竟私有的访问不了。


多态


什么是多态呢? 多态是指事物的多种形态。比如上面的这个动物例子,动物都会跑,但是猫和狗的跑的方法是不一样的。


额。。这个java有重载和重写,分别对应编译时的多态和运行时的多态。但是这个python是解释型的。。这个应该就只有运行时的多态。


运行时的多态,这个可是面向对象最精髓的东西了。


多态的好处就是,当我们需要传入Dog、Cat、Tortoise……时,我们只需要接收Animal类型就可以了,因为Dog、Cat、Tortoise……都是Animal类型,然后,按照Animal类型进行操作即可。由于Animal类型有run()方法,因此,传入的任意类型,只要是Animal类或者子类,就会自动调用实际类型的run()方法,这就是多态的意思

来感受下这个例子: 定义一个函数,接受animal类型


def run_twice(animal):
    animal.run()
    animal.run()
run_twice(Animal())
# Animal is running...
# Animal is running...
run_twice(Dog())
# Dog is running...
# Dog is running...
复制代码


运行时的多态的两个特点:


  1. 方法重写


  1. 父类引用指向子类对象


静态语言VS动态语言


对于静态语言(例如Java)来说,如果需要传入Animal类型,则传入的对象必须是Animal类型或者它的子类,否则,将无法调用run()方法。 对于Python这样的动态语言来说,则不一定需要传入Animal类型。我们只需要保证传入的对象有一个run()方法就可以了:


🤨


这就是动态语言的“鸭子类型”,它并不要求严格的继承体系,一个对象只要“看起来像鸭子,走起路来像鸭子”,那它就可以被看做是鸭子。Python的“file-like object“就是一种鸭子类型。对真正的文件对象,它有一个read()方法,返回其内容。但是,许多对象,只要有read()方法,都被视为“file-like object“。许多函数接收的参数就是“file-like object“,你不一定要传入真正的文件对象,完全可以传入任何实现了read()方法的对象。


注意:动态语言的鸭子类型特点决定了继承不像静态语言那样是必须的。


获取对象信息


type()


用于判断对象类型, 返回对应的Class类型 判断一个对象是否是函数,可以使用types模块中定义的常量


import types
print(type(123))  #<class 'int'>
print(type('hello world'))  #<class 'str'>
print(type(abs))  #<class 'builtin_function_or_method'>
def fn():
    pass 
print(type(fn)) #<class 'function'>
print(types.BuiltinFunctionType) #<class 'builtin_function_or_method'>
print(types.FunctionType) #<class 'function'>
复制代码


isinstance()


这个之前就用得很多啦,


print(isinstance('hello',str)) # True
# 是不是其中的一种类型
print(isinstance('hello',(int,list,str))) # True
复制代码


能用type()判断的基本类型也可以用isinstance()判断 总是优先使用isinstance()判断类型,可以将指定类型及其子类“一网打尽”


dir()


获得一个对象的所有属性和方法,,它返回一个包含字符串的list


print(dir(str))
 ['__add__', '__class__', '__contains__', '__delattr__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__getitem__', '__getnewargs__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__iter__', '__le__', '__len__', '__lt__', '__mod__', '__mul__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__rmod__', '__rmul__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__',
 'capitalize', 'casefold', 'center', 'count', 'encode', 'endswith', 'expandtabs', 'find', 'format', 'format_map', 'index', 'isalnum', 'isalpha', 'isascii', 'isdecimal', 'isdigit', 'isidentifier', 'islower', 'isnumeric', 'isprintable', 'isspace', 'istitle', 'isupper', 'join', 'ljust', 'lower', 'lstrip', 'maketrans', 'partition', 'replace', 'rfind', 'rindex', 'rjust', 'rpartition', 'rsplit', 'rstrip', 'split', 'splitlines', 'startswith', 'strip', 'swapcase', 'title', 'translate', 'upper', 'zfill']
复制代码


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


len('ABC') # 3
'ABC'.__len__() # 3
# 重写len方法
class MyDog(object):
    def __len__(self):
        return 10
len(MyDog()) #10
复制代码


getattr()、setattr()以及hasattr()


直接操作一个对象的状态: 要注意的是,只有在不知道对象信息的时候,我们才会去获取对象信息。


len('ABC')
'ABC'.__len__()
class MyDog(object):
    def __init__(self):
        self.x=9
    def power(self):
        return self.x**2
    def __len__(self):
        return 10
d=MyDog()
print(len(d)) #  10
print(d.x) # 9
print(hasattr(d,'x')) # True
if(not hasattr(d,'y')):
    setattr(d,'y',100)
print('y is {}'.format(getattr(d,'y'))) # y is 100
# 试图获取不存在的属性,会抛出AttributeError的错误:
print('z is {}'.format(getattr(d,'z'))) # AttributeError: 'MyDog' object has no attribute 'z'
print('z is {}'.format(getattr(d,'z',222))) # 可以给个默认值 222 ,不存在时返回。
fn=getattr(d,'power') # 可以获取函数
print(d.power()) #81
print(fn()) # 81
复制代码


正确用法


假设我们希望从文件流fp中读取图像,我们首先要判断该fp对象是否存在read方法,如果存在,则该对象是一个流,如果不存在,则无法读取。hasattr()就派上了用场。 请注意,在Python这类动态语言中,根据鸭子类型,有read()方法,不代表该fp对象就是一个文件流,它也可能是网络流,也可能是内存中的一个字节流,但只要read()方法返回的是有效的图像数据,就不影响读取图像的功能。


def readImage(fp):
    if hasattr(fp, 'read'):
        return readData(fp)
    return None




目录
相关文章
|
25天前
|
存储 数据采集 人工智能
Python编程入门:从零基础到实战应用
本文是一篇面向初学者的Python编程教程,旨在帮助读者从零开始学习Python编程语言。文章首先介绍了Python的基本概念和特点,然后通过一个简单的例子展示了如何编写Python代码。接下来,文章详细介绍了Python的数据类型、变量、运算符、控制结构、函数等基本语法知识。最后,文章通过一个实战项目——制作一个简单的计算器程序,帮助读者巩固所学知识并提高编程技能。
|
30天前
|
机器学习/深度学习 数据可视化 数据挖掘
使用Python进行数据分析的入门指南
本文将引导读者了解如何使用Python进行数据分析,从安装必要的库到执行基础的数据操作和可视化。通过本文的学习,你将能够开始自己的数据分析之旅,并掌握如何利用Python来揭示数据背后的故事。
|
26天前
|
IDE 程序员 开发工具
Python编程入门:打造你的第一个程序
迈出编程的第一步,就像在未知的海洋中航行。本文是你启航的指南针,带你了解Python这门语言的魅力所在,并手把手教你构建第一个属于自己的程序。从安装环境到编写代码,我们将一步步走过这段旅程。准备好了吗?让我们开始吧!
|
26天前
|
测试技术 开发者 Python
探索Python中的装饰器:从入门到实践
装饰器,在Python中是一块强大的语法糖,它允许我们在不修改原函数代码的情况下增加额外的功能。本文将通过简单易懂的语言和实例,带你一步步了解装饰器的基本概念、使用方法以及如何自定义装饰器。我们还将探讨装饰器在实战中的应用,让你能够在实际编程中灵活运用这一技术。
38 7
|
27天前
|
关系型数据库 开发者 Python
Python编程中的面向对象设计原则####
在本文中,我们将探讨Python编程中的面向对象设计原则。面向对象编程(OOP)是一种通过使用“对象”和“类”的概念来组织代码的方法。我们将介绍SOLID原则,包括单一职责原则、开放/封闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则有助于提高代码的可读性、可维护性和可扩展性。 ####
|
27天前
|
开发者 Python
Python中的装饰器:从入门到实践
本文将深入探讨Python的装饰器,这一强大工具允许开发者在不修改现有函数代码的情况下增加额外的功能。我们将通过实例学习如何创建和应用装饰器,并探索它们背后的原理和高级用法。
41 5
|
26天前
|
机器学习/深度学习 人工智能 算法
深度学习入门:用Python构建你的第一个神经网络
在人工智能的海洋中,深度学习是那艘能够带你远航的船。本文将作为你的航标,引导你搭建第一个神经网络模型,让你领略深度学习的魅力。通过简单直观的语言和实例,我们将一起探索隐藏在数据背后的模式,体验从零开始创造智能系统的快感。准备好了吗?让我们启航吧!
67 3
|
30天前
|
Python
Python编程入门:从零开始的代码旅程
本文是一篇针对Python编程初学者的入门指南,将介绍Python的基本语法、数据类型、控制结构以及函数等概念。文章旨在帮助读者快速掌握Python编程的基础知识,并能够编写简单的Python程序。通过本文的学习,读者将能够理解Python代码的基本结构和逻辑,为进一步深入学习打下坚实的基础。
|
26天前
|
人工智能 数据可视化 数据挖掘
探索Python编程:从基础到高级
在这篇文章中,我们将一起深入探索Python编程的世界。无论你是初学者还是有经验的程序员,都可以从中获得新的知识和技能。我们将从Python的基础语法开始,然后逐步过渡到更复杂的主题,如面向对象编程、异常处理和模块使用。最后,我们将通过一些实际的代码示例,来展示如何应用这些知识解决实际问题。让我们一起开启Python编程的旅程吧!
|
13天前
|
Unix Linux 程序员
[oeasy]python053_学编程为什么从hello_world_开始
视频介绍了“Hello World”程序的由来及其在编程中的重要性。从贝尔实验室诞生的Unix系统和C语言说起,讲述了“Hello World”作为经典示例的起源和流传过程。文章还探讨了C语言对其他编程语言的影响,以及它在系统编程中的地位。最后总结了“Hello World”、print、小括号和双引号等编程概念的来源。
101 80