Python_oldboy_自动化运维之路(四)
本节内容
- 集合
- 字符编码与转码
- 函数语法及基本特性
- 函数参数与局部变量
- 返回值和嵌套函数
- 递归
- 匿名函数
- 高阶函数
1.集合
集合是一个无序的,不重复的数据组合,它的主要作用如下:
- 去重,把一个列表变成集合,就自动去重了
- 关系测试,测试两组数据之前的交集、差集、并集等关系
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/ a = {2,3,4,5,6}
b = {1,3,5,8,9} print(a)
print(b)
print(a & b) #交集
print(a.intersection(b))
#print(a.intersection_update(b)) #a= a.a.intersection(b) print(a | b) #并集
print(a.union(b)) print(a - b) #差集a有b没有
print(a.difference(b))
print(b - a) #b有a没有
print(b.difference(a)) print(a ^ b) #对称差集,就是除了ab的交接
print(a.symmetric_difference(b)) c = {3}
print(c)
print(b) print(c.issubset(b)) #测试是否 c 中的每一个元素都在 b 中 c是不是b的子集
print(b.issuperset(c)) #b是不是c的父集
2.字符编码与转码
【python2环境下演示:】
1.先用python2写个代码(utf-8),然后在windows运行(GBK),发现无法正常显示。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
msg = "中国"
decode:解码
encode:编码
2.utf-8转换成gbk,需要先解码,然后在编码。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/ msg = "中国"
print msg
print msg.decode(encoding="utf-8").encode(encoding="gbk") #第一个是告诉原来是什么编码,第二个是告诉要编码成什么
3.编码之间的转换
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
msg = "中国"
print msg gbk_str = msg.decode(encoding="utf-8").encode(encoding="gbk") #第一个是告诉原来是什么编码,第二个是告诉要编码成什么
print gbk_str print gbk_str.decode(encoding="gbk").encode(encoding="utf-8") #在从GBK编码成uft-8
print gbk_str.decode(encoding="gbk").encode(encoding="gb2312") #在从GBK编码成gb2312
4.看一下uncode是什么模样,为什么也能显示出来?原因很简单,命令行默认将uncode解码成gbk显示的屏幕上,在内存中还是uncode
print gbk_str.decode(encoding="gbk") #看下uncode是什么样的,只有解码,没有编码,还能正常显示中国的原因是:windows默认GBK,默认支持unicode和GBK print [gbk_str,] #在内存中对比下两种编码,不变成列表看不出区别来。
print [gbk_str.decode(encoding="gbk"),]
【python3.5下演示】
注:python3的代码在内存中是unicode,在硬盘存储是(文件编码)utf-8,因为python3的解释器就是unicode,也就是说不用解码decode,只需要编码encode。
1.所以python3只需要编码encode,为啥还是不能正常显示?
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/ msg = "中国"
print (msg.encode(encoding="utf-8")) #在编码就变成的bytes格式。
2.试一下不用转的,是直接可以打印显示。因为刚才已经说过,windows命令行默认支持gbk和uncode,所以不需要转码就可以显示。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/ msg = "中国"
print (msg.encode(encoding="utf-8")) print(msg)
【总结知识点】
1.windows默认是gbk,默认支持unicode和gbk。
2.python3默认文件编码是utf-8,解释器编码是unicode,文件加载到内存会自动解码成unicode,同时把字符转换成bytes格式,bytes=8bits(位)。
3.python2默认是assic,python是1989年诞生,unicode是1991年诞生,所以不支持中文,#-*-coding:utf-8 -*- 表示告诉后面的代码,我用uft-8来解释。
4.python3解决了所以国家通用的编码,因为python的解释器用的是unicode,window支持unicode,所以不需要转码。
5.python2的str就是python3的bytes,python3的str就是unicode,python3多出来的bytes格式是一个单独的数据类型。
6.python2在windows上解码decode动作是必须的,编码成gbk这个动作不是必须的,在linux上gbk>uft-8,解码必须,编码成utf-8不是必须。
7.所有的程序在内存里默认都是unicode,编码encode动作不是必须的,当存储到硬盘才需要编码。
8.python3不需要解码和编码。
3.函数基本语法及特性
背景提要
现在老板让你写一个监控程序,监控服务器的系统状况,当cpu\memory\disk等指标的使用量超过阀值时即发邮件报警,你掏空了所有的知识量,写出了以下代码
while True:
if cpu利用率 > 90%:
#发送邮件提醒
连接邮箱服务器
发送邮件
关闭连接 if 硬盘使用空间 > 90%:
#发送邮件提醒
连接邮箱服务器
发送邮件
关闭连接 if 内存占用 > 80%:
#发送邮件提醒
连接邮箱服务器
发送邮件
关闭连接
上面的代码实现了功能,但即使是邻居老王也看出了端倪,老王亲切的摸了下你家儿子的脸蛋,说,你这个重复代码太多了,每次报警都要重写一段发邮件的代码,太low了,这样干存在2个问题:
- 代码重复过多,一个劲的copy and paste不符合高端程序员的气质
- 如果日后需要修改发邮件的这段代码,比如加入群发功能,那你就需要在所有用到这段代码的地方都修改一遍
你觉得老王说的对,你也不想写重复代码,但又不知道怎么搞,老王好像看出了你的心思,此时他抱起你儿子,笑着说,其实很简单,只需要把重复的代码提取出来,放在一个公共的地方,起个名字,以后谁想用这段代码,就通过这个名字调用就行了,如下
def 发送邮件(内容)
#发送邮件提醒
连接邮箱服务器
发送邮件
关闭连接 while True: if cpu利用率 > 90%:
发送邮件('CPU报警') if 硬盘使用空间 > 90%:
发送邮件('硬盘报警') if 内存占用 > 80%:
发送邮件('内存报警')
你看着老王写的代码,气势恢宏、磅礴大气,代码里透露着一股内敛的傲气,心想,老王这个人真是不一般,突然对他的背景更感兴趣了,问老王,这些花式玩法你都是怎么知道的? 老王亲了一口你儿子,捋了捋不存在的胡子,淡淡的讲,“老夫,年少时,师从京西沙河淫魔银角大王 ”, 你一听“银角大王”这几个字,不由的娇躯一震,心想,真nb,怪不得代码写的这么6, 这“银角大王”当年在江湖上可是数得着的响当当的名字,只可惜后期纵欲过度,卒于公元2016年, 真是可惜了,只留下其哥哥孤守当年兄弟俩一起打下来的江山。 此时你看着的老王离开的身影,感觉你儿子跟他越来越像了。。。
函数是什么?
函数一词来源于数学,但编程中的「函数」概念,与数学中的函数是有很大不同的,具体区别,我们后面会讲,编程中的函数在英文中也有很多不同的叫法。在BASIC中叫做subroutine(子过程或子程序),在Pascal中叫做procedure(过程)和function,在C中只有function,在Java里面叫做method。
定义: 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数,只需调用其函数名即可
特性:
- 减少重复代码
- 使程序变的可扩展
- 使程序变得易维护
语法定义
def sayhi():#函数名
print("Hello, I'm nobody!") sayhi() #调用函数
可以带参数
#下面这段代码
a,b = 5,8
c = a**b
print(c) #改成用函数写
def calc(x,y):
res = x**y
return res #返回函数执行结果 c = calc(a,b) #结果赋值给c变量
print(c)
案例1:函数内部是可以修改列表,字典,集合,实例,类。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/ # def sayhi(name): #函数名
# print("my name is %s" % name)
#
# sayhi("lijun")
# sayhi("licai")
# sayhi("yongsan") #函数内部是可以修改列表,字典,集合,实例,类
#print(id(names),id(names[0])) //内存中的单元格是不一样
#5380696 5386592
#为什么以上可以改?因为names是又在内存中指向的chenlijun,panglicai,重新在内存中指向别的内存空间是可以的,原来的指向并没有改变,就是元素可以改。 names = ["chenlijun","panglicai"]
info = {"names":"星驰"} def change():
print(names)
names.append("liyongsan")
info["age"] = 22 change()
print(names,info)
4.函数参数与局部变量
形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量
实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值
【默认参数】
def stu_register(name,age,country,course):
print("----注册学生信息------")
print("姓名:",name)
print("age:",age)
print("国籍:",country)
print("课程:",course) stu_register("王山炮",22,"CN","python_devops")
stu_register("张叫春",21,"CN","linux")
stu_register("刘老根",25,"CN","linux")
发现 country 这个参数 基本都 是"CN", 就像我们在网站上注册用户,像国籍这种信息,你不填写,默认就会是 中国, 这就是通过默认参数实现的,把country变成默认参数非常简单
def stu_register(name,age,course,country="CN"):
这样,这个参数在调用时不指定,那默认就是CN,指定了的话,就用你指定的值。
另外,你可能注意到了,在把country变成默认参数后,我同时把它的位置移到了最后面,为什么呢?
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#函数
#默认参数
#增加两个默认参数,不填写,会自动打印 def stu_register(name, age,course,country="CN",hobbie="girl"):
print("----注册学生信息------")
print("姓名:", name)
print("age:", age)
print("国籍:", country)
print("课程:", course)
print("爱好:", hobbie) stu_register("王山炮", 22, "python_devops")
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#函数
#位置参数
#若要传参数进去,会自动更改默认参数 def stu_register(name, age,course,country="CN",hobbie="girl"):
print("----注册学生信息------")
print("姓名:", name)
print("age:", age)
print("国籍:", country)
print("课程:", course)
print("爱好:", hobbie) stu_register("王山炮", 22, "python_devops","JP","boy")
【关键参数】
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#函数
#关键参数
#正常情况下,给函数传参数要按顺序,不想按顺序就可以用关键参数,只需指定参数名即可,但记住一个要求就是,关键参数必须放在位置参数之后。 def stu_register(name, age,course,country="CN",hobbie="girl"):
print("----注册学生信息------")
print("姓名:", name)
print("age:", age)
print("国籍:", country)
print("课程:", course)
print("爱好:", hobbie) stu_register("王山炮", 22, "python_devops",hobbie="boy",country="JP")
【非固定参数】
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#函数
#非固定参数:*args,**kwargs
#*args 会把多传入的参数变成一个元组形式,不指定值
#*kwargs 会把多传入的参数变成一个dict形式,指定值 def stu_register(name, age,course,country="CN",hobbie="girl",*args,**kwargs):
print("----注册学生信息------")
print("姓名:", name)
print("age:", age)
print("国籍:", country)
print("课程:", course)
print("爱好:", hobbie)
print(args,kwargs) stu_register("王山炮", 22, "python","jp","boy","test1","test2",id="")
总结:
实参:所有的数据类型都可以被当做参数传给函数
形参:只有在被调用时才分配内存,调用结束后立即释放内存,值仅在函数内有效(局部变量,形参的作用域只在当前函数内部)。 位置参数:按顺序
默认参数:必须放到位置参数后面
关键参数:必须放到位置参数后面
非固定参数:*args=() 以位置的形式传人,**kwargs = {}以关键字的形式传人
【局部变量】
name = "Alex Li" def change_name(name):
print("before change:",name)
name = "金角大王,一个有Tesla的男人"
print("after change", name) change_name(name) print("在外面看看name改了么?",name)
输出:
before change: Alex Li
after change 金角大王,一个有Tesla的男人
在外面看看name改了么? Alex Li
全局与局部变量
5.返回值和嵌套函数
要想获取函数的执行结果,就可以用return语句把结果返回
注意:
- 函数在执行过程中只要遇到return语句,就会停止执行并返回结果,so 也可以理解为 return 语句代表着函数的结束
- 如果未在函数中指定return,那这个函数的返回值为None
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#函数
#非固定参数:*args,**kwargs
#*args 会把多传入的参数变成一个元组形式,不指定值
#*kwargs 会把多传入的参数变成一个dict形式,指定值 def stu_register(name, age,course,country="CN",hobbie="girl",*args,**kwargs):
print("----注册学生信息------")
print("姓名:", name)
print("age:", age)
print("国籍:", country)
print("课程:", course)
print("爱好:", hobbie)
print(args,kwargs)
print(kwargs['addr']) #如何调用字典里的元素
return True,name rest = stu_register("王山炮", 22, "python","jp","boy","test1","test2",id="",addr="回龙观") print(rest)
【嵌套函数】
函数可以嵌套函数,但是change_name2就不能被调用了,局部变量只能用在当前作用域。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/ name = "lijun" def change_name():
name = "lijun2" def change_name2():
name = "lijun3"
print("第三层name",name) change_name2() #调用内层函数
print("第二层name", name) change_name()
print("最外层名字",name)
6.递归
在函数内部,可以调用其他函数。如果一个函数在内部调用自身本身,这个函数就是递归函数。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
def calc(n):
print(n)
if int(n/2) > 0:
calc(int(n/2)) print(n) calc(10)
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#如果想要显示最小的那个除2大于0的那个数,就给赋个值
def calc(n):
print(n)
if int(n/2) > 0:
n = calc(int(n/2)
print(n)
return n print("res",calc(10))
递归特性:
1. 必须有一个明确的结束条件
2. 每次进入更深一层递归时,问题规模相比上次递归都应有所减少
3. 递归效率不高,递归层次过多会导致栈溢出(在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出)
堆栈扫盲http://www.cnblogs.com/lln7777/archive/2012/03/14/2396164.html
【利用递归实现个算法,二分查找】
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#输入一个值在列表里查找出来
dat = [1,2,3,4,5,6,7,8,9,23,34,54,66,77,89,666] def binary_search(find_str,det_set):
mid = int(len(det_set)/2)
if mid == 0: #如果列表长度就只有一个
if det_set[mid] == find_str: #等于查找的值就显示
print("find it",find_str)
else: #不等于就说明不存在
print("not find,not list",find_str) return #结束 if det_set[mid] == find_str:
print("find it",find_str,mid) #拆分一半的值正好等于要查找的值,就说明找到了 elif det_set[mid] > find_str:
print("going to search in left",det_set[mid],det_set[0:mid])
binary_search(find_str,det_set[0:mid]) #如果拆分一半的值比查找的值大,就打印拆分一半的值左边的内容 else:
print("going to search in right",det_set[mid],det_set[mid+1:])
binary_search(find_str,det_set[mid+1:]) #如果拆分一半的值比查找的值小,就打印拆分一半的值右边的内容 binary_search(23,dat)
7.匿名函数
匿名函数就是不需要显式的指定函数
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#匿名函数 def calc(n):
return n**n # print(calc(2))
#
# calc2 = lambda n:n**n
# print(calc2(2))
#
# calc3 = lambda n,y:n**y
# print(calc3(4,2)) #匿名函数的用法。好处:普通的函数一般是先定义好,重复被调用,而匿名函数应用场景为,我只调用一次函数,用完就不在调用了。
#map(函数,[1,2,3]) //前面是函数,后面的参数是可调用的,可循环的
for i in map(calc,[4,5,6]):
print(i) for k in map(lambda x:x*x,[1,2,3]):
print(k)
lambda复杂的用法:
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#匿名函数,复杂的语法,最复杂的运算只支持三元运算。
#
# def calc(n):
# if n>5:
# return n*2
# else:
# return n-1
#
# for i in map(calc,[1,2,3,4,5,6,7,8,9]):
# print(i) for i in map(lambda x:x*2 if x>5 else x-1,[1,2,3,4,5,6,7,8,9]):
print(i)
8.高阶函数
变量可以指向函数,函数的参数能接收变量,那么一个函数就可以接收另一个函数作为参数,这种函数就称之为高阶函数。
# -*- coding: UTF-8 -*-
#blog:http://www.cnblogs.com/linux-chenyang/
#高阶函数
#1.把一个函数的内存地址当做参数传给另外一个函数。
#2.一个函数把另外一个函数当做返回值返回。 def add(x, y, f):
return f(x) + f(y) res = add(3, -6, abs) #abs() 返回绝对值,永远得正
print(res)
Python_oldboy_自动化运维之路(四)的更多相关文章
- Python_oldboy_自动化运维之路_socket编程(十)
链接:http://www.cnblogs.com/linhaifeng/articles/6129246.html 1.osi七层 引子: 须知一个完整的计算机系统是由硬件.操作系统.应用软件三者组 ...
- Python_oldboy_自动化运维之路_面向对象2(十)
本节内容: 面向对象程序设计的由来 什么是面向对象的程序设计及为什么要有它 类和对象 继承与派生 多的态与多态性 封装 静态方法和类方法 面向对象的软件开发 反射 类的特殊成员方法 异常处理 1.面向 ...
- Python_oldboy_自动化运维之路(八)
本节内容: 列表生成式,迭代器,生成器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 1.列表生成式,迭代器,生成器 1.列表生成式 #[列表生成] #1.列 ...
- Python_oldboy_自动化运维之路_函数,装饰器,模块,包(六)
本节内容 上节内容回顾(函数) 装饰器 模块 包 1.上节内容回顾(函数) 函数 1.为什么要用函数? 使用函数之模块化程序设计,定义一个函数就相当于定义了一个工具,需要用的话直接拿过来调用.不使用模 ...
- Python_oldboy_自动化运维之路(一)
python简介: Python 是一个高层次的结合了解释性.编译性.互动性和面向对象的脚本语言. Python 的设计具有很强的可读性,相比其他语言经常使用英文关键字,其他语言的一些标点符号,它具有 ...
- Python_oldboy_自动化运维之路_paramiko,mysql(十二)
本节内容: paramiko mysql 1.paramiko http://www.cnblogs.com/wupeiqi/articles/5095821.html paramiko是一个模块,s ...
- Python_oldboy_自动化运维之路_线程,进程,协程(十一)
本节内容: 线程 进程 协程 IO多路复用 自定义异步非阻塞的框架 线程和进程的介绍: 举个例子,拿甄嬛传举列线程和进程的关系: 总结:1.工作最小单元是线程,进程说白了就是提供资源的 2.一个应用程 ...
- Python_oldboy_自动化运维之路_面向对象(十)
面向对象编程 OOP编程是利用“类”和“对象”来创建各种模型来实现对真实世界的描述,使用面向对象编程的原因一方面是因为它可以使程序的维护和扩展变得更简单,并且可以大大提高程序开发效率 ,另外,基于面向 ...
- Python_oldboy_自动化运维之路_全栈考试(七)
1. 计算100-300之间所有能被3和7整除的所有数之和 # -*- coding: UTF-8 -*- #blog:http://www.cnblogs.com/linux-chenyang/ c ...
随机推荐
- CF960G Bandit Blues 【第一类斯特林数 + 分治NTT】
题目链接 CF960G 题解 同FJOI2016只不过数据范围变大了 考虑如何预处理第一类斯特林数 性质 \[x^{\overline{n}} = \sum\limits_{i = 0}^{n}\be ...
- JsonTools
public class JsonTools { /// <summary> /// Generate Json string from the object /// </summa ...
- 解题:APIO/CTSC 2007 数据备份
题面 用双向链表把相邻两项的差串起来,用大根堆维护价值,每次贪心取最大的$x$.取完之后打标记删掉$pre[x]$和$nxt[x]$,之后用$val[pre[x]]+val[nxt[x]]-val[x ...
- Java入门:基础算法之计算三角形面积
本部分介绍如何计算三角形面积. /** * @author: 理工云课堂 * @description: 程序计算三角形的面积.三角形的底和高由用户输入 */ import java.util.Sca ...
- element ui 栅格布局
<el-row> <el-col :span="24"><div class="grid-content bg-purple-dark&qu ...
- mongo同步到es
刚开始我找到的方案是利用 ElasticSearch 的 River 来同步数据,并在 GitHub 上到了 MongoDB River 插件:elasticsearch-river-mongodb. ...
- 应用jfinal发送微信模板消息的一个bug
严格来讲,这不是一个bug,只是我们应用的方式不对.微信发送模板消息的方法是: HttpUtils.post(sendApiUrl + AccessTokenApi.getAccessTokenStr ...
- HDU 5446 lucas CRT
n中选m个模M,M为多个素数之积 $n, m, k (1 \leq m \leq n \leq 10^{18}, 1 \leq k \leq 10)$,$M = p_1 · p_2 · · · p_k ...
- Git之远程库与本地库交互
对于开发人员而言,有时候在公司干完会到家还得接着干,为了方便同步代码,于是乎,可通过GitHub代码托管平台实现代码云同步. GitHub账号开通.创建远程仓库及初始化等操作此处不做赘述: Git内部 ...
- 接口测试Case之面向页面对象编写及规范
一.什么是页面对象化 主要提倡的思想是:万物皆对象,即把一个Page看成一个对象,来进行接口自动化Case的编写,不要闲扯,直接讲怎么个操作法呢? 二.有什么优势? 2.1 Case层次清晰,便于管理 ...