感谢前辈们的指导,

深度优先:

广度优先:https://www.cnblogs.com/zhaof/p/7092400.html

个人将五篇博客里面的优点总结到一起,并怀着对前辈们的尊敬,以下均是纯手打,代码已经过验证

可以在python3.6.3中运行

面向对象是一种编程方式,需要类和对象来实现,面向对象编程其实就是对类和对象的使用

类就是指一个模板,模板里的的函数可以有多个,函数里实现一些功能

对象根据模板创建实例,通过实例对象执行类中的函数

面向对象;【创建对象】【通过对象执行方法】

class Foo:

    def Bar(self):

        print("Bar")

    def Hello(self,name):

        print("i am %s"%(name))

obj=Foo()

obj.Bar()

obj.Hello("wupeiqi")

面向对象三大特点:封装,继承,多态

一,封装:将内容放在某个地方,以后如果有需要,可以用某种函数调用此内容

第一步:将内容封装到某处:

第二步:从某处调用被封装的内容

第一步详解:

class Foo:

    def __init__(self,name,age):

        self.name=name

        self.age=age

#根据类对象Foo创建对象

#自动执行Foo类的__init__方法

obj1=Foo('狂',18)#将狂和18分别封装到name和age属性中

 

第二步详解:

从某处调用相关函数:

有两种方案(所有语言中的特性)

1,通过对象直接调用

2,通过self间接调用

我们先来看第一种

class Foo:

    def __init__(self,name,age):

        self.name=name

        self.age=age

obj1=Foo('wupeiqi',18)

print(obj1.name)

print(obj1.age)

obj2=Foo('alex',73)

print(obj2.name)

print(obj2.age)

 

第二种介绍:

class Foo:

    def __init__(self,name,age):

        self.name=name

        self.age=age

    def detail(self):

        print(self.name)

        print(self.age)

obj1=Foo('wupeiqi',18)

obj1.detail()

obj2=Foo('alex',73)

obj2.detail()

 

由此可见,第二种通过self的间接调用指的是通过函数去访问

练习一:在终端输出如下信息

  • 小明,10岁,男,上山去砍柴

  • 小明,10岁,男,开车去东北

  • 小明,10岁,男,最爱大保健

  • 老李,90岁,男,上山去砍柴

  • 老李,90岁,男,开车去东北

  • 老李,90岁,男,最爱大保健

  • class Foo:

  •     def __init__(self,name,age,sex):

  •         self.name=name

  •         self.age=age

  •         self.sex=sex

  •     def kanchai(self):

  •         print("%s,%s岁,%s,上山去砍柴" % (self.name, self.age, self.sex))

  •     def qudongbei(self):

  •         print("%s,%s岁,%s,开车去东北" % (self.name, self.age, self.sex))

  •     def dabaojian(self):

  •         print("%s,%s岁,%s,最爱大保健" % (self.name, self.age, self.sex))

  • xiaoming = Foo('小明',10, '男')

  • xiaoming.kanchai()

  • xiaoming.qudongbei()

  • xiaoming.dabaojian()

  • laoli = Foo('老李',90, '男')

  • laoli.kanchai()

  • laoli.qudongbei()

  • laoli.dabaojian()

 

二,继承:

继承,面向对象中的继承和现实生活中的继承相同,即:子可以继承父的内容

比如说 人 ,学生,工作者

学生和工作者就是人的两个扩展子类

class Animal:

    def eat(self):

        print('%s 吃'%self.name)

    def drink(self):

        print('%s 喝'%self.name)

    def shit(self):

        print('%s 拉'%self.name)

    def pee(self):

        print('%s 撒'%self.name)

    def pee(self):

        print('%s 撒'%self.name)

class Cat(Animal):

    def __init__(self,name):

        self.name=name

        self.bread='猫'

    def cry(self):

        print('喵喵叫')

class Dog(Animal):

    def __init__(self, name):

        self.name = name

        self.bread = '狗'

    def cry(self):

        print('汪汪叫')

c1 = Cat('小白家的小黑猫')

c1.eat()

c2 = Cat('小黑的小白猫')

c2.drink()

d1 = Dog('胖子家的小瘦狗')

d1.eat()

 

所以对于面向对象的继承来讲,其实就是将多个类共有的方法提取到父类中,子类仅需继承父类而不必实现每一种方法

那么一个很关键的问题来了,对于多继承,我们该怎么办呢,如何准备呢

是否可以继承多个类

如果继承多个类每个类中定义了相同的函数,那么哪一个会被使用呢

1、Python的类可以继承多个类,Java和C#中则只能继承一个类

2、Python的类如果继承了多个类,那么其寻找方法的方式有两种,分别是:深度优先和广度优先

对深度优先和广度优先的详解请见附录:

当类是经典类的时候,多继承情况下,会按照深度优先方式查找

当类是新式类的时候,多继承情况下,会按照广度优先方式查找

区分:当前类或者父类继承了object类,那么该类便是新式类,否则是经典类

class C1: #C1是经典类

    pass

class C2(object)#C1是经典类

    pass

class C1: #C1是新式类

    pass

class C2(C1)#C1是新式类

    pass

 

class D:

    def bar(self):

        print('D bar')

class C(D):

    def bar(self):

        print('C.bar')

class B(D):

    def bar(self):

        print('B.bar')

class A(B,C):

    def bar(self):

        print('A.bar')

a=A()

a.bar()

# 执行bar方法时# 首先去A类中查找,如果A类中没有,则继续去B类中找,如果B类中么有,则继续去C类中找,如果C类中么有,则继续去D类中找,如果还是未找到,则报错# 所以,查找顺序:A --> B --> C --> D# 在上述查找bar方法的过程中,一旦找到,则寻找过程立即中断,便不会再继续找了经典类:首先去A类中查找,如果A类中没有,则继续去B类中找,如果B类中么有,则继续去D类中找,如果D类中么有,则继续去C类中找,如果还是未找到,则报错新式类:首先去A类中查找,如果A类中没有,则继续去B类中找,如果B类中么有,则继续去C类中找,如果C类中么有,则继续去D类中找,如果还是未找到,则报错class Animal(object):    def __init__(self, name):  # Constructor of the class        self.name = name    def talk(self):  # Abstract method, defined by convention only        raise NotImplementedError("Subclass must implement abstract method")class Cat(Animal):    def talk(self):        print('%s: 喵喵喵!' % self.name)class Dog(Animal):    def talk(self):        print('%s: 汪!汪!汪!' % self.name)def func(obj):  # 一个接口,多种形态    obj.talk()c1 = Cat('小晴')d1 = Dog('李磊')func(c1)func(d1)正确代码Func函数需要接收一个F1类型或者F1子类的类型"""    obj.show()s1_obj = S1()Func(s1_obj) # 在Func函数中传入S1类的对象 s1_obj,执行 S1 的show方法,结果:S1.shows2_obj = S2()Func(s2_obj) # 在Func函数中传入Ss类的对象 ss_obj,执行 Ss 的show方法,结果:S2.show复制代码class F1:    passclass S1(F1):    def show(self):        print        'S1.show'class S2(F1):    def show(self):        print        'S2.show'# 由于在Java或C#中定义函数参数时,必须指定参数的类型# 为了让Func函数既可以执行S1对象的show方法,又可以执行S2对象的show方法,所以,定义了一个S1和S2类的父类# 而实际传入的参数是:S1对象和S2对象def Func(F1 obj):"""Func函数需要接收一个F1类型或者F1子类的类型"""     print(obj.show())s1_obj = S1()Func(s1_obj)  # 在Func函数中传入S1类的对象 s1_obj,执行 S1 的show方法,结果:S1.shows2_obj = S2()Func(s2_obj)  # 在Func函数中传入Ss类的对象 ss_obj,执行 Ss 的show方法,结果:S2.show错误代码,版本不对

深度优先和广度优先会在爬虫等多个领域进行运用

第一详解:

深度优先算法:

(1)访问初始顶点v并标记顶点v已访问。

(2)查找顶点v的第一个邻接顶点w。
(3)若顶点v的邻接顶点w存在,则继续执行;否则回溯到v,再找v的另外一个未访问过的邻接点。
(4)若顶点w尚未被访问,则访问顶点w并标记顶点w为已访问。
(5)继续查找顶点w的下一个邻接顶点wi,如果v取值wi转到步骤(3)。直到连通图中所有顶点全部访问过为止。

广度优先算法:

(1)顶点v入队列。

(2)当队列非空时则继续执行,否则算法结束。
(3)出队列取得队头顶点v;访问顶点v并标记顶点v已被访问。
(4)查找顶点v的第一个邻接顶点col。
(5)若v的邻接顶点col未被访问过的,则col入队列。
(6)继续查找顶点v的另一个新的邻接顶点col,转到步骤(5)。直到顶点v的所有未被访问过的邻接点处理完。转到步骤(2)。

第二详解:

深度优先是指网络爬虫会从起始页开始,一个链接一个链接跟踪下去,处理完这条线路之后再转入下一个起始页,继续追踪链接,通过下图进行理解:

注:scrapy默认采用的是深度优先算法

A-B-D-E-I-C-F-G-H (递归实现)

广度优先,有人也叫宽度优先,是指将新下载网页发现的链接直接插入到待抓取URL队列的末尾,也就是指网络爬虫会先抓取起始页中的所有网页,然后在选择其中的一个连接网页,继续抓取在此网页中链接的所有网页,通过下图进行理解:

A-B-C-D-E-F-G-H-I (队列实现)