python高级编程--高阶函数、匿名函数、返回函数、闭包、装饰器、偏函数

高阶函数

函数式编程

函数是Python内建支持的一种封装,我们通过把大段代码拆成函数,通过一层一层的函数调用,就可以把复杂任务分解成简单的任务,这种分解可以称之为面向过程的程序设计。函数就是面向过程的程序设计的基本单元。而函数式编程(请注意多了一个“式”字)——Functional Programming,虽然也可以归结到面向过程的程序设计,但其思想更接近抽象的计算。我们首先要搞明白计算机(Computer)和计算(Compute)的概念。在计算机的层次上,CPU执行的是加减乘除的指令代码,以及各种条件判断和跳转指令,所以,汇编语言是最贴近计算机的语言。而计算则指数学意义上的计算,越是抽象的计算,离计算机硬件越远。对应到编程语言,就是越低级的语言,越贴近计算机,抽象程度低,执行效率高,比如C语言;越高级的语言,越贴近计算,抽象程度高,执行效率低,比如Python语言。函数式编程就是一种抽象程度很高的编程范式。函数式编程的一个特点就是,允许把函数本身作为参数传入另一个函数,还允许返回一个函数!

Python对函数式编程提供部分支持。由于Python允许使用变量,因此,Python不是纯函数式编程语言。

什么是高阶函数

高阶函数英文叫Higher-order function。什么是高阶函数?我们来看代码变量可以指向函数
例如Python内置求绝对值函数abs()

>>> abs(-5)
5
>>> abs
<built-in function abs>

可以看见,只写abc只是函数本身,而abs(-5)才是函数调用,实际我们如果需要函数的结果,可以将函数赋值给变量
例如:

>>> i = abs(-5)
>>> i
5
>>> f = abs
>>> f
<built-in function abs>
>>> f(-5)
5

我们可以看到,我们将调用函数的结果,赋值给变量 i ,这样变量就可以打印出结果,如果将函数本身赋值给变量f ,那么变量也拥有这个函数的功能,这个变量将指向这个函数,使用变量 f ()来调用函数和直接调用abs()效果是一样的
函数名也可以是变量
函数是由def定义,函数名,括号,括号参数,冒号,函数体组成
那么函数名是什么尼,可以发现,函数名是指向函数的变量,例如abs()这个函数,可以将abs看成变量,它指向一个可以求绝对值的函数,如果把abs指向其他的对象,例如我们给abs赋值,那看看还会指向求绝对值的函数么

>>> abs = 5
>>> abs(-5)
Traceback (most recent call last):
File "<pyshell#23>", line 1, in <module>
abs(-5)
TypeError: 'int' object is not callable
>>> abs
5

TypeError: ‘int’ object is not callable 提示,类型错误,int类型是不可以被调用的,我们看到,abs这个变量被赋值5,然后使用abs(-5)调动函数,发现异常,此时abs变量指向的不是函数,而是一个int类型的 5 ,实际上,我们工作或是开发中写代码,是不能这么写的,由此可以看出函数名其实就是变量
注意:由于 abs 函数实际上是定义在 import builtins 模块中的,所以要让修改 abs 变量的指向在其它模块也生效可以使用

import builtins
builtins.abs = 10

我们想到,函数可以传参数,而函数名可以做变量,那我们函数里面的参数,也可以传入函数名,

# -*- coding: UTF-8 -*-
# 文件名:fun_a.py
def fun(i):
	return i * 2
def total(x, y, fun):
	return fun(x) + fun(y)
add_sum = total(1, 2, fun)
print(add_sum)

那我们可以看到,将函数作为参数,传给另一个参数就是高阶函数

常用的高阶函数

map函数

map(func, *iterables) --> map object
参数详解
func:代表传入参数为函数,这里的函数指定指向函数的函数名,
*iterables:代表参数指定的可迭代的,
返回值:返回处理好的数据
map()函数:是将传入的func函数作用于,可迭代的数据里面每个元素,并将处理好的新的结果返回
>>> def fun_a(x):
... return x * 10
...
>>> list_a = map(fun_a, [1, 2, 3, 4, 5])
>>> list(list_a)
[10, 20, 30, 40, 50]

map() 传入的第一个参数是 fun_a ,即函数对象本身。由于结果 list_a 是一个 Iterator , Iterator 是惰性序列,因此通过 list() 函数让它把整个序列都计算出来并返回一个list。
很多情况下,也可以使用for循环也可以解决问题,但实际上map作为高级函数,将运算抽象化,还可计算复杂的函数,例如将列表的元素int类型转换为str类型,只需要一行代码

>>> list(map(str, [1, 2, 3, 4, 5]))
['1', '2', '3', '4', '5']

reduce函数

我们来看一下map函数的参数与返回值,注意使用reduce函数时需要先导入reduce函数是在 functools模块里面的

from functools import reduce
reduce(function, sequence[, initial]) -> value
参数详解
function:一个有两个参数的函数
sequence:是一个序列,是一些数据的集合,或者是一组数据,可迭代对象
initial:可选,初始参数
返回值:返回函数计算的结果
reduce()函数,使用function函数(有两个参数)先对集合中的sequence第 12 个元素进行操作,如果存在
initial参数,则将会以sequence中的第一个元素和initial作为参数,用作调用,得到的结果再与sequence中的
下一个数据用 function 函数运算,最后得到一个结果。
from functools import reduce
list_a = [1, 2, 3, 4, 5]
def fun_b(x, y):
	return x + y
print(reduce(fun_b, list_a))
# 运算结果如下
15

依次按照顺序从列表list_a中提取两个元素作为参数,进入fun_b中进行运算,得到的结果,作为下次运算时的其中一个参数,再从列表中取出一个元素,再进行运算。最终得到的结果是总和的计算。

filter函数

Python内建的 filter() 函数用于过滤序列,和 map() 类似, filter() 也接收一个函数和一个序列但是不同的是 filter() 把传入的函数依次作用于每个元素,然后根据返回值是 True 还是 False 决定元素的保留与丢弃

filter(function, iterable)
function:判断函数。
iterable:序列,(可迭代对象)。
返回值:返回列表
filter函数,序列(可迭代对象)的每个元素作为参数传递给函数进行判断,然后返回 TrueFalse,最后将返
回 True 的元素放到新列表中
def not_odd(num):
	return n % 2 == 0
newlist = filter(not_odd, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10])
print(newlist)
# 运算结果
[2, 4, 6, 8, 10]

这里定义了一个函数not_odd,顾名思义,不是奇数的函数,这个函数,只有当参数为2的整数倍时返回True,这里filter函数的两个参数第一个是过滤方法,第二个是需要过滤的列表,将列表里面的元素依次带入函数中进行运算,得到的结果如果为True时,将此结果作为新的filter对象保留,等待函数里面的列表执行完成后,返回最终的值,这里的值为列表,也就是过滤掉了False的数据或元素。

简单例子

计算素数的其中一个方法是埃氏筛法。
给出要筛数值的范围n,找出以内的素数。先用2去筛,即把2留下,把2的倍数剔除掉;再用下一个质数,也就是3筛,把3留下,把3的倍数剔除掉;接下去用下一个质数5筛,把5留下,把5的倍数剔除掉;不断重复下去…用Python来实现这个算法,我们先写一个生成器构造一个从3开始的无限奇数序列,首先偶数列先排除

def oddnum():
	n = 1
	while True:
		n = n + 2
		yield n
# 写一个筛选的函数,这里使用了匿名函数,这个函数是返回一个判断,判断是否为可整除数
def undivisible(n):
	return lambda x: x % n > 0
# 使用filter来过滤,不断返回素数的生成迭代,
def primes():
	yield 2
	it = oddnum() # 初始序列
	while True:
		n = next(it) # 返回序列的第一个数
		yield n
		it = filter(undivisible(n), it) # 构造新序列
# 一个基本的判断素数方法就产生了,这里我们需要手动结束一下
# 打印100以内的素数
for n in primes():
	if n < 100:
		print(n)
	else:
		break

好的我们来解释一下代码,看到我们这里用到了几个知识点
生成器与迭代器、匿名函数、filter函数,其中
第一段代码生成了以3开始的奇数序列
第二段代码自定义过滤函数,包含匿名函数,判断值的取余是否能被整除
第三段代码用来返回素数,这里先返回一个2为素数,因为偶数都被排除了所以整除3为基础进行排除,将数据不断地迭代生成,留下对应的素数序列。那这里就对应的filter函数就是用来过滤的方法进行返回数据

匿名函数

匿名函数,顾名思义没有名字的函数我来看一下语法

lambda [list] : 表达式
#函数参数
[list]:表示参数列表,
注意:参数与表达式之间需要冒号来区分
表达式 :表达式方法非常多,表达形式也非常多
返回值 :为表达式的结果value
#例如,上方的代码这里:
lambda x: x % n > 0

x为需要传入的参数,而x % n > 0为表达式,之间需要用冒号进行引用,计算的表达式结果为返回值这里举例说明:如果设计一个求 2 个数之和的函数,使用普通函数的方式,定义如下:

def add(x, y):
return x+ y
print(add(3,4))

我们看到只有一行表达式,直接运算结果返回值,那这是时我们如果使用匿名函数一行代码即可完成

add = lambda x,y:x+y
print(add)

这里我们将直接写出我们的结果,我们可以看到对于比较单行返回的函数,使用 lambda 表达式可以省去定义函数的过程,让代码更加简洁,针对不需要多次复用的函数,使用 lambda 表达式可以在用完之后立即释放,提高程序执行的性能。而且还能配合其他的一些高阶函数配合使用

匿名函数与高阶函数的配合使用

#map函数
print(list(map(lambda x:x**2,[1,2,3,4])))
#reduce函数
from functools import reduce
print(reduce(lambda x,y:x+y,[1,2,3,4]))
#filter函数
print(list(filter(lambda x:x>2,[1,2,3,4])))
#max和min函数
print(max([1,3,4,5],key=lambda x:x))
print(max([1,2,3,4,5],key=lambda x:x))
#sorted排序函数
print(sorted([1,4,7,2,5],key=lambda x:x,reverse=True))
"""
[1, 4, 9, 16]
10
[3, 4]
5
5
[7, 5, 4, 2, 1]
"""

返回函数

高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回。我们在操作函数的时候,如果不需要立刻求和,而是在后面的代码中,根据需要再计算

def sum_fun_a(*args):
	a = 0
	for n in args:
		a = a + n
	return a

这是我不需要立即计算我的结果sum_fun方法,不返回求和的结果,而是返回求和的函数,例如下方

def sum_fun_b(*args):
	def sum_a():
		a = 0
		for n in args:
			a = a + n
		return a
	return sum_a

当我们调用 sum_fun_b() 时,返回的并不是求和结果,而是求和函数 sum_a , 当我们在调用sum_fun_b函数时将
他赋值给变量

>>> f1 = sum_fun_b(1, 2, 3, 4, 5)
# 此时f为一个对象实例化,并不会直接生成值,
>>> f1()
15
>>> f2 = sum_fun_b(1, 2, 3, 4, 5)
>>> f3 = sum_fun_b(1, 2, 3, 4, 5)
>>> print(f2, f3)
<function sum_fun_b.<locals>.sum_a at 0x0000029D496C8168>
<function sum_fun_b.<locals>.sum_a at 0x0000029D496D0EE8>
>>> print(id(f2), id(f3))
1920197821496 1920197820776

此时我们直接拿到的值就是15,那可以想一想,此时 f = sum_a,那这里存在一个疑问参数去哪里了?
而且我们看到创建的两个方法相互不影响的,地址及值是不相同的
在函数 sum_fun_b 中又定义了函数 sum_a ,并且,内部函数 sum_a 可以引用外部函数 sum_fun_b 的参数和局部变量,当 sum_fun_b 返回函数 sum_a 时,而对应的参数和变量都保存在返回的函数中,这里称为 闭包 。

闭包

内部函数对外部函数作⽤域⾥变量的引⽤(⾮全局变量),则称内部函数为闭包。这里闭包需要有三个条件:
1)必须有一个内嵌函数(函数里定义的函数)——这对应函数之间的嵌套
2)内嵌函数必须引用一个定义在闭合范围内(外部函数里)的变量——内部函数引用外部变量
3)外部函数必须返回内嵌函数——必须返回那个内部函数

# python交互环境编辑器
>>> def counter(start=0):
		count = [start]
		def incr():
			count[0] += 1
			return count[0]
		return incr
>>> c1 = counter(5)
>>> print(c1())
6
>>> print(c1())
7
>>> c2=counter(50)
>>> print(c2())
51
>>> print(c2())
52
>>>

当一个函数在本地作用域找不到变量申明时会向外层函数寻找,这在函数闭包中很常见
但是在本地作用域中使用的变量后,还想对此变量进行更改赋值就会报错

def test():
	count = 1
	def add():
		print(count)
		count += 1
	return add
a = test()
a()

报错信息
Traceback (most recent call last):

UnboundLocalError: local variable ‘count’ referenced before assignment
如果我在函数内加一行nonlocal count就可解决这个问题

# -*- coding: UTF-8 -*-
# 文件名 : nonlocal_a.py
def test():
	count = 1
	def add():
		nonlocal count
		print(count)
		count += 1
		return count
	return add
a = test()
a()
# 1
a()
# 2

nonlocal声明的变量不是局部变量,也不是全局变量,而是外部嵌套函数内的变量。
如果从另一个角度来看我们给此函数增加了记录函数状态的功能。当然,这也可以通过申明全局变量来实现增加函数状态的功能。当这样会出现以下问题:

  1. 每次调用函数时,都得在全局作用域申明变量。别人调用函数时还得查看函数内部代码。
  2. 当函数在多个地方被调用并且同时记录着很多状态时,会造成非常地混乱。

使用nonlocal的好处是,在为函数添加状态时不用额外地添加全局变量,因此可以大量地调用此函数并同时记录
着多个函数状态,每个函数都是独立、独特的。针对此项功能其实还个一个方法,就是使用类,通过定义
call 可实现在一个实例上直接像函数一样调用

# -*- coding: UTF-8 -*-
# 文件名:clos_c.py
def line_conf(a, b):
	def line(x):
		return a * x + b
	return line
line1 = line_conf(1, 1)
line2 = line_conf(4, 5)
print(line1(5))
print(line2(5))

从这段代码中,函数line与变量a,b构成闭包。在创建闭包的时候,我们通过line_conf的参数a,b说明了这两个变量
的取值,这样,我们就确定了函数的最终形式(y = x + 1和y = 4x + 5)。我们只需要变换参数a,b,就可以获得不同的直线表达函数。由此,我们可以看到,闭包也具有提⾼代码可复⽤性的作⽤。如果没有闭包,我们需要每次创建函数的时候同时说明a,b,x。这样,我们就需要更多的参数传递,也减少了代码的可移植性
1.闭包似优化了变量,原来需要类对象完成的⼯作,闭包也可以完成
2.由于闭包引⽤了外部函数的局部变量,则外部函数的局部变量没有及时释放,消耗内存
但是还没有结束,我们知道,函数内部函数,引用外部函数参数或值,进行内部函数运算执行,并不是完全返回一个函数,也有可能是一个在外部函数的值,我们还需要知道返回的函数不会立刻执行,而是直到调用了函数才会执行。

def fun_a():
	fun_list = []
	for i in range(1, 4):
		def fun_b():
			return i * i
		fun_list.append(fun_b)
	return fun_list
f1, f2, f3 = fun_a()
print(f1(), f2(), f3())

这里创建了一个fun_a函数,外部函数的参数fun_list定义了一个列表,在进行遍历,循环函数fun_b,引用外部变
量i 计算返回结果,加入列表,每次循环,都创建了一个新的函数,然后,把创建的3个函数都返回了
但是实际结果并不是我们想要的1,4,9,而是9,9,9,这是为什么尼?
这是因为,返回的函数引用了变量 i ,但不是立刻执行。等到3个函数都返回时,它们所引用的变量i已经变成了3,每一个独立的函数引用的对象是相同的变量,但是返回的值时候,3个函数都返回时,此时值已经完整了运算,并存储,当调用函数,产生值不会达成想要的,返回函数不要引用任何循环变量,或者将来会发生变化的变量,但是如果一定需要尼,如何修改这个函数尼?可以再创建一个函数,用该函数的参数绑定循环变量当前的值,无论该循环变量后续如何更改,已绑定到函数参数的值不变,那我们就可以完成下面的代码

# -*- coding: UTF-8 -*-
# 文件名 : closure_a.py
def fun_a():
def fun_c(i):
def fun_b():
return i * i
return fun_b
fun_list = []
for i in range(1, 4):
# f(i)立刻被执行,因此i的当前值被传入f()
fun_list.append(fun_c(i))
return fun_list
f1, f2, f3 = fun_a()
print(f1(), f2(), f3())
# 1 4 9

装饰器

装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。
它经常用于有以下场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。装饰器是解决这类问题的绝佳设计
装饰器的作用就是为已经存在的对象添加额外的功能

# -*- coding: UTF-8 -*-
# 文件名 : decor_b.py
# 装饰器
def decor_a(fun):
	def decor_b():
		print('I am decor_b')
		fun()
		print('I am fun')
	return decor_b
@decor_a
def decor_c():
	print('I am decor_c')
decor_c()
#outputs: I am decor_b
# I am decor_c
# I am fun

我们没有直接将decor_c函数作为参数传入decor_a中,只是将decor_a函数以@方式装饰在decor_c函数上。
也就是说,被装饰的函数,函数名作为参数,传入到装饰器函数上,不影响decor_c函数的功能,再次基础上可以
根据业务或者功能增加条件或者信息。
(注意:@在装饰器这里是作为Python语法里面的语法糖写法,用来做修饰。)
但是我们这里就存在一个问题这里引入魔术方法 name 这是属于 python 中的内置类属性,就是它会天生就存在与一个 python 程序中,代表对应程序名称,一般一段程序作为主线运行程序时其内置名称就是 main ,当自己作为模块被调用时就是自己的名字

print(decor_c.__name__)
#outputs: decor_b

这并不是我们想要的!输出应该是" decor_c "。这里的函数被decor_b替代了。它重写了我们函数的名字和注释文档,那怎么阻止变化尼,Python提供functools模块里面的wraps函数解决了问题

# -*- coding: UTF-8 -*-
# 文件名 : decor_c.py
from functools import wraps
# 装饰器
def decor_a(fun):
	@wraps(fun)
	def decor_b():
		print('I am decor_b')
		fun()
		print('I am fun()')
	return decor_b
@decor_a
def decor_c():
	print('I am decor_c')
print(decor_c.__name__)
#outputs: decor_c

我们在装饰器函数内,作用decor_c的decor_b函数上也增加了一个装饰器wraps还是带参数的。
这个装饰器的功能就是不改变使用装饰器原有函数的结构。
我们熟悉了操作,拿来熟悉一下具体的功能实现,我们可以写一个打印日志的功能

# -*- coding: UTF-8 -*-
# 文件名 : decor_d.py
from functools import wraps
def logs(fun):
	@wraps(fun)
	def with_logging(*args, **kwargs):
		print(fun.__name__ + " Debug")
		return fun(*args, **kwargs)
	return with_logging
@logs
def decor(x):
	return x + x
result = decor(4)
#outputs: decor Debug

带参装饰器

我们也看到装饰器wraps也是带参数的,那我们是不是也可以定义带参数的装饰器尼,
我们可以使用一个函数来包裹装饰器,调入这个参数。

# -*- coding: UTF-8 -*-
# 文件名 : log_b.py
from functools import wraps
def logs(logfile='out.log'):
	def logging_decorator(fun):
		@wraps(fun)
		def wrapped_function(*args, **kwargs):
			log_string = fun.__name__ + " Debug"
			print(log_string)
			# 打开logfile,并写入内容
			with open(logfile, 'a') as opened_file:
			# 现在将日志打到指定的logfile
				opened_file.write(log_string + '\n')
			return fun(*args, **kwargs)
		return wrapped_function
	return logging_decorator
@logs()
def decor_a():
	pass
decor_a()
# Output: decor_a Debug
# 现在一个叫做 out.log 的文件出现了,里面的内容就是上面的字符串
@logs(logfile='out1.log')
def decor_b():
	pass
decor_b()
# Output: decor_b Debug
# 现在一个叫做 func2.log 的文件出现了,里面的内容就是上面的字符串

这里我们将带参数的带入进去根据代码流程执行生成了两个文件并将文件打印进去
现在我们有了能用于正式环境的logs装饰器,但当我们的应用的某些部分还比较脆弱时,异常也许是需要更紧急关注的事情。比方说有时你只想打日志到一个文件。而有时你想把引起你注意的问题发送到一个email,同时也保留日志,留个记录。这是一个使用继承的场景,但目前为止我们只看到过用来构建装饰器的函数

# -*- coding: UTF-8 -*-
# 文件名 : log_c.py
from functools import wraps
class Logs(object):
	def __init__(self, logfile='out.log'):
		self.logfile = logfile
	def __call__(self, fun):
		@wraps(fun)
		def wrapped_function(*args, **kwargs):
			log_string = fun.__name__ + " was called"
			print(log_string)
		# 打开logfile并写入
			with open(self.logfile, 'a') as opened_file:
			# 现在将日志打到指定的文件
				opened_file.write(log_string + '\n')
			# 现在,发送一个通知
			self.notify()
			return fun(*args, **kwargs)
		return wrapped_function
def notify(self):
	# Logs只打日志,不做别的
	pass
@Logs()
def decor_a():
	pass

但还可以使用类的方式,这个实现有一个优势,在于比嵌套函数的方式更加整洁,而且包裹一个函数还是使用跟以前一样的语法现在,我们给 logit 创建子类,来添加 email 的功能,当然这个功能不在这里详细赘述

# -*- coding: UTF-8 -*-
# 文件名 : log_c.py
from mysit.text4.log_c import Logs
class EmailLogs(Logs):
'''
一个logit的实现版本,可以在函数调用时发送email给管理员
'''
	def __init__(self, email='admin@myproject.com', *args, **kwargs):
		self.email = email
		super(EmailLogs, self).__init__(*args, **kwargs)
	def notify(self):
		# 发送一封email到self.email
		# 这里就不做实现了
		pass

这里我们继续继承并重写notify方法,完成发送邮件的功能此时@EmailLogs 将会和 @Logs 产生同样的效果,但是在打日志的基础上,还会多发送一封邮件给管理员。

偏函数

Python的 functools 模块提供了很多有用的功能,其中一个就是偏函数(Partial function)。要注意,这里的
偏函数和数学意义上的偏函数不一样。
在介绍函数参数的时候,我们讲到,通过设定参数的默认值,可以降低函数调用的难度。而偏函数也可以做到这一点。
例如:int() 函数可以把字符串转换为整数,当仅传入字符串时, int() 函数默认按十进制转换

>>> int('123')
123

但 int() 函数还提供额外的 base 参数,默认值为 10 。如果传入 base 参数,就可以做进制的转换

>>> int('12345', base=8)
5349
>>> int('12345', 16)
74565

如果要转换大量的二进制字符串,每次都传入 int(x, base=2) 非常麻烦,于是,我们想到,可以定义一个
int2() 的函数,默认把 base=2 传进去

# 定一个转换义函数
>>> def int_1(num, base=2):
return int(num, base)
>>> int_1('1000000')
64
>>> int_1('1010101')
85

把一个函数的某些参数给固定住(也就是设置默认值),返回一个新的函数,调用这个新函数会更简单
继续优化,functools.partial 就是帮助我们创建一个偏函数的,不需要我们自己定义 int_1() ,可以直接使用下面的代码创建一个新的函数 int_1

# 导入
>>> import functools
# 偏函数处理
>>> int_2 = functools.partial(int, base=2)
>>> int_2('1000000')
64
>>> int_2('1010101')
85

理清了 functools.partial 的作用就是,把一个函数的某些参数给固定住(也就是设置默认值),返回一个新的
函数,调用这个新函数会更简单。
注意到上面的新的 int_2 函数,仅仅是把 base 参数重新设定默认值为 2 ,但也可以在函数调用时传入其他值
实际上固定了int()函数的关键字参数 base

int2('10010')

相当于是:

kw = { base: 2 }
int('10010', **kw)

当函数的参数个数太多,需要简化时,使用 functools.partial 可以创建一个新的函数,这个新函数可以固定住原函数的部分参数,从而在调用时更简单

内容来源于马士兵网课摘录

  • 25
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值