昨天说了函数的一些最基本的定义,今天我们继续研究函数。今天主要研究的是函数的命名空间、作用域、函数名的本质、闭包等等

预习:

1、写函数,用户传入修改的文件名,与要修改的内容,执行函数,完成整个文件的批量修改操作
2、写函数,检查用户传入的对象(字符串、列表、元组)的每一个元素是否含有空内容。
3、写函数,检查传入字典的每一个value的长度,如果大于2,那么仅保留前两个长度的内容,并将新内容返回给调用者。
dic = {"k1": "v1v1", "k2": [11,22,33,44]}
PS:字典中的value只能是字符串或列表

预习题

本篇导航:

一、命名空间

命名空间的本质:存放名字与值的绑定关系

命名空间

局部命名空间

全局命名空间

内置命名空间

*内置命名空间中存放了python解释器为我们提供的名字:input,print,str,list,tuple...它们都是我们熟悉的,拿过来就可以用的方法。

三种命名空间之间的加载与取值顺序:

加载顺序:内置命名空间(程序运行前加载)->全局命名空间(程序运行中:从上到下加载)->局部命名空间(程序运行中:调用时才加载)

取值:

在局部调用:局部命名空间->全局命名空间->内置命名空间

x = 1
def f(x):
print(x) print(10)

在局部使用变量取值

在全局调用:全局命名空间->内置命名空间

x = 1
def f(x):
print(x) f(10)
print(x)

在全局引用变量x

print(max)

在全局引用内置max


二、作用域

为什么要有作用域的概念:

为了函数内的变量不会影响到全局

作用域就是作用范围,按照生效范围可以分为全局作用域和局部作用域。

全局作用域:包含内置名称空间、全局名称空间,在整个文件的任意位置都能被引用、全局有效

局部作用域:局部名称空间,只能在局部范围内生效

在子程序中定义的变量称为局部变量,在程序的一开始定义的变量称为全局变量。

全局变量作用域是整个程序,局部变量作用域是定义该变量的子程序。

当全局变量与局部变量同名时:在定义局部变量的子程序内,局部变量起作用;在其它地方全局变量起作用。

作用域:

小范围的可以用大范围的但是大范围的不能用小范围的

范围从大到小(图)

在小范围内,如果要用一个变量,是当前这个小范围有的,就用自己的如果在小范围内没有,就用上一级的,上一级没有就用上上一级的,以此类推。如果都没有,报错

globals和locals方法

a = 20
b = 50
def func():
x = 1
y = 2
print(globals()) #全局作用域中的名字
print(locals()) #局部作用域中的名字 func()
print(globals()) #全局作用域中的名字
print(locals()) #全局的局部还是全局

globals和locals

global关键字

a = 10
def func():
a = 20
def func1():
global a #改变全局变量
a = 20 print(a)
func()
print(a)
func1()
print(a)

global关键字


三、函数的嵌套和作用域链

函数的嵌套调用

def max2(x,y):
m = x if x>y else y
return m def max4(a,b,c,d):
res1 = max2(a,b)
res2 = max2(res1,c)
res3 = max2(res2,d)
return res3 max4(23,-7,31,11)

函数的嵌套调用

函数的嵌套定义

def f1():
print("in f1")
def f2():
print("in f2") f2()
f1()

函数的嵌套定义(一)

def f1():
def f2():
def f3():
print("in f3")
print("in f2")
f3()
print("in f1")
f2() f1()

函数的嵌套定义(二)

函数的作用域链

def f1():
a = 1
def f2():
print(a)
f2() f1()

作用域链(一)

def f1():
a = 1
def f2():
def f3():
print(a)
f3()
f2() f1()

作用域链(二)

def f1():
a = 1
def f2():
a = 2
f2()
print('a in f1 : ',a) f1()

作用域链(三)

nolocal关键字

1、外部必须有这个变量

2、在内部函数声明nonlocal变量之前不能再出现同名变量

3、内部修改这个变量如果想在外部有这个变量的第一层函数中生效

def f():
a = 3
def f1():
a = 1
def f2():
nonlocal a
a = 2
f2()
print('a in f1 : ', a)
f1()
print('a in f : ',a)
f()

nolocal关键字


四、函数名的本质

函数名本质上就是函数的内存地址

1、可以被引用

def func():
print('in func') f = func
print(f)

函数被引用

2、可以被当作容器类型的元素

def f1():
print('f1') def f2():
print('f2') def f3():
print('f3') l = [f1,f2,f3]
d = {'f1':f1,'f2':f2,'f3':f3}
#调用
l[0]()
d['f2']()

函数被当作容易类型的元素

3、可以当作函数的参数和返回值

def func():
print('func')
def func2(f):
f()
func2(func)

函数做参数

def func():
def func2():
print('hello')
return func2 f2 = func()
f2()
f = func

函数做返回值

第一类对象(first-class object)指

1、可在运行期创建

2、可用作函数参数或返回值

3、可存入变量的实体。

不明白就记住一句话,就当普通变量用


五、闭包

闭包函数:内部函数,包含了对外部作用域中变量的引用,内部函数包含对外部作用域而非全剧作用域名字的引用,该内部函数称为闭包函数
#函数内部定义的函数称为内部函数

闭包

1、闭 内部的函数

2、包 包含了对外部函数作用域中变量的引用

def func():
name = 'eva'
def inner():
print(name)
def func():
name = 'eva'
def inner():
print(name)
return inner f = func()
f()

闭包函数常用方法

判断闭包函数的方法__closure__

#输出的__closure__有cell元素 :是闭包函数
def func():
name = 'eva'
def inner():
print(name)
print(inner.__closure__)
return inner f = func()
f() #输出的__closure__为None :不是闭包函数
name = 'egon'
def func2():
def inner():
print(name)
print(inner.__closure__)
return inner f2 = func2()
f2()

闭包函数的判断

def wrapper():
money = 1000
def func():
name = 'eva'
def inner():
print(name,money)
return inner
return func f = wrapper()
i = f()
i()

闭包嵌套

from urllib.request import urlopen

def index():
url = "http://www.cnblogs.com/liluning/"
def get():
return urlopen(url).read()
return get cnblogs = index()
content = cnblogs()
print(content)

闭包函数获取网络应用


思维导图:

预习答案

import os
def file_l(demo,i,j) :
'''
用户传入修改的文件名,与要修改的内容,执行函数,完成整个文件的批量修改操作
:param demo:传入的文件
:param i:需要修改的字符串
:param j:要改变后的字符串
:return:返回源文件和改后文件的句柄
'''
with open(demo, 'r', encoding='utf-8') as read_f, \
open("demo1.py", 'w', encoding='utf-8') as write_f :
for line in read_f :
write_f.write(line.replace(i, j))
return read_f,write_f
file_l1,file_l2 = file_l("demo.py","def","hahaha")
os.remove(file_l1.name)
os.rename(file_l2.name, file_l1.name) def space_l(str) :
'''
检查用户传入的对象(字符串、列表、元组)的每一个元素是否含有空内容。
:param str: 传入数据
:return: 布尔值
'''
for i in str :
if i.isspace() :
return False
return True msg = "hello world!"
print(space_l(msg)) def length(dic) :
'''
检查传入字典的每一个value的长度,如果大于2,那么仅保留前两个长度的内容,
并将新内容返回给调用者。
:param dic:
:return:
'''
for i in dic :
if len(dic[i]) > 2 :
dic[i] = dic[i][0:2]
return dic
dic = {"k1": "v1v1", "k2": [11,22,33,44]}
length(dic)
print(dic)

预习答案


小知识点:

#三元运算
# a = 20
# b = 10
# if a > b:
# c = 5
# else:
# c = 10 #if条件成立的结果 if 条件 else else条件成立的结果
# a = 20
# b = 10
# c = 5 if a>b else 10
# print(c)

三元运算

课外娱乐:

>>> import this
The Zen of Python, by Tim Peters Beautiful is better than ugly.
Explicit is better than implicit.
Simple is better than complex.
Complex is better than complicated.
Flat is better than nested.
Sparse is better than dense.
Readability counts.
Special cases aren't special enough to break the rules.
Although practicality beats purity.
Errors should never pass silently.
Unless explicitly silenced.
In the face of ambiguity, refuse the temptation to guess.
There should be one-- and preferably only one --obvious way to do it.
Although that way may not be obvious at first unless you're Dutch.
Now is better than never.
Although never is often better than *right* now.
If the implementation is hard to explain, it's a bad idea.
If the implementation is easy to explain, it may be a good idea.
Namespaces are one honking great idea -- let's do more of those!

python之禅

在python之禅中提到过:命名空间是一种绝妙的理念,让我们尽情的使用发挥吧!

python函数(2):函数进阶的更多相关文章

  1. python基础 (初识函数&函数进阶)

    函数基础部分 .什么是函数? 函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段. 函数能提高应用的模块性,和代码的重复利用率. 2.定义函数 定义:def 关键词开头,空格之后接函数名 ...

  2. Python基础之函数参数与返回值进阶

    参数作用:如果外界希望在函数内部处理数据,就可以将数据作为参数传入函数内部: 返回值作用:如果希望一个函数函数执行完成后,向外界报告函数的执行结果,就可以使用函数的返回值. 函数的返回值 进阶 利用元 ...

  3. python语法基础-函数-进阶-长期维护

    ###############    函数的命名空间和作用域    ############## """ # 函数进阶 命名空间和作用域 命名空间 全局命名空间——我们自 ...

  4. 周末班:Python基础之函数进阶

    迭代器和生成器 迭代和可迭代 什么是迭代(iteration)? 如果给定一个list或tuple,我们要想访问其中的某个元素,我们可以通过下标来,如果我们想要访问所有的元素,那我们可以用for循环来 ...

  5. Python全栈开发之路 【第五篇】:Python基础之函数进阶(装饰器、生成器&迭代器)

    本节内容 一.名称空间 又名name space,就是存放名字的地方.举例说明,若变量x=1,1存放于内存中,那名字x存放在哪里呢?名称空间正是存放名字x与1绑定关系的地方. 名称空间共3种,分别如下 ...

  6. Python进阶:函数式编程(高阶函数,map,reduce,filter,sorted,返回函数,匿名函数,偏函数)...啊啊啊

    函数式编程 函数是Python内建支持的一种封装,我们通过把大段代码拆成函数,通过一层一层的函数调用,就可以把复杂任务分解成简单的任务,这种分解可以称之为面向过程的程序设计.函数就是面向过程的程序设计 ...

  7. python 内置函数(二) 进阶函数 递归内容及二分法查找 知识点

    1,lambda:  匿名函数 2.sorgted()  排序函数 3,filter()   过滤函数 筛选 4,map()  映射函数 5.递归 6.二分法 一. 匿名函数: lambda lamb ...

  8. (转)Python进阶:函数式编程(高阶函数,map,reduce,filter,sorted,返回函数,匿名函数,偏函数)

    原文:https://www.cnblogs.com/chenwolong/p/reduce.html 函数式编程 函数是Python内建支持的一种封装,我们通过把大段代码拆成函数,通过一层一层的函数 ...

  9. Python进阶---python strip() split()函数实战(转)

    先看一个例子: >>> ipaddr = 10.122.19.10 File "", line 1 ipaddr = 10.122.19.10 ^ SyntaxE ...

  10. python之路——函数进阶

    阅读目录   楔子 命名空间和作用域 函数嵌套及作用域链 函数名的本质 闭包 本章小结 楔子 假如有一个函数,实现返回两个数中的较大值: def my_max(x,y): m = x if x> ...

随机推荐

  1. 手机wap网站建设的方法和技巧

    随着互联网技术的不断进步,越来越多的运营商对于手机wap网站的建设有了更多的投入,手机wap网站的建设和开发要根据网站的特点和经营范围来进行设计和建设,这样才可以提升手机wap网站建设的效果.现在智能 ...

  2. JS如何实现真正的对象常量

    前言 众所周知ES6新增的const关键字可以用来声明常量,但是它只对基本数据类型生效(Number.String.Boolean等),那如果我们想声明一个常量对象呢?该如何实现,Object内置对象 ...

  3. Java软件系统功能设计实战训练视频教程

    Java软件系统功能设计实战训练视频教程 第01节课:整体课程介绍和杂项介绍第02节课:软件功能设计常见理念和方法第03节课:关于软件设计的一些思考第04节课:第一周作业的业务和相应模式:综合应用简单 ...

  4. php使用openssl加密数据

    <?php /** * Created by PhpStorm. * User: hanks * Date: 6/2/2017 * Time: 6:03 PM */ //使用php函数生成密钥对 ...

  5. PHP Startup: Unable to load dynamic library '/usr/lib64/php/modules/event.so'

    这是因为启动apche 的时候重复加载了php的动态库模块了, 将:/etc/php.d/ 目录下的event.ini中的内容注释掉或者将文件删除即可

  6. c#中的委托和c++中的bind/function对比

    在c++中,如果要实现这样一个功能,比如定时器,在指定的时间执行指定的函数,接口可以采用如下的设计 uint64_t addtimer(uint64_t t, std::function<voi ...

  7. JQuery——banner旋转木马效果

    博主在浏览网页时无意间发现了一种banner图的轮播方式--像旋转木马一样的轮播方式,博主感觉非常新颖独特,经过查阅资料,观看某课网教程总算搞了出来的,其原理主要利用了JQuery代码实现,好了不多说 ...

  8. for循环语句

    for循环格式:  for(表达式1:表达式2:表达式3)        {             循环语句        } 1.首先计算表达式1的值. 2.再计算表达式2 的值,若值为真(非0) ...

  9. js动态增加秒数(自动,手动)

    //获取当前的日期及时间Date var myDate = new Date();      myDate.getYear();        //获取当前年份(2位)      myDate.get ...

  10. mybatis学习笔记(四)-- 为实体类定义别名两种方法(基于xml映射)

    下面示例在mybatis学习笔记(二)-- 使用mybatisUtil工具类体验基于xml和注解实现 Demo的基础上进行优化 以新增一个用户为例子,原UserMapper.xml配置如下: < ...