函数和函数式编程

  • 函数定义;

  • 函数调用;

  • 过程定义;

  • 过程调用;

  • 面向过程的编程方法;

"""
面向对象-----类------class
面向过程-----过程---- def
函数式编程----函数----def 函数定义:将一组语句的集合通过一个名字(即函数名)封装起来,要想执行这个函数,只需调用其函数名即可 函数和过程都是可以调用的,过程就是一个没有返回值的函数而已 面向过程的编程方法:一段一段的逻辑或者一段一段的功能包含在def定义的过程中,使用时直接调用
函数式编程:?
""" # 语法定义
# 定义一个函数,有返回值
def func1(): # 函数名,括号里可以写入参数
"""testing"""
print('in the func1') # 函数体,表示复杂逻辑的代码
return 0 # 返回值 # 定义一个过程,没有返回值
def func2():
"""testing2"""
print("in the func2") # 表示复杂逻辑的代码 # 调用
x = func1()
y = func2()
print('from func1 return is %s' %x)
print('from func2 return is %s' %y) # 也被当成了一个隐式的函数

示例:使用函数减少重复代码

为什么要使用函数:

1.减少重复代码

"""
为什么要使用函数:减少重复代码;使程序变得可扩展;使程序变得易维护
日志文件处理(框架搭建必须有的)
"""
import time def logger(): with open('a.txt', 'a+') as f:
f.write('end action\n') # 类似写日志文件 # 模拟一个功能,并将日志文件写入日志文件
def test1():
"""文档描述"""
print('in the test1') # 日志追加
logger() # 写入日志 # 模拟另外一个功能
def test2():
print('in the test2') # 日志描述
logger() # 写入日志 # 模拟第三个功能
def test3():
print('in the test3')
logger() # 写入日志 # 调用各个功能模块
test1()
test2()
test3()

示例:使用函数使程序变得可扩展

2.使程序变得可扩展

"""
为什么要使用函数:减少重复代码;使程序变得可扩展;使程序变得易维护
日志文件处理(框架搭建必须有的)
"""
import time def logger():
time_format = '%Y-%m-%d %X' # 年月日,时分秒
time_current = time.strftime(time_format)
with open('a.txt', 'a+') as f:
f.write('%s end action\n' % time_current) # 类似写日志文件 # 模拟一个功能,并将日志文件写入日志文件
def test1():
"""文档描述"""
print('in the test1') # 日志追加
logger() # 写入日志 # 模拟另外一个功能
def test2():
print('in the test2') # 日志描述
logger() # 写入日志 # 模拟第三个功能
def test3():
print('in the test3')
logger() # 写入日志 # 调用各个功能模块
test1()
test2()
test3()

示例:函数返回值

编写函数为什么要有返回值,作用:想要这个函数的执行结果。

函数体中可以是一行代码,也可以是一万行代码。这个函数执行的结果是什么,我后面的程序需要这个程序的返回结果。

"""
函数返回值
函数返回值的作用:后面的程序需要这个函数的返回结果
""" def test1():
"""未定义返回值"""
print('in the test1') # 函数体 def test2():
'''定义返回值'''
print('in the test2') # 函数体
return 0 # 定义返回值,结束函数的效果 def test3():
'''定义返回值有数字,字符,元组,字典'''
print('in the test3') # 函数体
return 1, 'hello', ['zhangsan', 'lisi'], {'name': 'emy'} x = test1() # 函数体的返回值
y = test2() # 函数体的返回值
z = test3() # 函数体的返回值
print(x)
print(y)
print(z)

示例:带参数的函数

学会以下传参方法:位置参数;默认参数;参数组传参;字典参数传参

"""
带参数的函数:关键字参数和位置参数
关键参数不能写在位置参数前面
""" def test(x, y):
print(x)
print(y) test(5, 8) # 位置参数与形参一一对应
test(x=1, y=6) # 关键字参数与形参顺序无关
test(y=9, x=3) # 关键字参数与形参顺序无关 # 默认参数
def test1(x, y=2):
print(x)
print(y) test1(1)
test1(1, y=3) # 默认参数特点:调用函数时默认参数可有可无,不是必传参数
# 默认参数应用场景:占位或默认初始设置路径或默认端口3306

示例:参数带参数组的函数定义方式

# 参数组*args:实参不固定的情况下,怎么定义实参,在这种情况下,采用参数组,接收N个位置参数,不是关键字方式,转换成元组的方式;
# 应用场景:给函数传不固定的参数,定义参数时,根据业务发展,可能有预想参数个数不一样的的情况下
# 规范:*args ,如果星号后面采用其他名字,程序不会报错,但是最后用规范的带args名,这样便于阅读
def test2(*args):
print(args) test2(1, 2, 3, 4, 5)
test2(*[1, 2, 3, 4, 5, 6]) # args=tuple(1,2,3,4,5,6) def test3(x, *args):
print(x)
print(args) test3(1, 2, 3, 4, 5, 6, 7)

示例:参数带字典的函数定义方式

# **kwargs接收N个关键字参数,转换成字典的形式

def test2(**kwargs):
print(kwargs)
print(kwargs['name'])
print(kwargs['age'])
print(kwargs['sex']) test2(name='zhangsan', age=8, sex='male')
test2(**{'name': 'zhangsan', 'age': '', 'sex': 'male'})

示例:参数可扩展的函数定义方式

# 参数可扩展的函数

# 与位置参数一起使用
def test3(name, **kwargs):
print(name)
print(kwargs) test3('zhangsan') # 这样转参数,会接收一个name和一个空字典----->>正确
test3('lisi', age=8, sex='male') # 与位置参数结合,以关键字方式传参------>>正确 # 与默认参数一起使用
def test4(name, age=8, **kwargs): # 默认参数不能写在参数最后,参数组置后
print(name)
print(age)
print(kwargs) test4('wangwu', sex='male', hobby='study', age=3) # 此处的age传参可以写在最后,也可以写在对应的位置 # 所有类型的参数结合在一起(位置参数,默认参数,带数组的参数,带字典的参数)
def test5(name, age=18, *args, **kwargs):
print(name)
print(age)
print(args)
print(kwargs) test5('amy', age=28, sex='male', hoddy='study') # 在这个没有位置参数,所以*args在运行的时候接收为空元组;后面两个参数为两个关键字参数。所以传给**kwargs

函数和函数式编程:不是同一种方式,了解就行,不用深究

函数式编程、面向过程、面向对象编程是三种编程范式。函数式编程是一种抽象很高的一种编程范式,输入是什么输出就是什么,函数是输入和输出可以不同。

如有疑问,可加入QQ学习群:484601288

​局部变量和与全局变量

​是否可在一个函数中引用另外一个函数呢?

示例:局部变量

# 局部变量:函数里面的变量即为局部变量,只在函数里有效

def change_name(name):
print("befor change", name)
name = 'San Zhang' # 这个函数就是这个变量的作用域
print("after change", name) name = 'zhangsan'
change_name(name)
print(name)

示例:全局变量

# 全局变量:在整个程序中都生效的变量,在整个代码的顶层定义的变量就是全局变量

company = 'ali'

def change_name(name):
company = 'tengxu'
print("befor change", name, company)
name = 'zhangsan' # 这个函数就是这个变量的作用域
print("after change", name) print(company) name = 'lisi'
change_name(name)
print(name, company)

示例:局部变量改为全局变量1

在函数里面,默认情况下“字符串”和“常量”,这两类局部变量不能更改全局变量,如果想在函数中将局部变量改为全局变量的方法:改之前申明global,不建议使用这种方法。

# 在函数里面,默认情况下局部变量不能更改全局变量,如果想在函数中将局部变量改为全局变量的方法:改之前申明global,不建议使用这种方法

company = 'ali'  # 全局变量

def change_name(name):
global company
company = 'tengxu' # 局部变量
print("befor change", name, company)
name = 'zhangsan' # 这个函数就是这个变量的作用域
print("after change", name) print('company: ', company) # 执行查看函数调用前打印的情况
name = 'lisi'
change_name(name)
print(name)
print('company: ', company) # 执行查看函数调用后打印的情况

示例:局部变量改为全局变量2

未像<局部变量改为全局变量1>那样申明global,也将局部变量更改为全局变量了。所以默认不能更改只针对“字符串和整数”外。复杂的数据类型:列表,类,字典,集合都是可以将局部变量改为全局变量。

# 局部变量改全局变量

company = 'ali'  # 全局变量
names = ['zhangsan', 'sili', 'wangwu'] def change_name():
names[0] = '张三'
print("inside_func: ", names) # 运行查看执行结果 change_name()
print('ouside_func:', names) # 运行查看执行结果

示例:作死的全局变量定义

或许你在其他地方也会看见这种方式,忘掉这种方式,不能使用这种方法,这种在逻辑复杂的代码中不利于调试,尽管是自己写的代码也会如此。

# 不建议使用这种方法
def chang_name():
global name
name = 'zhangsan' chang_name()
print(name)

示例:使用递归方式实现,传一个值,每次除以2直到不能除为止

递归特性:1、必须有一个明确的结束条件;2、每次进入更深一层递归时,问题规模相比上次递归都应有减少;3、递归效率不高,可能会导致内存资源耗尽

做递归最好的运行方式,采用断点方式调试。

简单的一个递归方式实现传一个值,每次除以2直到不能除为止:

# 传一个值,每次除以2直到不能除为止

def calc(n):
print(n)
if int(n/2)>0:
return calc(int(n/2))
print('--->>', n) calc(10)

所有示例程序均在Ubuntu16.04系统下的python3.5下成功执行。

高阶函数:变量可以指向函数,函数的参数能接收变量,那么一个函数就可以接收另一个函数作为参数

# 把一个函数当成一个参数传给另一个函数,这个函数就是高阶函数

# 这是一般普通的函数
def add(a, b):
return a + b # 高阶函数
def add(a, b, f):
return f(a) + f(b) res = add(3, -6, abs)
print(res)

如有疑问,可加入QQ学习群讨论:484601288

Python学习示例源码的更多相关文章

  1. Tyrion中文文档(含示例源码)

    Tyrion是一个基于Python实现的支持多个WEB框架的Form表单验证组件,其完美的支持Tornado.Django.Flask.Bottle Web框架.Tyrion主要有两大重要动能: 表单 ...

  2. Tyrion 中文文档(含示例源码)

    原文出处: Mr.Seven   Tyrion是一个基于Python实现的支持多个WEB框架的Form表单验证组件,其完美的支持Tornado.Django.Flask.Bottle Web框架.Ty ...

  3. 一起学习vue源码 - Object的变化侦测

    作者:小土豆biubiubiu 博客园:www.cnblogs.com/HouJiao/ 掘金:https://juejin.im/user/58c61b4361ff4b005d9e894d 简书:h ...

  4. 手牵手,从零学习Vue源码 系列一(前言-目录篇)

    系列文章: 手牵手,从零学习Vue源码 系列一(前言-目录篇) 手牵手,从零学习Vue源码 系列二(变化侦测篇) 手牵手,从零学习Vue源码 系列三(虚拟DOM篇) 陆续更新中... 预计八月中旬更新 ...

  5. 【菜鸟学习jquery源码】数据缓存与data()

    前言 最近比较烦,深圳的工作还没着落,论文不想弄,烦.....今天看了下jquery的数据缓存的代码,参考着Aaron的源码分析,自己有点理解了,和大家分享下.以后也打算把自己的jquery的学习心得 ...

  6. WEB前端开发学习:源码canvas 雪

    WEB前端开发学习:源码canvas 雪 双旦节要到了,程序员们为了响应气氛,特别用代码制作了动态雪花,WEB前端开发学习的初学者们一起跟着案例做一遍吧! <!DOCTYPE html> ...

  7. 安卓图表引擎AChartEngine(三) - 示例源码折线图、饼图和柱状图

    折线图: package org.achartengine.chartdemo.demo.chart; import java.util.ArrayList; import java.util.Lis ...

  8. python的paramiko源码修改了一下,写了个操作命令的日志审计 bug修改

    python的paramiko源码修改了一下,写了个操作命令的日志审计,但是记录的日志中也将backspace删除键记录成^H这个了,于是改了一下代码,用字符串的特性. 字符串具有列表的特性 > ...

  9. Python解析器源码加密系列之(二):一次使用标准c的FILE*访问内存块的尝试

    摘要:由于近期打算修改Python解释器以实现pyc文件的加密/解密,出于保密的要求,解密之后的数据只能放在内存中,不能写入到文件中.但是后续的解析pyc文件的代码又只能接受FILE*作为入参,所以就 ...

随机推荐

  1. 编程之美 set 12 快速找出故障机器

    题目 1. 所有的 ID 都出现 2 次, 只有一个例外, 找到那个例外的 ID 2. 所有的 ID 都出现两次, 只有两个例外, 找出例外的那两个 总计 1. 剑指 offer 上有这两道题的解法, ...

  2. shell脚本学习总结03--别名的使用

    1.创建别名 $ alias dms='cd Oracle/Middleware/user_projects/domains/7001_costctl/' $ dms $ dms $ pwd /hom ...

  3. handlebears使用

    Handlebars 文档笔记: http://www.ghostchina.com/handlebars-wen-dang-bi-ji/ Handlebars模板引擎中的each嵌套及源码浅读: h ...

  4. Javascript通过bind()掌控this

    Javascript通过bind()掌控this: http://blog.csdn.net/rznice/article/details/26134201 bind能为我们做些什么,同时它的好处在哪 ...

  5. eclipse中的SVN文件还原到历史版本

    转载自:http://www.softown.cn/post/103.html 由于某些特殊原因,我们可能需要将SVN资源库中的某个文件回滚到以前的某个历史版本(准确地说,这不是"回滚&qu ...

  6. img元素的alt和title的区别?

    alt是图片加载失败时显示在网页上的提示信息: title是鼠标放上面时显示的文字(图片加载失败鼠标放显示的代替文字上时仍然会显示提示信息): 除此之外,alt是img的必要属性,只能用在img.ar ...

  7. http协议----->请求头和响应头

    http实用头字段-----Range 如果请求里有这个range头,那么响应里也有 1.首先在webroot下放好a.txt 内容如下: 2.然后在本地有个下载未完成的a.txt 本地a.txt内容 ...

  8. A TCP connection is distinguished by four values

    4个值唯一地定义一条TCP连接. HTTP The Definitive Guide A computer might have several TCP connections open at any ...

  9. Python实现HMM(隐马尔可夫模型)

    1. 前言 隐马尔科夫HMM模型是一类重要的机器学习方法,其主要用于序列数据的分析,广泛应用于语音识别.文本翻译.序列预测.中文分词等多个领域.虽然近年来,由于RNN等深度学习方法的发展,HMM模型逐 ...

  10. 剑指Offer——字符串的排列

    题目描述: 输入一个字符串,按字典序打印出该字符串中字符的所有排列.例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bca,cab和cba. 输入描述: ...