NO.3:自学python之路
引言
本来计划每周完成一篇Python的自学博客,由于上一篇到这一篇遇到了过年、开学等杂事,导致托更到现在。现在又是一个新的学期,春天也越来越近了(冷到感冒)。好了,闲话就说这么多。开始本周的自学Python之路。而且,同时从这周开始,也要开始自学Tensorflow。希望能严格要求自己,不会托更。加油啦。
正文
这个周主要学习了集合,文件的操作,以及一点函数中的知识。下面将会详细介绍各个内容。
集合
集合也是Python的一种变量类型。它与列表不同,集合中没有顺序,没有重复。通常定义一个集合可以用以下两种方法。例子:
#直接写出集合中的各个元素 list = set([2,1,10,15,18]) #将列表转换为集合,会去掉重复的值 list = [1,5,8,2,4,3,6,1,3] list = set(list)
对集合的编辑首先从添加开始,向集合中添加一个或多个元素的方法,例子:
list = set([2,1,10,15,18])#建立集合 list.add(99)#添加一项 list.update([99,100,101])#添加多项
说完了添加,下面将会带来几种删除的方法,例子:
list = set([2,1,10,15,18])#建立集合 list.remove(2)#删除2,若list中不存在会报错 list.discard(114514)#删除114514,若list中不存在不会报错 list.pop()#随机删除一个并返回
讲完了集合的编辑,就不得不提集合的几种运算,即交集、并集、差集。在Python中还多了一种对称差集,也就是集合A与集合B中所有不属于A∩B的元素的集合。例子:
list = set([2,1,10,15,18])#建立集合 list_new = set([10,15,18,99,65])#建立集合 #交集 list.intersection(list_new) list & list_new #并集 list.union(list_new) list | list_new #差集 list.difference(list_new) list - list_new #对称差集 list.symmetric_difference(list_new) list ^ list_new
同样,Python中也预设了一些判断语句,方便判断时使用。例子:
list = set([2,1,10,15,18])#建立集合 list_new = set([10,15,18,99,65])#建立集合 #判断是否为子集 list.issubset(list_new) list_new in list #判断是否为父集 list.issuperset(list_new) #未相交判断 list.isdisjoint(list_new)
文件操作
文件操作是任何一种语言学习中的重点。读写硬盘中的文件一方面是为内存中的程序运行提供必要数据,另一方面也是保存内存中运算结果的重要手段。
在Python中,文件操作主要由三个步骤组成:打开——操作——关闭。但是这里首先讲解文件操作中的打开和关闭。
打开一个文件与关闭这个文件是一一对应的。之要打开,就需要关闭。虽然,在程序运行结束后,会自动从内存中释放打开的文件,但是在整个程序运行的过程中,这个文件都是打开的,这对于数据的安全和运行都是不利的。所以,一定要关闭文件。打开和关闭的方法比较简单。例子:
f = open('test.txt','r')#打开文件,('文件名及绝对或相对路径','操作权限') #操作 f.close()#关闭文件
为了避免忘记关闭文件的不便,鉴于Python缩进编程的优势,还有一种无需写关闭文件的打开方法。例子:
#打开一个文件 with open('test.txt','r') as f: #操作 #后续代码 #打开多个文件 with open('test.txt','r') as f,\ open('test_new','w') as f_new: #操作 #后续代码
在填写路径时,由于Linux用户的路径使用的是正斜杠,'/';而Windows用户的路径使用的是反斜杠,'\'。而反斜杠容易被组合形成转义字符,发生冲突。这里Windows用户也可以一律在路径中填写正斜杠,也会识别。下面就要介绍几种常用的操作权限:
''' 'r' 只读 'w' 只写,若文件存在会覆盖原文件 'a' 添加,在文件后继续添加,不能读 'b' 二进制模式,与r和w组合使用,rb,wb '+' 添加另一种模式,与r和w组合使用,r+,w+,但只会写在最后 'U' #linux与windows的区别,将\r\n自动转换为\n '''
之后,就要介绍文件的操作方法了。首先,要将文件全部读为字符一次性读入内存,可以用这个方法。例子:
with open('test.txt','r') as f: f.readlines()#每行一个元素,全部读入
这种方法只适用于小文件读取。当文件比较大是,如果用这种方法有可能会爆掉内存。这里就要介绍另外一种方法了,及一行一行的读取。文件在刚打开时,文件读取的指针放在了文件初始位置。每读取一次一行,读取指针就会移动到下一行的开头。当读取为空时,就完成了文件的读取。例子:
with open('test.txt','r') as f: f.readline()#读取第一行 f.readline()#读取第二行 f.readline()#读取第三行
当读取到一半时,很难确定此时指针的位置。Python中提供了获取指针位置和移动指针到某一位置的方法。例子:
with open('test.txt','r') as f: f.readline()#读取一行 f.tell()#返回此时指针的位置 f.seekable(0)#将指针移动到括号中的位置,0为开始位置
写入文件的方法比较简单。例子:
with open('test.txt','w') as f: f.write('字符串')#写入文件要想写入后换行可以在字符串结尾加入\n
出了读写外,Python还为文件操作增加了一些其他内容,这里一并介绍。例子:
with open('test.txt','w') as f: f.encoding#文件的编码 f.fileno()#返回操作系统调用文件接口的编号 f.truncate()#清空文件,或进行括号内字符量的截断,与光标指针位置无关 f.flush()#强制刷新
同样,Python也为文件操作提供了一些判断。例子:
with open('test.txt','w') as f: f.seekable()#判断光标能否移动 f.readable()#判断是否可读 f.writable()#判断是否可写 f.closed#判断是否关闭
文件内容的循环。例子:
with open('test.txt','r+') as f: #低效循环 for line in f.readlines(): print(line.strip())#读一行,strip去掉前后空格 for index,line in enumerate(f.readlines()): print(index) print(line.strip()) #高效循环 count = 0 for line in f:#一行行读入内存 print(count) print(line.strip()) count += 1
对于硬盘上的文件修改,也有两种方法。一种是低效方法,即将文件中的内容全部读入内存后,利用正则表达式等方法对其修改后再将其重新写入文件。另一种方法是同时打开两个文件,一个读一个写,然后循环每行读出需要修改的文件,然后逐行修改后分别存入另一个文件。这里不再举例。
字符编码与转码
字符编码有多种多样,如ASCII码、UTF-8、Unicode、GBK、GB2312。字符编码的不同往往会导致程序中出现各种各样的问题。所以学会转码是避免字符编码导致问题的一种手段。
起初美国确立了ASCII码,在这里面每个字母符号均占一个字节。为了显示各国文字,各个国家分别建立了自己的字符编码,如GBK,但是它们之间又不互相兼容。为了解决各国编码不兼容的问题,万国码Unicode应用而生。但是Unicode中所有的符号都占用2个字节,使得英文体积变大。为此,又出现了UTF-8,在UTF-8中,英文占一个字节其他文字占3个字节。Python3.X中的默认编码格式为UTF-8。
Unicode由于其兼容性,成为了各种编码间互相转换的桥梁。UTF-8和Unicode可以直接转换为各国的编码。各国的编码要想互相转换,就需要先解码为Unicode然后由Unicode转换为其他编码。
转换例子:
#-*- coding:UTF-8 -*- #声明文档的编码格式 utf8 = 'hello world!' utf8_gbk = utf8.encode('gbk')#UTF-8转GBK gbk_utf8 = utf8_gbk.decode('gbk').encode('utf-8')#GBK转其他
函数
编写函数可以大大减轻编程中重复代码编写为编程人员带来的麻烦。函数具有:代码可重复利用、可扩展性以及保持一致性的优点。下面将介绍函数的一般形式。例子:
def 函数名(变量名=默认值,…):#变量名与默认值非必须填写 '''函数功能介绍''' 函数体 return 返回值1,返回值2…
当不写return时,函数返回None。调用函数时,括号中的变量应该互相对应。每个变量都应该赋有确定的值。但是当该变量由默认值时,可以省略不写。当依靠位置为变量赋值时,对应位置的值,与付给的值相对应。例子:
def number(x,y,z=0):#z初始值为0 print(x) print(y) print(z) return x+y,x-z number(3,2,1)#此时,x=3,y=2,z=1 number(2,1)#此时不会报错,x=2,y=1,z=0
当使用关键字参数赋值时,可以更改位置。当位置参数与关键字参数混用时关键字参数不能写在位置参数前面,且不能赋给已赋值的位置参数。例子:
def number(x,y,z=0):#z初始值为0 print(x) print(y) print(z) return x+y,x-z number(x=3,2,1)#报错 number(2,x=1)#报错 number(2,z=1,y=3)#不报错,x=2,y=3,z=1
当赋入位置参数个数不确定时,可以将不确定个数个位置参数以元组的形式传入。例子:
def number(*args):#z初始值为0 print(args) return 0 number(1,2,3) number(*[1,2,3])
当赋入关键字参数个数不确定时,可以将不确定个数个关键字参数以字典的形式传入。例子:
def number(**kwargs):#z初始值为0 print(kwargs) return 0 number(age = 12,name = 'kai') number(**{'age':12,'name':'kai'})
终极混乱函数形式。字典形式应放在最后,原组放在字典前。例子:
def number(name,age=12,*args,**kwargs): print(name) print(age) print(args) print(kwargs) return 0 number('kai',22,sex='kai',age=5)#age报错 number('kai',22,33,44,55,sex='men')#name=kai,age=22,args=(33,44,55),kwargs={'sex':'men'}
除了普通的函数,还有一种叫递归函数。递归函数是指函数内部可以调用其他函数,如果调用自己就是递归函数。递归函数必须有明确的结束条件,每进入更深一层,问题规模应该有所减少。
递归函数的运行效率低,并且,程序每调用一个函数,栈中会增加一层栈帧,每返回一个函数,栈中会减少一层栈帧。多层的递归函数容易导致栈溢出。
下面将写一个简单的递归函数。例子:
def number(n): '''判断奇偶''' if n-2>0: return number(n-2) else: return n
高阶函数是接收另一个函数作为参数的函数。下面将写一个简单的高阶函数。例子:
def number(x,y,f):#这里f作为某种函数被当作输入参数 z = f(x)+f(y) return z sum(-3,9,sqrt)#sqrt开方函数
介绍了函数,就不得不讲一下全局变量与局部变量。而这个定义与C++等其他语言中相似,所以不再赘述。在函数中想要修改或定义全局变量需要声明global。但一般情况下不建议这样做。
x = 3 def number(): global x x = 5 return 0 print(x)#x = 3 number() print(x)#x = 5
作业
对一个用户文件实现增删改查
流程图:
主程序:
#-*- coding:UTF-8 -*- #Author:猛男落泪 #配置文件的增删改查 import os #打印标题 title = '欢迎进入用户管理系统' print(title.center(50,'-')) def option_error(): '''输入错误提醒''' print('请输入正确的选项编号!') quit_flag = True while quit_flag: back_flag = True print(''' 1.查询用户 2.增加用户 3.删除用户 4.修改用户 q.退出系统 ''') option = input('请输入所需求的功能:') if option == 'q': #退出 quit_flag = False elif option.isdigit(): option = int(option) if option == 1: #查询系统 while back_flag: info = input('请输入要查询的用户姓名:') if info == 'b': back_flag = False elif info == 'q': back_flag = False quit_flag = False else: with open('user information.txt','r') as f: for line in f: if info in eval(line): print(eval(line)[info]) break else: option_error() elif option == 2: #增加系统 while back_flag: info = input('请输入增加用户的姓名,年龄,性别,职业(空格隔开,b返回q退出):') if info == 'b': back_flag = False elif info == 'q': back_flag = False quit_flag = False else: info = info.split() dic_info = {} dic = {} dic_info['年龄'] = info[1] dic_info['性别'] = info[2] dic_info['职业'] = info[3] dic[info[0]] = dic_info info = str(dic) with open('user information.txt','a') as f: f.write('%s\n' %info) f.flush()#强制刷新 print('%s保存成功!' %info) elif option == 3: #删除系统 while back_flag: info = input('请输入要删除的用户姓名:') if info == 'b': back_flag = False elif info == 'q': back_flag = False quit_flag = False else: f = open('user information.txt','r') f_new = open('user information_new.txt','w') for line in f: if info in eval(line): print('用户删除') break else: option_error() f.seek(0)#光标指针回归 for line in f: if info in eval(line): continue f_new.write('%s' %line) f.close() f_new.close() os.remove('user information.txt') os.rename('user information_new.txt','user information.txt') elif option == 4: #修改系统 while back_flag: info = input('请输入要修改的用户姓名:') if info == 'b': back_flag = False elif info == 'q': back_flag = False quit_flag = False else: f = open('user information.txt','r') f_new = open('user information_new.txt','w') for line in f: if info in eval(line): info_new = input('请输入要修改用户的年龄,性别,职业(空格隔开):') info_new = info_new.split() dic_info = {} dic = {} dic_info['年龄'] = info_new[0] dic_info['性别'] = info_new[1] dic_info['职业'] = info_new[2] dic[info] = dic_info info_new = str(dic) print('%s保存成功!' %info_new) break else: option_error() f.seek(0)#光标指针回归 for line in f: if info in eval(line): f_new.write('%s\n' %info_new) else: f_new.write('%s' %line) f.close() f_new.close() os.remove('user information.txt') os.rename('user information_new.txt','user information.txt') else: option_error() else: option_error() print('感谢你的使用!')
用户信息文件:
user information.txt
{'凯凯王': {'年龄': '17', '性别': '男', '职业': '学生'}} {'卖力头': {'年龄': '16', '性别': '男', '职业': '学生'}} {'飘飘流': {'年龄': '17', '性别': '男', '职业': '学生'}}