玩游戏的话, 如果没有存盘和读取, 简直就没法玩了. 写程序也差不多, 除了要实时保存自己写好的代码, 还需要能够保存和读取数据. 比如做个临床研究, 不大可能手工一个一个按照numpy array的格式录入眼轴长L, 角膜曲率K之类的数据. 这类的数据往往是由机器导出的, 或者由哪个可怜的研究生去病案室抄出来然后存成一个文件的.
这个文件很有可能是excel的. 所以, 这节课我们就以excel为例子读取一组病人的数据, 然后计算IOL度数, 再写入到excel里面.
欢迎来到现实世界. 前面几课我们面对的都是非常理想化的环境, 我们自己设定什么, 计算机就可以按照我们设想的去做. 但当涉及到文件操作的时候, 就不一定了. 比如, 对于桌面电脑, 可能主流有三类操作系统: Windows, macOS, Linux, 大多数非程序员是用Windows, 但也有相当多的眼科医生用macOS. 在程序员的世界里, 比如我推荐使用的CoCalc或者Azure Notebooks, 其实后台的操作系统是Linux的.
如果你的文件IOLdata.xlsx存储在一个叫做data的文件夹下面, 那么, 问题出现了. 文件名在不同的系统写法会有一点细微的差别:
斜杠左右不同, 所以在读取文件的时候, 你不能简单指定文件名是'data\IOLdata.xlsx' 还是 'data/IOLdata.xlsx', 因为你并不知道用程序的人用着什么样的电脑.
萨特说 ** 他人即地狱 **
所以我们需要用os库中的os.path.join函数来把文件的路径和文件名按照当前运行时的操作系统要求组合起来.
import os
pathname='data'
fname='IOLdata.xlsx'
print(os.path.join(pathname,fname))
data/IOLdata.xlsx
上面的输出结果在不同的电脑上看起来可能会是不同的, 但如果你使用的是CoCalc或者Azure Notebooks所提供的在线服务, 那么应该显示的是data/IOLdata.xlsx
你的合作伙伴很有可能给你的多个文件, 而不仅仅是一个, 当然你可以在excel中依次把它们打开, 然后复制粘贴到一起, (记得把第二个以后文件的标题行删掉). 但如果能够直接用程序读取并且合并不是更好么.
下面, 我要讲解一个之前刻意跳过的内容: List
其实也没什么神秘的, 就是用方括号[ ]装起来的一组东西, 这组东西必须是一个类型的, 比如都是数字, 或者都是字符. 如果要访问第0个元素, 就用 list名[0], 第1个元素就用 list名[1]. ** 注意python是从0开始计数的. **
filenames=['IOLdata00.xlsx','IOLdata01.xlsx']
print(filenames)
print(filenames[0])
print(filenames[1])
['IOLdata00.xlsx', 'IOLdata01.xlsx'] IOLdata00.xlsx IOLdata01.xlsx
试试用os.path.join产生一组带有路径的文件名?
os.path.join(pathname,filenames)
--------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-29-ecdbda71648b> in <module>() ----> 1 os.path.join(pathname,filenames) ~/anaconda3/envs/opencvenv/lib/python3.5/posixpath.py in join(a, *p) 87 path += sep + b 88 except (TypeError, AttributeError, BytesWarning): ---> 89 genericpath._check_arg_types('join', a, *p) 90 raise 91 return path ~/anaconda3/envs/opencvenv/lib/python3.5/genericpath.py in _check_arg_types(funcname, *args) 141 else: 142 raise TypeError('%s() argument must be str or bytes, not %r' % --> 143 (funcname, s.__class__.__name__)) from None 144 if hasstr and hasbytes: 145 raise TypeError("Can't mix strings and bytes in path components") from None TypeError: join() argument must be str or bytes, not 'list'
看到出错了吧, 想想你对os.path.join()的要求也太高了, 它怎么知道pathname要跟list里面的每一个元素依次join呢
这也是我之前刻意逃避的一个内容, 如果是处理数字, 我推荐尽量避免使用循环, 而直接用向量来处理. 但现在要处理字符串的部分, 可能还是需要介绍一下循环的使用
filename_list=[]
for f in filenames:
filename_list.append(os.path.join(pathname,f))
print(filename_list)
['data/IOLdata00.xlsx', 'data/IOLdata01.xlsx']
上面这一段代码中, 我先建立了一个空的list
filename_list=[]
然后用list的一个功能 .append()往里添加元素
filename_list.append(xxxx)
顺便说, x.y()这样的形式, 也是函数, 只不过.y()是x自带的函数, 如果需要知道x都自带了哪些函数, 可以用dir(x)这样的方式查询, 如果需要了解具体那个函数怎么用, 则可以用help. 当然更优选的方案是面向google/stackoverflow的编程.
dir(filename_list)
help(filename_list.append)
上面的代码中, 还有
for f in filenames:
filename_list.append(os.path.join(pathname,f))
这就是一个for循环了. 循环变量f会依次跑遍filenames这个list里面的每一个元素, 然后f携带着list中的元素依次代入到循环体中参与工作.
注意格式,
for循环最常见的例子, 恐怕就是从1一直加到100了
import numpy as np
s=0
for i in np.arange(1,100+1): # 注意arange也是<, 所以要加到100, arange要到101
s=s+i
print(s)
5050
# 我更喜欢这样的方式, 运算速度上会有一点差别, 但这个加法太简单, 不明显
s=np.sum(np.arange(1,101))
print(s)
5050
关于循环, 还有下面这样的形式,
f_list=[os.path.join(pathname,f) for f in filenames]
这句话和
filename_list=[]
for f in filenames:
filename_list.append(os.path.join(pathname,f))
是完全一致的. 写法上简单了很多:
f_list=[os.path.join(pathname,f) for f in filenames]
print(f_list)
['data/IOLdata00.xlsx', 'data/IOLdata01.xlsx']
这个叫做list comprehension, 如果看python程序员写的代码, 会有非常多这样的结构出现, 初学者可能认为这样的方式可读性差一些, 属于奇技淫巧, 但由于使用的场景非常多, 所以呈现特殊形式也不难理解. 就像英文里复数并不总加s, 过去式并不总是ed, 而且越是常用的东西往往越不按照语法规则来.
终于收拾好了文件名, 接下来我们可以读取excel文件了
Pandas是一个python的数据处理库, 类似于numpy是一个科学计算库一样. 很多跟数据处理相关的工作需要用pandas来完成, 这堂课里我们只需要使用到跟excel读写相关的内容, 但pandas能做得东西更多, 具体的请参考Python for Data Analysis这本书, 这本书有免费的在线版本, 也有中文版<利用Python进行数据分析> (居然还是kindle unlimited可免费借阅的, 总算让我觉得没白买).
使用pandas, 也要先import, 一般把pandas简称为pd, 所以就是
import pandas as pd
(如果在本地电脑上安装pandas, 要小心里面还有两个坑, excel的读取和写入要借助另一组包xlrd和openpyxl, 需要安装他们, 否则报错. 我而anaconda居然没有自动解决这个问题. 不过cocalc和azure notebooks里面没问题)
读取excel文件就是用pd.read_excel(文件名), 读取的结果是一种pandas的叫做DataFrame的数据结构, 看起来和带有索引的表格很接近.
testdata=pd.read_excel(f_list[1])
print(testdata)
ID name A AL K1 K2 Ref 0 21010 Bob00 119 21.7 41.3 40.3 -3 1 21011 Bob01 119 20.7 39.1 40.9 -3 2 21012 Bob02 119 24.7 40.8 40.1 -3 3 21013 Bob03 119 19.3 39.8 42.4 -3 4 21014 Bob04 119 20.7 40.6 39.1 -3 5 21015 Bob05 119 22.0 42.2 42.6 -3 6 21016 Bob06 119 19.3 40.0 39.7 -3 7 21017 Bob07 119 25.5 41.0 39.3 -3 8 21018 Bob08 119 21.3 40.6 39.4 -3
如果要取某一列的数据, 只需要用 .列名来调用, 如果要调用一列数据中的具体某个数据, 就用方括号括上序号即可.
数据列和numpy的列向量在某种程度上是互通的.
print(testdata.AL)
print(testdata.K1[1])
0 21.7 1 20.7 2 24.7 3 19.3 4 20.7 5 22.0 6 19.3 7 25.5 8 21.3 Name: AL, dtype: float64 39.1
IOLdata_list=[pd.read_excel(f) for f in f_list]
合并数据表有多种方法, 在这里我使用的是pd.concat, 它能够把列表中的多个DataFrame合并成一个, 其中ignore_index=True是将索引序号给忽略重排了, 否则会出现两组1号, 两组2号. . . 关于pandas.concat的详细说明请Google查询.
IOLdata=pd.concat(IOLdata_list,ignore_index=True) # 这里可以试试把True改成False
print(IOLdata[17:21])
ID name A AL K1 K2 Ref 17 11027 Sam17 118.4 24.6 42.6 39.5 0.0 18 11028 Sam18 118.4 19.6 41.7 40.8 0.0 19 21010 Bob00 119.0 21.7 41.3 40.3 -3.0 20 21011 Bob01 119.0 20.7 39.1 40.9 -3.0
读取了数据以后, 就可以直接当作列向量丢进函数里面了. 重写一下上节课的SRK_2函数. 如果上节课作业没有写的同学, 请不要看下面的剧透, 先回去把上节课作业写完.
def on_L_change_A(L,A,Lmin,Lmax,deltaA):
#A=np.asarray(A).copy()
#L=np.asarray(L).copy()
assert A.shape()==L.shape()
pickout=np.logical_and(L>Lmin, L<=Lmax)
A[pickout] += deltaA
return A
def SRK_2(A,K_1,K_2,L,REF=0):
A = np.asarray(A).copy() # 避免pandas修改原始数据, 还有更好的方案么?
A = on_L_change_A(L,A,0, 20, 3)
A = on_L_change_A(L,A,20, 21, 2)
A = on_L_change_A(L,A,21, 22, 1)
A = on_L_change_A(L,A,22, 24.5, 0)
A = on_L_change_A(L,A,24.5, 50, -0.5)
K = (K_1+K_2)/2
P_emme= A - 0.9*K -2.5*L
CR = np.ones(P_emme.shape)
CR[ P_emme>=14 ]=1.25
P_ammc=P_emme-REF*CR
return P_ammc
前面说过, 就是用 .列名, 就可以调用已经读入的excel数据了, 而且在使用上和numpy array没有区别. 所以就可以直接送入到已经写好的函数中.
IOLpower=SRK_2(IOLdata.A, IOLdata.K1,IOLdata.K2,IOLdata.AL)
print(IOLpower[0:5])
IOLpower=pd.DataFrame(IOLpower)
0 25.590 1 23.235 2 36.000 3 19.465 4 26.205 dtype: float64
计算完成的结果, 应该存储下来, 供以后处理, 只保存计算好的IOL度数似乎没什么意义, 因为看不出与其他信息之间的关系了. 最简单的方法就是在原来的表后面加上一列计算结果, 然后保存.
之前在合并多个excel表格的时候见过了pd.concat, 此处我们使用DataFrame的 .join()函数来给原来的IOLdata后面添加一列. (这里也可以用pd.concat)
newIOLdata=IOLdata.join(IOLpower)
print(newIOLdata)
ID name A AL K1 K2 Ref 0 0 11010 Sam00 118.4 22.4 42.3 39.5 -0.5 25.590 1 11011 Sam01 118.4 23.0 42.4 41.3 -0.5 23.235 2 11012 Sam02 118.4 19.4 42.7 39.3 -0.5 36.000 3 11013 Sam03 118.4 25.1 39.5 39.8 -0.5 19.465 4 11014 Sam04 118.4 22.1 41.5 40.6 -0.5 26.205 5 11015 Sam05 118.4 24.0 42.2 41.2 -0.5 20.870 6 11016 Sam06 118.4 22.6 41.3 40.5 -0.5 25.090 7 11017 Sam07 118.4 21.6 41.7 39.2 -3.0 28.995 8 11018 Sam08 118.4 24.5 40.5 42.7 -3.0 19.710 9 11019 Sam09 118.4 19.0 41.6 41.8 -3.0 36.370 10 11020 Sam10 118.4 24.4 39.0 40.4 -3.0 21.670 11 11021 Sam11 118.4 19.0 42.4 39.3 -3.0 37.135 12 11022 Sam12 118.4 22.0 39.8 42.9 -3.0 27.185 13 11023 Sam13 118.4 24.1 39.7 39.8 0.0 22.375 14 11024 Sam14 118.4 23.2 39.8 42.2 0.0 23.500 15 11025 Sam15 118.4 20.0 40.4 40.9 0.0 34.815 16 11026 Sam16 118.4 25.5 41.2 41.3 0.0 17.025 17 11027 Sam17 118.4 24.6 42.6 39.5 0.0 19.455 18 11028 Sam18 118.4 19.6 41.7 40.8 0.0 35.275 19 21010 Bob00 119.0 21.7 41.3 40.3 -3.0 29.030 20 21011 Bob01 119.0 20.7 39.1 40.9 -3.0 33.250 21 21012 Bob02 119.0 24.7 40.8 40.1 -3.0 20.345 22 21013 Bob03 119.0 19.3 39.8 42.4 -3.0 36.760 23 21014 Bob04 119.0 20.7 40.6 39.1 -3.0 33.385 24 21015 Bob05 119.0 22.0 42.2 42.6 -3.0 26.840 25 21016 Bob06 119.0 19.3 40.0 39.7 -3.0 37.885 26 21017 Bob07 119.0 25.5 41.0 39.3 -3.0 18.615 27 21018 Bob08 119.0 21.3 40.6 39.4 -3.0 30.750
最后就是写入excel文件了. 动作与读取类似, 但这个 .to_excel(文件名)的函数是直接由DataFrame数据自带的函数, 所以用如下形式调用
# 要确保装入了openpyxl
newIOLdata.to_excel(os.path.join(pathname,'output.xlsx'))
用Linux命令列一下data文件夹下的文件名, 确认是否新产生了'output.xlsx'的文件
!ls data
IOLdata00.xlsx IOLdata01.xlsx output.xlsx
如果在cocalc或者azure notebooks上, 就可以在文件中点击下载看看了