zoukankan      html  css  js  c++  java
  • Python学习笔记八_函数

    一、函数是什么

    定义:函数是指一组语句的集合通过一个名字(函数名)封装起来,想要执行这个函数,只需要调用函数名即可。

    C中的函数叫function,java中的函数叫method,python中的函数用def做关键字。

    二、使用函数的好处

    1.提高代码的复用性

    2.让代码更简洁、简化代码

    3.代码可扩展

    三、python中函数的定义

    定义函数使用def关键字,后面是函数名,函数名不能重复

    def sayHello(): #定义函数,函数名
        print('hello') #函数体
    #函数不调用是不会被执行的
    sayHello() #调用函数

    四、函数的参数

    1、形参与实参

    函数在调用的时候,可以传入参数,有形参和实参。形参就是函数接收的参数,而实参就是你实际传入的参数

    形参:

    只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。

    实参:

    可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。函数调用结束返回主调用函数后则不能再使用该形参变量。

    def calc(a,b): #形参:形式参数
        print('%s * %s = %s'%(a,b,a*b))
    calc(7,8) #实参:实际参数

    2、函数的四种形参

       (1) 位置参数

    按照参数的位置来传参,calc这个函数中,a和b就是位置参数,是必传的,没有会报错

    def calc(a,b): #位置参数,必填,不填会报错missing 2 required positional arguments
        print('%s * %s = %s'%(a,b,a*b))
    calc(7,8) 

    有几个位置参数在调用的时候就要传几个,否则会报错,如果有多个位置参数的话,记不住哪个位置传哪个了,可以使用位置参数的名字来指定调用

    def size(ength,width,height): 
        print('Size is %s'%(ength * width * height))
    size(height=7,ength=8,width=10)#关键字传参,参数过多可以使用未知参数的名字来指定调用
    def op_mysql(host,port,username,passwd,charset,sql):
        print('连接数据库')
    #一一对应传参 op_mysql('192.168.1.1',3306,'Amy','sfsd','utf-8','select * from user')

    #不按顺序传参(关键字传参) op_mysql(sql = 'select * from user', host = '192.168.1.1', username = 'Amy', passwd = 'sfsd', charset = 'utf-8', port = '3306' )
    #也可以将两种传参相结合 op_mysql('192.168.1.1',3306, sql = 'select * from user', username = 'Amy', passwd = 'sfsd', charset = 'utf-8', )

     (2) 默认参数

    在定义形参的时候,给函数默认赋一个值,比如说数据库的端口这样的,默认给它一个值,这样就算你在调用的时候没传入这个参数,它也是有值的。

    默认参数不是必填的,如果给默认参数传值,它就会使用你传入的值。如果使用默认值参数的话,必须放在位置参数后面定义

    def op_file(file_name,content=None): #content:默认值参数,它是非必填的
        f = open(file_name, 'a+', encoding='utf-8')
        f.seek(0)
        if content: #不为空代表写文件
            f.write(content)
            f.flush()
        else: #为空代表读文件
            all_user = f.read() #all_users是局部变量
            return all_user #调用完函数之后,返回什么结果
        f.close()
    print(op_file('username'))

       (3) 非固定参数

    位置参数和默认参数的参数个数都是固定的,如果一个函数参数是不固定的,以后也不知道会扩展成什么样,那么再用固定函数,后面程序就不好扩展了。这时候需要非固定参数。非固定参数有两种,可变参数和关键字参数。

         (i) 可变参数

        可变参数用*来接收,后面想传多少个参数就传多少个,如果位置参数、默认值参数、可变参数一起使用的的话,可变参数必须在位置参数和默认值参数后面。

        a、非必填参数

        b、它没有限制参数个数

        b、它把传回来的参数放在元组

    def syz(name,*args): #参数组
        # 位置参数、默认值参数、可变参数,可变参数会把后面多传的参数都放到args这个元组中
        # args名字是随便写的,不过一般我们都写args
        print(name,args)
    syz('双鱼座') #双鱼座 (),非必填
    syz('双鱼座','111') #双鱼座 ('111',)
    syz('asd','sdf',23224) #asd ('sdf', 23224),没有限制参数个数

         (ii) 关键字参数

        关键字参数用**来接收,后面的参数也是不固定的,位置参数、默认值参数、可变参数,关键字参数一起使用的话,关键字参数必须在最后面。

        a、非必填参数

        b、它没有限制参数个数

        c、它把传回来的参数放在字典

        d、使用关键字参数的话,调用的时候必须使用关键字传参

    def syz(name,**kwargs): #关键字参数,必须有k-v,两个参数
        # 位置参数、关键字参数,调用的时候会把传入的关键字参数放到kwargs这个字典中
        print(name,kwargs)
    syz('双鱼座') #非必填,双鱼座 {}
    syz('双鱼座','111') #报错,必须有k-v,两个参数
    syz('双鱼座',room = '111') #双鱼座 {'room': '111'}
    syz('双鱼座',room = '111',addr = '北京') #没有限制参数个数,双鱼座 {'addr': '北京', 'room': '111'}

    五、函数的返回值

      每个函数都有返回值,如果没有在函数里指定返回值,则默认返回None,函数可以有多个返回值,如果有多个返回值,会把返回值放到一个元组中,返回的是一个元组。

    def calc(x,y):
        res = x * y
        return x,y,res
    print(calc(2,3)) #多个返回值的函数,返回一个元组(2, 3, 6)

      返回值作用:

      1、把函数处理的结果返回回来,供后面的程序用。

      2、结束函数,函数里面遇到return,函数会立即结束

    def r():
        for i in range(5):
            if i == 3:
                print(i)
                return #只写一个return的话,就返回None
    r() #3,遇到return函数立即结束
    res = r()
    print(res) #默认返回None
    import string
    def check(pwd):
        if len(pwd) >5 and len(pwd) <12: #长度6-11
            if set(pwd) & set(string.ascii_letters) and set(pwd) & set(string.digits):#必须包含字符和数字
                print('密码合法')
            else:
                print('密码不合法')
        else:
            print('密码不合法')
    res = check('ad1234') #调用函数,并把函数返回值赋给res
    # 函数里面如果没有return的话,默认返回none,return不是必须写的
    print(res)

    六、局部变量和全局变量

    常量:

    一个不变的值,通常用大写字母定义

    PORT = 3306 #常量
    FLIENAME = 'user.txt'

    局部变量:

    函数里面定义的变量,都是局部变量,只能在函数里面用,出了函数之后就不能用了

    name = 'Lily' #全局变量
    def sayName():
        name = 'Kate'#局部变量
        print(name) #kate
    sayName()
    print(name) #Lily

    全局变量:

    公共的变量,都可以用的变量。

    在整个程序里面都生效的,在程序最前面定义的都是全局变量,全局变量如果要在函数中修改的话,需要加global关键字声明,如果是list、字典和集合的话,则不需要加global关键字,直接就可以修改

    name = 'Lily' #字符串全局变量
    names = [] #list全局变量
    def sayName():
        global name #修改全局变量name,加global关键字
        name = 'Kate'
        names.append(name)#修改全局list变量,不用加global关键字
        print(name) #kate
    sayName()
    print(name) #kate
    print(names) #['Kate'] 

    最好少用全局变量的原因:

      (1) 不安全,因为所有人都可以改
      (2) 全局变量它会一直占着内存

    七、递归

     函数自己调用自己。

     在函数内部,可以调用其他函数。如果一个函数在内部调用自身本身,这个函数就是递归函数。

    递归调用的意思就是,在这个函数内部自己调用自己,就有点循环的意思。

    def test1():
        num = int(input('please enter a number:'))
        if num%2==0:#判断输入的数字是不是偶数
           return True #如果是偶数的话,程序就退出了,返回True
        print('不是偶数请重新输入!')
        return test1()#如果不是偶数的话继续调用自己,输入值
    print(test1())#调用test

    递归调用的特性:

    1. 必须有一个明确的结束条件

    2. 每次进入更深一层递归时,问题规模相比上次递归都应有所减少

    3. 递归效率不高,递归层次过多会导致栈溢出(在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出)

    少用递归,递归最多递归999,递归的效率不高

    i = 0
    def test():
        global i
        i += 1
        print(i)
        test()
    test() #循环999次后会报错,3.6版本以后不会报错

    八、高阶函数

    如果一个函数的参数是一个函数的话,那么这个函数就是一个高阶函数

    def s_int(n):#把传入的参数类型转换成int类型
        return int(n)
    def add(x,y,z):#接收3个参数,x,y,z,z是一个函数
        print(z(x)+z(y))#z是一个函数,把x和y的值传给z,然后用z函数的返回值把两个值相加
    add('8','9',s_int)#调用,传入x和y的值,再把上面的那个定义好的函数传进去
  • 相关阅读:
    MySQL权限详解
    MySql 详解
    顶级Python库
    第一次读到就震撼的句子
    Windows快捷键大全
    Pycharm超级好用的快捷键——效率之王
    Django框架
    前端入门和进阶必会
    正则表达式BREs,EREs,PREs的比较
    selenium模块基础用法详解
  • 原文地址:https://www.cnblogs.com/dongrui624/p/8885132.html
Copyright © 2011-2022 走看看