函数的基础概念:
函数是python为了代码最大程度地重用和最小化代码冗余而提供的基本结构
函数是一种设计工具,它能让程序员将复杂的系统分解为可管理的部件
函数用于将相关功能打包并参数化
在python中能够建立4种函数
(1)全局函数:定义在模块
(2)局部函数:嵌套于其它函数中
(3)lambda函数:表达式,如需屡次调用
(4)方法:与特定数据类型关联的函数,而且只能与数据类型关联一块儿使用
python提供了不少内置函数html
建立函数:python
def functionName(arg1,arg2,...): suite return
函数的定义主要有以下要点:git
一些相关的概念:
def是一个可执行语句
所以能够出如今任何可以使用语句的地方,甚至能够嵌套于其余语句,例如if或while中
def建立了一个对象并将其赋值给一个变量名(即函数名),函数体则存在内存
return用于返回结果对象,其为可选;无return语句的函数自动返回None对象
返回多个值时,彼此间使用逗号分隔,且组合为元组形式返回一个对象
函数一旦执行到return,函数就终止了,若是return下面还有执行语句则终止
def语句运行以后,能够在程序中经过函数后附加()进行调用程序员
函数做用域:
python建立、改变或查找变量名都是在名称空间中进行
在代码中变量名被赋值的位置决定了其能被访问到的范围
函数定义了本地做用域,而模块定义了全局做用域编程
1.全局变量所有用大写表示
2.全局变量均可以被访问,函数内部的变量则为本地做用域
3.在函数内若是要修改全局变量,须要global
4.特殊:字典,列表能够在函数内修改,可是不能从新赋值tomcat
更多做用域服务器
NAME = 'tomcat' def f1(): age = 18 global NAME NAME = 'xiao' print(age,NAME) def f2(): age = 27 print(age,NAME) f1() f2()
函数的参数:
默认状况下,参数经过其位置进行传递,从左至右,这意味着,必须精确地传递和函数头部参数同样多的参数
但也能够经过关键字参数、默认参数或参数容器等改变这种机制
(1)普通参数:定义函数时从左至右
(2)默认参数:定义函数时是使用"name=value"的语法直接给变量一个值,从而传入的值能够少于参数个数
(3)指定参数:调用函数时指定"name形式参数=value实际参数"的语法经过参数名进行匹配
(4)动态参数:定义函数时形式参数中收集任意多基于普通参数
定义函数时使用* :收集普通参数,返回元组,*args
定义函数时使用**:收集指定参数,返回列表,**kwargs
(5)动态参数解包:调用函数时,使用**开头的参数,从而传递任意多基于普通或指定参数
注意:定义函数时
1.混用普通参数和默认参数,应当把默认参数写到右侧
2.混用有默认和无默认值的参数时,无默认值放左侧app
一、返回值ide
函数是一个功能块,该功能到底执行成功与否,须要经过返回值来告知调用者。函数式编程
以上要点中,比较重要有参数和返回值:
def 发送短信(): 发送短信的代码... if 发送成功: return True else: return False while True: # 每次执行发送短信函数,都会将返回值自动赋值给result # 以后,能够根据result来写日志,或重发等操做 result = 发送短信() if result == False: 记录日志,短信发送失败...
二、参数
没有参数的状况
#!/usr/bin/env python # -*- coding:utf-8 -*- ''' 1.须要开启邮箱服务sendmail 2.邮箱服务器须要开启SMTP服务 ''' def sendmail(): try: import smtplib from email.mime.text import MIMEText from email.utils import formataddr msg = MIMEText('邮件内容', 'plain', 'utf-8') msg['From'] = formataddr(["发件人", 'pythonxiao@126.com']) msg['To'] = formataddr(["收件人", '329275108@qq.com']) msg['Subject'] = "邮件主题" server = smtplib.SMTP("smtp.126.com", 25) server.login("pythonxiao@126.com", "xiaozhiqi2016") server.sendmail('pythonxiao@126.com', ['329275108@qq.com', ], msg.as_string()) server.quit() except: return False else: return True ret = sendmail() if ret == True: print("发送成功") else: print("发送失败")
有参数的状况:
#!/usr/bin/env python # -*- coding:utf-8 -*- def sendmail(email,content): try: import smtplib from email.mime.text import MIMEText from email.utils import formataddr msg = MIMEText(content, 'plain', 'utf-8') msg['From'] = formataddr(["发件人", 'pythonxiao@126.com']) msg['To'] = formataddr(["收件人", '329275108@qq.com']) msg['Subject'] = "邮件主题" server = smtplib.SMTP("smtp.126.com", 25) server.login("pythonxiao@126.com", "xiaozhiqi2016") server.sendmail('pythonxiao@126.com', [email, ], msg.as_string()) server.quit() except: return "失败" else: return 'cc' while True: msg = input("请输入邮箱地址:") #实际参数 ret = sendmail(msg,"SB") if ret == 'cc': print("发送成功") break else: print("发送失败")
从上面例子能够看出有参数的函数表现的很是灵活可扩展!
函数的有三中不一样的参数(定义函数时):
指定参数和动态参数解包是发生在调用函数
(1)# xxoo为普通参数也叫形式参数,简称:形参 def f1(name): print(name) # 'test' 叫作函数的实际参数,简称:实参 f1('tomcat')
(2)# name 是普通参数,ab=5是默认参数 def f1(name,ab=5): print(name,ab) # 指定参数 f1('tomcat',12) # 使用默认参数 f1('tomcat') 注意:默认参数须要放在参数列表最后
(3)# 动态参数(1):*args 返回的是一个元组 def f1(*args): print(args,type(args)) # 执行方式一 f1(11) # 若是传入一个列表,则这个列表是看成元组中的一个元素 li = ['a','b','c'] f1(li) # 执行方式二:执行函数时有*,把全部迭代对象拆分为单个元素做为元组的元素,如传入列表,会把列表中每个元素遍历添加到元组中看成一个元素 f1(*l1)
(4)#动态参数(2):**kwargs 返回的是一个字典 def f1(**kwargs): print(args,type(args)) # 执行方式一:只能传入指定参数 f1(n1='alex',n2=18) dic = {'k1':'v1','k2':'v2'} f1(kk=dic) # 执行方式二:实际参数若是有**,传入的应该是一个字典,会把每一对键值对像不带**同样把指定参数传入函数中 f1(**dic)
(5)# 俗称万能参数 def f1(*args,**kwargs): print(args,type(args)) print(kwargs,type(kwargs)) # 执行方式一 f1(11,22,33,k1='v1',k2='v2') # 执行方式二 l1 = [1,2,3,4] d1 = {'a':'xiao','b':'zhi','c':'qi'} f1(*l1,**d1)
在python中能够常常看到万能参数,好比str.format()用于格式化的方法,这里用不一样的参数传递方式实现相同的效果
>>> s1 = 'i am {0}, age {1}'.format('alex',18) >>> print(s1) i am alex, age 18 >>> s2 = 'i am {0}, age {1}'.format(*['alex',18]) >>> print(s2) i am alex, age 18 >>> s3 = 'i am {0}, age {1}'.format(*('alex',18)) >>> print(s3) i am alex, age 18 >>> s4 = 'i am {name}, age {age}'.format(name='alex',age=18) >>> print(s4) i am alex, age 18 >>> dic = {'name':'alex','age':18} >>> s5 = 'i am {name}, age {age}'.format(**dic) >>> print(s5) i am alex, age 18
3.补充:
(1).在定义相同函数时,后一个函数名会指向新的内存对象,函数名是函数体在内存中的引用,就像变量赋值同样,变量名是对对象在内存中的引用
def f1(a,b): return a + b def f1(a,b): return a * b ret = f1(8,8) print(ret)
(2).函数在传递参数的时候是内存中的引用,而不是复制,因此l1的值是[1,2,3,4,999]
def f1(a1): a1.append(999) l1 = [1,2,3,4] f1(l1) print(l1)
稍微改变下,用一个变量来接收函数的返回值,这里的l1的值不是[1,2,3,4,999],而是None,由于函数内并无定义返回值,因此为None
def f1(a1): a1.append(999) l1 = [1,2,3,4] l1 = f1(l1) print(l1)
函数式编程:
之后写函数按照下面这个例子来编写,规范,这是一个登陆注册的函数编程
1.函数与函数间隔2行
2.函数须要注释,便于记忆
3.函数名须要见名知意
db文件格式
admin|123
xiao|12345
代码
#!/usr/bin/env python # coding: utf8 def datainput(): username = input("input your username:") password = input("input your password:") return username,password def login(username,password): f = open('db','r') for line in f: data = line.strip().split('|') if data[0] == username and data[1] == password: return True return False def register(username,password): f = open('db','r') for line in f: user = line.strip().split('|') #print(user) if username in user: return False else: f = open('db','a') userdata = "\n" + username + "|" + password f.write(userdata) f.close() return True def main(): num = input("1.登陆,2.注册:") if int(num) == 1: data = datainput() result = login(data[0],data[1]) if result: print('登陆成功') else: print('登陆失败') elif int(num) == 2: data = datainput() ret = register(data[0],data[1]) if ret: print("注册成功") else: print("注册失败,用户已存在") main()
三元表达式,三元运算,三目运算符都是指同一个意思:能够简化条件语句的缩写,可使代码看起来更加简洁,三目能够简单的理解为有三个变量,它的形式是这样的 name= k1 if 条件 else k2 ,若是条件成立,则 name=k1,不然name=k2,下面从代码里面来加深一下理解,从下面的代码明显能够看出三目运算符可使代码更加简洁。
a=1 b=2 if a<b: #通常条件语句的写法 k=a else: k=b c=a if a<b else b #三目运算符的写法,若是a<b为True,则c=a,不然c=b
对于简单的函数,也存在一种简便的表示方式,即:lambda表达式
# ###################### 普通函数 ###################### # 定义函数(普通方式) def func(arg): return arg + 1 # 执行函数 result = func(123) # ###################### lambda ###################### # 定义函数(lambda表达式) my_lambda = lambda arg : arg + 1 # 执行函数 result = my_lambda(123)
lambda表达式会自动return返回值,条件为真返回True,条件为假返回False.
若是一个函数在其内部调用它本身,就叫作递归,可是递归的时候要设置一个退出递归的条件,否则会一直递归下去,变成一个死循环。从另外一个方面来讲,递归其实和循环实际上是等价的。想要明白递归的话,咱们先从实际的例子上来讲明这一点,好比说咱们要写一个阶乘函数 f(n)算出n的阶乘,阶乘函数实现的方法不少,下面讲如何用递归实现
def f(n): if 0==n: # n=0 的话直接返回空,对用户输入的零进行判断 return None elif 1==n: # n=1 的话就再也不递归 return n else: return n*f(n-1) # 递归在执行f(n-1) 直到f(1) print(f(5)) # 120 ''' f(5)的执行过程以下 ===> f(5) ===> 5 * f(4) ===> 5 * (4 * f(3)) ===> 5 * (4 * (3 * f(2))) ===> 5 * (4 * (3 * (2 * f(1)))) ===> 5 * (4 * (3 * (2 * 1))) ===> 5 * (4 * (3 * 2)) ===> 5 * (4 * 6) ===> 5 * 24 ===> 120 '''
再来一个例子:
def func(n): n += 1 if n >= 4: return 'end' return func(n) r = func(4) print(r)
end #这是执行结果
下面的图帮助理解
传入1的时候,n=2不知足n>=4这个条件,return func(2)
传入2的时候,n=3不知足n>=4这个条件,return func(3)
传入3的时候,n=4知足n>=4这个条件,return end
return end --> return func(3) --> return func(2) --> r = func(1) 因此r的返回值就是end
def fun(s): digitnum, alphanum, sapcenum, othernum=0,0,0,0 for i in s: if i.isdigit(): digitnum+=1 elif i.isalpha(): alphanum+=1 elif i.isspace(): sapcenum+=1 else: othernum+=1 return (digitnum,alphanum,sapcenum,othernum)
def fun(s): ret=False if isinstance(s,str) or isinstance(s,str) or isinstance(s,tuple): if len(s)>5: ret=True return ret
def fun(s): ret=False if isinstance(s, str) or isinstance(s, str) or isinstance(s, tuple): for i in s: if i=='': ret=True break return ret
def fun(s): if isinstance(s,list): if len(s)>2: return s[0:2] return None
def fun(s): if isinstance(s,list) or isinstance(s,tuple): l=[] for i in range(1,len(s),2): l.append(s[i]) return l return None
def fun(s): if isinstance(s,dict): for i in s: if len(s[i])>2: s[i]=s[i][0:2] return s
def fun(n): if 1==n : return 0 elif 2==n: return 1 else: return fun(n-1)+fun(n-2)