好得很程序员自学网

<tfoot draggable='sEl'></tfoot>

python — 函数基础知识(二)

目录

1 返回值 2 作用域 3 函数小高级 4 函数中高级

1 返回值

def func(arg):
    # ....
    return 9 # 返回值为9 默认:return None

val = func('adsfadsf')
# 1. 让用户输入一段字符串,计算字符串中有多少A字符的个数。有多少个就在文件a.txt中写多少个“李邵奇”。

def get_char_count(data):
    sum_counter = 0
    for i in data:
        if i == 'A':
            sum_counter += 1
            
    return sum_counter

def write_file(line):
    if len(line) == 0:
        return False  # 函数执行过程中,一旦遇到return,则停止函数的执行。
    with open('a.txt',mode='w',encoding='utf-8') as f:
        f.write(line)
    return True 

content = input('请输入:')
counter = get_char_count(content)
write_data = "李邵奇" * counter 
status = write_file(write_data)
if status:
    print('写入成功')
else:
    print('写入失败')

函数没有返回值时,默认返回None。

函数内部执行过程中遇到return就终止。

def func1():
    return "完成" # 函数每次执行到此,就返回;所以下面代码永远不执行。
    for i in range(10):
        print(i)
func1()               


def func2():
    for i in range(10):
        print(i)
        return "完成"
    print(666)
func2()        # 只打印0

return可以返回任意值

特殊情况:return返回多个值时,返回的是元组,与返回值是元组时是一样的

def func():
    return (1,2,3)

v = func()
print(v)

# 特殊:返回元组
def func():
    return 5,8,"alex"

v = func()
print(v)

return的作用:a. 返回值 b.终止函数的执行

练习题

# 1. 写函数,计算一个列表中有多少个数字,打印: 列表中有%s个数字。
#    提示:type('x') == int 判断是否是数字。

# 方式一:
def get_list_counter1(data_list):
    count = 0
    for item in data_list:
        if type(item) == int:
            count += 1
    msg = "列表中有%s个数字" %(count,)
    print(msg)
    
get_list_counter1([1,22,3,'alex',8])

# 方式二:
def get_list_counter2(data_list):
    count = 0
    for item in data_list:
        if type(item) == int:
            count += 1
    return count
    
v = get_list_counter1([1,22,3,'alex',8])
msg = "列表中有%s个数字" %(v,)
print(msg)
# 2. 写函数,计算一个列表中偶数索引位置的数据构造成另外一个列表,并返回。
# 方式一:
def get_data_list1(arg):
    v = arg[::2]
    return v

data = get_data_list1([11,22,33,44,55,66])

# 方式二:
def get_data_list2(arg):
    v = []
    for i in range(0,len(arg)):
        if i % 2 == 0:
            v.append(arg[i])
    return v

data = get_data_list2([11,22,33,44,55,66])
# 3. 读取文件,将文件的内容构造成指定格式的数据,并返回。
"""
a.log文件
    alex|123|18
    eric|uiuf|19
    ...
目标结构:
a.  ["alex|123|18","eric|uiuf|19"] 并返回。
b. [['alex','123','18'],['eric','uiuf','19']]
c. [
    {'name':'alex','pwd':'123','age':'18'},
    {'name':'eric','pwd':'uiuf','age':'19'},
]
"""

with open('a.log.txt',mode = 'r',encoding = 'utf-8') as f:
    data = f.read()
    print(data)
    def get_file(a):
        date1 = []
        for i in a.split('\n'):
            date1.append(i)
        return date1
    v1 = get_file(data)
    print(v1)

    def get_file1(b):
        date2 = []
        d = []
        for i1 in b.split('\n'):
            i1 = i1.split('|')
            d.append(i1)
            date2 += d
        return date2
    v2 = get_file1(data)
    print(v2)

    def get_file2(c):
        date3 = []
        e = {}
        for i2 in c.split('\n'):
            i2 = i2.split('|')
            e['name'] = i2[0]
            e['pwd'] = i2[1]
            e['age'] = i2[2]
            date3.append(e)
        return date3
    v3 = get_file2(data)
    print(v3)
数据类型中的方法到底有没有返回值:

无返回值

v = [1,2,3,4]
v.append(55)   # 无返回值时不用写return了

list : append / insert / remove / clear / extend / reverse

dict : update

set : add / discard / update

仅有返回值

v = 'ddff2dd554cvc'
result = '-'.join(v)
return result

v = {'k1':12,'k2':'ased'}
result = v.get('k2')
result = v.keys()

str : upper / lower / replace / isdecimal / strip / split / startswith / endswith / encode / format / join

list : find / index

dict : keys / values / items / get

set : intersection / union / difference / symmitric_difference

有返回值 + 修改数据

pop

v = [11,22,33,44]
result = v.pop(22)

常用需要记住的

索引、切片都有返回值

str : split 返回列表

strip     返回字符串

  replace    返回字符串

  join      返回字符串

list : append 无

insert        无

   remove     无

   pop          返回要删除的数据

   find          返回索引的位置

   index       返回索引的位置

dict : keys 获取所有的键

values     获取所有的值

     items       获取所有的键值对

     get     索引存在:返回值 ,不存在:返回None

2 作用域

在python文件中:

py文件:全局作用域

函数:局部作用域

a = 1
def s1():
    x1 = 666
    print(x1)
    print(a)
    print(b)

b = 2
print(a)
s1()
a = 88888
def s2():
    print(a,b)
    s1()

s2()

每个作用域中的数据只有作用域自己可以调用,如果作用域中调用的数据没有,可以调用全局作用域的

全局作用域只能调用全局的

在全局作用域中的函数可以互相调用(调用已经存在的),但不可以直接调用作用域中的作用域

总结:

1.一个函数就是一个作用域

2.作用域中查找数据规则:优先在自己的作用域找数据,自己没有就去“父级”-->“父级”-->直到全局,全局没有就报错。 (作用域的嵌套)

注意:父级作用域中的值到底是多少?

x = 10
def func():
    x = 9
    print(x)

func()

小练习

# 示例一
x = 10
def func():
    x = 9
    print(x)
    def x1():
        x = 999
        print(x)        
func()

# 示例二
x = 10
def func():
    x = 9
    print(x)
    def x1():
        x = 999
        print(x)
    x1()   
func()

# 示例三
x = 10
def func():
    x = 9
    print(x)
    def x1():
        x = 999
        print(x)
    print(x)
    x1()
func()

# 示例四
x = 10
def func():
    x = 8
    print(x)
    def x1():
        x = 999
        print(x)
    x1()
    print(x)
func()

# 示例五
x = 10
def func():
    x = 8
    print(x)
    def x1():
        print(x)
    x1()
    print(x)
func()

# 示例六
x = 10
def func():
    x = 8
    print(x)
    def x1():
        print(x)
    x = 9
    x1()
    x = 10
    print(x)
func()

# 示例七
x = 10
def func():
    x = 8
    print(x)
    def x1():
        print(x)    
    x1()
    x = 9
    x1()
    x = 10
    print(x)

func()

3.子作用域中只能只能找到父级中的值,默认无法重新为父级的变量进行赋值。

不能进行赋值,只能对可变类型进行内部修改

# #####################
name = 'oldboy'
def func():
    name = 'alex' # 在自己作用域再创建一个这样的值。
    print(name)
func()
print(name)


# #####################
name = [1,2,43]
def func():
    name.append(999)
    print(name)
func()
print(name)

如果非要对全局的变量进行赋值需要加global(强制赋值)

#示例一
name = "老男孩“
def func():
    global name
    name = 'alex'
func()
print name

# 示例二
name = ["老男孩",'alex']
def func():
    global name
    name = '我'
func()
print(name)

# 示例三
name = "老男孩"
def func():
    name = 'alex'
    def inner():
        global name
        name = 999
    inner()
    print(name)
func()
print(name)

对父级的变量赋值用nonlocal,先找到父级的变量再进行赋值 (强制赋值)

name = "老男孩"
def func():
    name = 'alex'
    def inner():
        nonlocal name # 找到上一级的name
        name = 999
    inner()
    print(name)
func()
print(name)

补充:全局变量必须全部要大写

USER_LIST = [11,22,3]

def func():
    name = 'asdf'
    USER_LIST.append(12)
    USER_LIST.append(name)

func()
print(USER_LIST)

3 函数小高级

函数名可以当作变量来使用

def func():
    print(123)

v1 = func  # func代表函数的地址

func()
v1()   # v1、func的函数地址相同,执行调用的函数也相同
def func():
    print(123)

func_list = [func, func, func]
# func_list[0]()  a
# func_list[1]()  b
# func_list[2]()  c
for item in func_list:   # a/b/c的简化形式
    v = item()
    print(v)
def func():
    print(123)

def bar():
    print(666)

info = {'k1': func, 'k2': bar}

info['k1']()           # 函数也可以作为字典的值(也可以做为键,但是没有意义)
info['k2']()

注意:函数是不可变的,可以做集合的元素,也可以作为字典的键 (但是做键没有太大意义)。

集合中可以放多个重复的函数,但只执行一次。(因为集合的特性:不可重复的)

混淆你

def func():
    return 123

func_list1 = [func,func,func]
func_list2 = [func(),func(),func()]

print(func_list1)     # 打印的是func的函数地址
print(func_list2)     # 打印的是func执行完返回的值

info = {
    'k1':func,     # 函数的地址
    'k2':func(),   # 函数执行完返回的值
}

print(info)

函数也可以当作参数来进行传递

def func(arg):
    print(arg)

func(1)
func([1,2,3,4])

def show():
    return 999
func(show)      # 执行函数func,参数为show,show没有+(),表示show没有执行只是代表该函数的地址。
def func(arg):
    v1 = arg()
    print(v1)

def show():
    print(666)

func(show)
def func(arg):
    v1 = arg()
    print(v1)

def show():
    print(666)

result = func(show)
print(result)

多个函数的调用

def func():
    print('花费查询')

def bar():
    print('语音沟通')

def base():
    print('xxx')

def show():
    print('xxx')

def test():
    print('xxx')

info = {
    'f1': func,
    'f2': bar,
    'f3':base,
    'f4':show,
    'f5':test
}
choice = input('请选择要选择功能:')
function_name = info.get(choice)
if function_name:
    function_name()
else:
    print('输入错误')

总结:函数当作一个变量:参数传值 / 当作元素嵌套到字典、列表中

4 函数中高级

4.1 函数可以做返回值
# 示例一
def func():
    print(123)

def bar():
    return func

v = bar()
v()

# 示例二
name = 'oldboy'
def func():
    print(name)
    
def bar():
    return func

v = bar()
v()

# 示例三
def bar():
    def inner():
        print(123)
    return inner
v = bar()
v()

# 示例四
name = 'oldboy'
def bar():
    name = 'alex'
    def inner():
        print(name)
    return inner
v = bar()
v()

# 示例五
name = 'oldboy'
def bar(name):
    def inner():
        print(name)
    return inner
v1 = bar('alex') # { name=alex, inner }  # 闭包,为函数创建一块区域(内部变量供自己使用,存储的代码),为他以后执行提供数据。
v2 = bar('eric') # { name=eric, inner }
v1()
v2()

# 示例六
name = 'alex'
def base():
    print(name)

def func():
    name = 'eric'
    base()

func() # {name=eric, }
    
# 示例七
name = 'alex'
def func():
    name = 'eric'
    def base():
        print(name)
    base()
func()

# 示例八
name = 'alex'
def func():
    name = 'eric'
    def base():
        print(name)
    return base 
base = func()
base()

注意:函数在何时被谁创建?

函数是由谁创建的,执行函数就从哪里开始找
# 练习题一
info = []

def func():
    print(item)
    
for item in range(10):
    info.append(func)

info[0]()

# 练习题二
info = []

def func(i):
    def inner():
        print(i)
    return inner

for item in range(10):
    info.append(func(item))

info[0]()
info[1]()
info[4]()
4.2 闭包
def func(name):
    def inner():
        print(name)
    return inner 

v1 = func('alex')
v1()
v2 = func('eric')
v2()

返回值——分析函数执行的内存。(闭包是内存其中的一种)

# 并不是闭包
def func(name)
    def inner():
        return 123
    return inner
# 是闭包需要满足两个条件:1.封装值   2.内层函数需要使用
def func(name)
    def inner():
        print(name)
        return 123
    return inner
4.3 高阶函数

把函数当作参数传递

把函数当作返回值

注意:对函数进行赋值

4.4 小总结 1.函数执行流程的分析(函数到底是由谁创建的?) 2.闭包概念:为函数创建一块区域并为其维护自己的数据以后执行时方便调用。(应用场景:装饰器 / SQLAlchemy源码)

查看更多关于python — 函数基础知识(二)的详细内容...

  阅读:24次