08-装饰器&递归函数&栈-队列

一、装饰器【掌握】

1.案例

代码演示:

def test():
    print("拼搏到无能为力,坚持到感动自己")

f = test()  #变量可以指向指向函数,函数名也是一个变量,所以变量可以当做函数调用
f()

#思考问题:test增加功能,但是不能修改test函数内部----->装饰器

在代码运行期间,可以动态增加函数功能的方式,被称为装饰器【Decorator】

也就是说,在不修改原函数的基础上,给原函数增加功能

好处:在团队开发中,如果两个或者两个以上的程序员会用到相同的功能,但是功能又有细微的差别,采用装饰器:相互不影响,代码简化

2.使用

2.1简单装饰器

代码演示:

#1.简单的装饰器
def test():
    print("拼搏到无能为力,坚持到感动自己")

#a.书写闭包
#b.给外部函数设置参数,fun表示的是原函数
def outer(fun):
    def inner():
        # d.给原函数增加功能
        print("hello")

        #c.调用原函数
        fun()
    return inner

#e.使用闭包
f = outer(test)   #f = inner
f()   #inner()

#注意:增加的功能可以写在原函数调用的前面或者后面
#注意:outer函数就被称为装饰器


#练习:给下面的函数添加功能,打印九九乘法表
def show():
    for i in range(10):
        print(i)

def outer1(fun):
    def inner1():
        fun()
        for i in range(1,10):
            for j in range(1,i + 1):
                print("%dx%d=%d"%(j,i,i * j),end=" ")
            print("")
    return  inner1

f1 = outer1(show)
f1()

2.2有参数的装饰器

代码演示:

#2.原函数有参数的装饰器
def getAge(age):
    print(age)

getAge(10)
getAge(-5)

print("************")

#需求:在不修改原函数的基础上,进行数据的过滤:当用户输入age为负数的时候,则置为0
def wrapper(fun):
    #注意:当原函数有参数,装饰器的作用是为了操作原函数中的参数,给inner设置参数
    def inner(num):
        #增加新功能:过滤负数
        if num < 0:
            num = 0

        #调用原函数
        fun(num)  #age = num
    return  inner

f = wrapper(getAge)
f(10)   #num = 10
f(-5)

2.3系统的简写

代码演示:

#3.简化demo2中的操作:@装饰器的名称  应用到原函数中

#需求:在不修改原函数的基础上,进行数据的过滤:当用户输入age为负数的时候,则置为0
def wrapper(fun):
    #注意:当原函数有参数,装饰器的作用是为了操作原函数中的参数,给inner设置参数
    def inner(num):
        #增加新功能:过滤负数
        if num < 0:
            num = 0

        #调用原函数
        fun(num)  #age = num
    return  inner

#将wrapper装饰器应用在了getAge函数上,
@wrapper
def getAge(age):
    print(age)

getAge(10)
getAge(-5)

"""
@wrapper

等价于

f = wrapper(getAge)
f(10)   #num = 10

#注意;当使用@的时候,在同一个文件中,装饰器必须出现的原函数的前面

"""

2.4不定长参数的装饰器

代码演示:

#4.不定长参数的装饰器

#应用场景:当同一个装饰器作用于不同函数的时候,这些函数的参数的个数是不相同的
def wrapper(fun):
    def inner(*args):
        print("hello")

        fun(*args)   #a = args[0]   b = args[1]

    return  inner

@wrapper
def fun1(a,b):
    print(a + b)

@wrapper
def fun2(a,b,c,d):
    print(a,b,c,d)

fun1(10,20)   #args = (10,20)
fun2(1,2,3,4)

2.5多个装饰器作用于同一个函数

代码演示:

#5.将多个装饰器应用到同一个函数上
def wrapper1(fun):
    def inner1():
        print("1~~~~")
        fun()

    return inner1

def wrapper2(fun):
    def inner2():
        print("2~~~~")
        fun()

    return inner2

@wrapper1
@wrapper2
def show():
    print("hello")

show()

"""
1~~~~
2~~~~
hello
"""

#结论:多个装饰器作用于同一个函数的时候,从第一个装饰器开始,从上往下依次执行,但是,原函数只会被执行一次

二、函数递归【掌握】

1.概念

递归函数:一个会调用自身的函数【在一个函数的内部,自己调用自己】

递归调用

递归中包含了一种隐式的循环,他会重复指定某段代码【函数体】,但这种循环不需要条件控制

使用递归解决问题思路:

a.找到一个临界条件【临界值】

b.找到相邻两次循环之间的关系

c.一般情况下,会找到一个规律【公式】

2.使用

代码演示:

#案例一
"""
               1 2 3 4 5 6 7  8  9 10  11.。。。
斐波那契数列:1,1,2,3,5,8,13,21,34,55,89.....

解决问题:报一个数,输出数列中对应的数

规律:
a.第一个位置和第二个位置上数是固定的,都是1
b.第n个位置上的数:第 n - 1 的数  +   第 n - 2 的数

r1 = func1(1)  ------>1
r2 = func1(2)  ------>1
r3 = fun1(3) ------>func1(1) + func1(2)----->1 + 1 = 2
r4 = fun1(4)------->fun1(3) + fun1(2) ----->func1(1) + func1(2) +  fun1(2) ---->1  + 1 + 1 = 3
r5 = fun1(5) ----->fun1(4) + fun1(3) ----->fun1(3) + fun1(2) + func1(1) + func1(2)--->func1(1) + func1(2) ++ fun1(2) + func1(1) + func1(2)--->5
.....
rn = fun1(n) ----->fun1(n- 1) + fun1(n - 2)
"""

def func1(num):
    #临界值
    if num == 1 or num == 2:
        return 1
    else:
        #print("~~~~",num)
        result = func1(num- 1) + func1(num - 2)    #result = func1(1) + func1(2)  --->1 + 1 =2
        return result

print(func1(10))

#练习;使用递归计算1~某个数之间的和
"""
add(1) = 1   :临界值
add(2) = add(1) + 2
add(3) = add(2) + 3 ---->add(1) + 2 + 3 = 1 + 2 + 3
add(4) = add(3) + 4---->add(2) + 3 + 4 ---->add(1) + 2 + 3 + 4---->1 + 2 + 3 + 4
....
add(n) = add(n - 1) + n
"""
def add(num):

    """
    n = 1
    sum = 0
    while n <= 100:
        sum += n
        n += 1

    return sum

    sum1 = 0
    for i in range(1,num + 1):
        sum1 += i
    return  sum1
    """
    #使用递归实现
    if num == 1:
        return 1
    else:
        return add(num - 1) + num

print(add(100))

注意:以后在实际项目中尽量少用递归,如果隐式循环的次数太多,会导致内存泄漏【栈溢出】

优点:简化代码,逻辑清晰

三、栈和队列【了解】

用于存储数据的线性表

栈:在表的一端进行插入和删除

队列:在表的一端进行插入,在表的另一端进行数据的删除

1.栈

Stack

开口向上的容器:先进后出,后进先出

代码演示:

#list的底层维护了一个栈的线性表

myStack = []

#插入数据
#数据入栈【压栈】
myStack.append(1)
print(myStack)
myStack.append(2)
print(myStack)
myStack.append(3)
print(myStack)
myStack.append(4)
print(myStack)   #[1,2,3,4]

#出栈
myStack.pop()
print(myStack)
myStack.pop()
print(myStack)
myStack.pop()
print(myStack)
myStack.pop()
print(myStack)

2.队列

queue

水平放置的水管:先进先出,后进后出

代码演示:

import  collections   #数据结构的集合

queue  = collections.deque([1,2,3,4])
print(queue)

#入队【存储数据】
queue.append(5)
print(queue)
queue.append(6)
print(queue)

#出队【获取数据】
queue.popleft()
print(queue)
queue.popleft()
print(queue)
queue.popleft()
print(queue)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,192评论 6 511
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,858评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,517评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,148评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,162评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,905评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,537评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,439评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,956评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,083评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,218评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,899评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,565评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,093评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,201评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,539评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,215评论 2 358

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,400评论 8 265
  • 无意间看到朋友发了一条动态,说是教师编制笔试通过了,但是他自愿放弃了面试的机会。看到这条动态的时候,我有点惊讶。再...
    阿云阅读 1,155评论 0 1
  • 雪融冬去,盼春归,踏青赏梅。 暖阳抱,春风拂面,草青柳绿新。 潺潺水流,冰消尽,鸟嬉梅落响春雷。 雨如丝,润得残雪...
    心安何所归阅读 392评论 3 5
  • 静静今年21岁了,在这样一个原本花样的年华,本该令人羡慕的年龄,但与同村其他已经成了家的姑娘相比,总好像少了点啥。...
    悄然Edward阅读 730评论 0 2