高阶函数

# -*- coding: utf-8 -*-
# @Time    : 2019/11/8 11:15
# @Author  : John
# @Email   : 2398344557@qq.com
# @File    : 高阶函数.py
# @Software: PyCharm
高阶函数与我.jpg

满足,输入:接受一个或多个函数,输出一个函数

【练习1】两数的绝对值相加

def add(x, y, f):
    return f(x) + f(y)    # 传入函数,进行调用
res = add(3, -6, abs)    # 传入函数名
print(res)
# —— 9

【练习2】

def method():
    x = 2
    def double(n):
        return n * x
    return double
d = method()
res2 = d(10)
print(res2)
# —— 20

Python系统高阶函数

  1. map()
  2. reduce()
  3. filter()
  4. sorted()
  • map(funciton, iterable)
# 1. map(funciton, iterable)
# 该函数使用函数作为自己的一个参数,
# 它可以将iterable(可迭代对象)中的数据依次传给function函数,
# 最后把处理的结果作为新的可迭代对象返回

# 【实例】把[1, 2, 3, 4, 5]变成[1, 4, 9, 16, 25]
# 非函数形式:
print([i for i in range(1, 6)])   # 输出[1, 2, 3, 4, 5]
print([i**2 for i in range(1, 6)])    # 输出[1, 4, 9, 16, 25]
# 函数形式:
# def power(x):
#     return x*x
# result = map(power(), [1, 2, 3, 4, 5])    # 函数, 可迭代对象
# print(result)
# # —— TypeError: power() missing 1 required positional argument: 'x'
# -------------------
# def power(x):
#     return x*x
# result = map(power, [1, 2, 3, 4, 5])    # 函数, 可迭代对象
# print(result)
# # —— <map object at 0x00000087F9188048>
# -------------------
def power(x):
    return x*x
result = map(power, [1, 2, 3, 4, 5])    # 函数(一个规则), 可迭代对象
print(list(result))
# —— [1, 4, 9, 16, 25]


# 【实例2】把[1, 2, 3, 4, 5, 6, 7, 8, 9]变成['1', '2', '3', '4', '5', '6', '7', '8', '9', ]
# 可以定义一个函数power()
def power(x):
    return str(x)
result_str_1 = map(power, [1, 2, 3, 4, 5, 6, 7, 8, 9])
print(list(result_str_1))
# —— ['1', '2', '3', '4', '5', '6', '7', '8', '9']
# --------------
# 也可以直接传入内置函数str()
result_str_2 = map(str, [1, 2, 3, 4, 5, 6, 7, 8, 9])
print(list(result_str_2))
# —— ['1', '2', '3', '4', '5', '6', '7', '8', '9']
# --------------
# ***拓展***

# 把[1, 2, 3, 4, 5, 6, 7, 8, 9]变成[1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0]
result3_float = map(float, [1, 2, 3, 4, 5, 6, 7, 8, 9])
print(list(result3_float))

# 查看数据类型
result4_type = map(type, ['   xuebi   ', 1, 1.0, {}, (), [], {1, 2}])
print(list(result4_type))
# —— [<class 'str'>, <class 'int'>, <class 'float'>, <class 'dict'>, <class 'tuple'>, <class 'list'>, <class 'set'>]

# 判断布尔值
result5_bool = map(bool, [1, [0], {0}, (0), 0, [], {}, ()])
print(list(result5_bool))
# —— [True, True, True, False, False, False, False, False]

# 查看id内存地址
result6_id = map(id, [1, 2, 3, 4, 5, 6, 7, 8, 9])
print(list(result6_id))
# —— [140706190963536, 140706190963568, 140706190963600, 140706190963632, 140706190963664, 140706190963696, 140706190963728, 140706190963760, 140706190963792]

# 查看长度
result7_len = map(len, [(1,), 'ab', {1, 2, 3}])
print(list(result7_len))
# —— [1, 2, 3]
  • reduce(function, sequence)
    从functools引入reduce:
from functools import reduce
# 2. reduce()
# reduce(func, sequence)函数,累积操作
# func函数必须接收两个参数,
# reduce会把fuc的运行结果做一个参数,
# 然后从sequence中依次取出一个数据当做另一个参数

# 【练习】1*2*3*4*5
def mul(x, y):
    return x*y
li = [1, 2, 3, 4, 5]
res = reduce(mul,li)
print(res)
# —— 120
# 或:用lambda
li = [1, 2, 3, 4, 5]
# 把mul()封装起来
# mul = lambda x, y: x*y
print(reduce((lambda x, y: x*y), li))
# —— 120
# 分析:
# reduce()函数的具体运行:
# 第一次运算时从迭代对象中取出两个参数:1*2=2
# 第二次运算时将第一次运行的结果2作为第一个参数传入,然后再从迭代对象中取出3作为第二个参数传入:2*3=6
# 同理:6*4=24
# ...
# 最后返回最后一次处理结果

# 【实例2】把序列[1, 3, 5, 7, 9]变成整数13579
def fn(x, y):
    return x*10+y
print(reduce(fn, [1, 3, 5, 7, 9]))
# —— 13579
# 或:把fn()封装起来
# fn = lambda x, y: x*10+y
print(reduce((lambda x, y: x*10+y), [1, 3, 5, 7, 9]))
# —— 13579

# 【实例3-1自己版】写出str转化为int的函数,把'13579'转化成13579
# 步骤:
# 1. 编写函数fn进累加移位
def fn(x, y):
    return x*10+y
# 2. 编写一个函数将char转化成num,map函数中的映射模式
def lis2(x):
    return int(x)
list_0 = map(lis2, '13579')
# print(list(list_0))
# 3. 使用map和reduce函数结合编写str转化为int的函数
print(reduce(fn, list_0))
# —— 13579

# 【实例3-2老师版】写出str转化为int的函数,把'13579'转化成13579
def fn(x, y):
    return x*10 + y
def char2num(s):
    digits = {'0': 0, '1': 1, '2': 2, '3': 3, '4': 4, '5': 5, '6': 6, '7': 7, '8': 8, '9': 9}
    return digits[s]
print(reduce(fn, map(char2num, '13579')))
# —— 13579

# 【实例3-3快到牛B版】写出str转化为int的函数,把'13579'转化成13579
print(int('13579'))
# —— 13579

# 【实例4】编写一个str2int函数,把实例3的封装起来
digits = {'0': 0, '1': 1, '2': 2, '3': 3, '4': 4, '5': 5, '6': 6, '7': 7, '8': 8, '9': 9}
def str2int(z):
    def fn(x, y):
        return x * 10 + y
    def char2num(c):
        return digits[c]

    return reduce(fn, map(char2num, z))
print(str2int('13579'))
# —— 13579
# 或:用lambda;
digits = {'0': 0, '1': 1, '2': 2, '3': 3, '4': 4, '5': 5, '6': 6, '7': 7, '8': 8, '9': 9}
def str2int(z):
    # 把fn()和char2num()封装起来
    # fn = lambda x, y: x * 10 + y
    # char2num = lambda c: digits[c]
    return reduce((lambda x, y: x*10+y), map((lambda c: digits[c]), z))
print(str2int('13579'))
# —— 13579
  • filter(function, iterable)
# 3. filter()
# filter(function, iterable)
# 根据函数function来过滤,
# 将iterable中的数据传入function中,
# 如果函数返回True,则保留该数据,否则就不保留

li = [i for i in range(1, 15)]
# 讲li中所有的奇数筛选出来
print(list(filter((lambda x: x % 2 == 1), li)))
# —— [1, 3, 5, 7, 9, 11, 13]
  • sorted()
# 4. sorted()
# 对数据进行排序,并且返回一个新的对象,原对象不变,
# key参数可以用来指定排序的规则,
# key接受的值是一个函数
# reverse参数是指定排序的顺序(升序还是降序),默认是升序

from random import randint

# 【练习1】
li1 = [randint(-10, 10) for _ in range(5)]
print('排序前:', li1)
res = li1.sort()
print('排序后:', li1)    # 这个才是
print('返回值:', res)
# —— 排序前: [-5, -4, 0, 9, 8]
# —— 排序后: [-5, -4, 0, 8, 9]
# —— None

# 【练习2】
li2 = [randint(-10, 10) for _ in range(5)]
print('排序前:', li2)
res2 = sorted(li2)
print('排序后:', li2)
print('返回值:', res2)    # 这个才是
# 排序前: [6, 7, -4, -4, -7]
# 排序后: [6, 7, -4, -4, -7]
# 返回值: [-7, -4, -4, 6, 7]

# 【练习3】
li3 = [randint(-10, 10) for _ in range(5)]
print('排序前:', li3)
res3 = sorted(li3, key=abs)    # 排序时按照绝对值形式排序
print('返回值:', res3)    # 但输出时是排序后在加回原来的符号的新排序
# 排序前: [-10, 2, -2, 2, 2]
# 返回值: [2, -2, 2, 2, -10]

# 【练习4】
chars = ['Sylar','Bennet', 'Hiro Nakamura', 'Peter Petrelli', 'Claire']
# # 排序前:
print('排序前:', chars)
# 按照字母的长度对chars进行降序排序
chars1 = sorted(chars, key=len, reverse=True)
print('排序后:', chars1)
# 排序前: ['Sylar', 'Bennet', 'Hiro Nakamura', 'Peter Petrelli', 'Claire']
# 排序后: ['Peter Petrelli', 'Hiro Nakamura', 'Bennet', 'Claire', 'Sylar']
# 或:使用lambda
# 排序前:
print('排序前:', chars)
chars2 = sorted(chars, key=lambda x: len(x), reverse=True)
print('排序后:', chars2)
# 排序前: ['Sylar', 'Bennet', 'Hiro Nakamura', 'Peter Petrelli', 'Claire']
# 排序后: ['Peter Petrelli', 'Hiro Nakamura', 'Bennet', 'Claire', 'Sylar']

# ***拓展***
# ----------------------------------
# 复合列表结构的列表排序
# 【练习5】
grade_list = [
    {'name': 'Bennet', 'grade': 90},
    {'name': 'Peter', 'grade': 30},
    {'name': 'Hiro', 'grade': 70},
    {'name': 'Sylar', 'grade': 52}
]
print('排序前:', grade_list)
# 按照成绩进行排序
grade_list = sorted(grade_list, key=lambda d: d['grade'])
print('排序后:', grade_list)
# —— 排序前: [{'name': 'Bennet', 'grade': 90}, {'name': 'Peter', 'grade': 30}, {'name': 'Hiro', 'grade': 70}, {'name': 'Sylar', 'grade': 52}]
# —— 排序后: [{'name': 'Peter', 'grade': 30}, {'name': 'Sylar', 'grade': 52}, {'name': 'Hiro', 'grade': 70}, {'name': 'Bennet', 'grade': 90}]

# 【练习6】
stu_info = [
    ('wunan', 22),
    ('jiulin', 39),
    ('hanlei', 17),
    ('zhenqi', 90)
]
print('排序前:', stu_info)
# 按照年龄进行降序排序
stu_info = sorted(stu_info, key=lambda x: x[1], reverse=True)
print('排序后:', stu_info)
# —— 排序前: [('wunan', 22), ('jiulin', 39), ('hanlei', 17), ('zhenqi', 90)]
# —— 排序后: [('zhenqi', 90), ('jiulin', 39), ('wunan', 22), ('hanlei', 17)]

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

推荐阅读更多精彩内容