• 0
  • 0
分享
  • Python3,这个方法,应该是读取文件天花板了。
  • Carl_奕然 2022-11-24 11:23:00 字数 5805 阅读 1342 收藏 0

1、引言

小鱼:小屌丝, 这段代码为什么要开两个线程?

小屌丝:因为我要读写文件,还要备份文件,所以就开两个线程了。

小鱼:嗯,想法是对的,但是,还有一种简便的方法, 不需要开两个线程就能搞得定的。

小屌丝:额…难道是with open?

小鱼:不是。

小屌丝:那还有啥呢? 我咋想不起来了。

小鱼:嗯,这个方法很奈斯,但是很少人使用,因为大部分码农都是直接open 文件的。

小屌丝:那你就别藏着掖着了,赶紧展示一下,让我也长长见识。

2、 fileinput

说到fileinput,可能90%的码农表示没用过,甚至没有听说过。

这不奇怪,因为在python界,既然open可以走天下,何必要fileinput呢?。

但是,今天小鱼还是要介绍fileinput这个方法,因为太奈斯了。

不止是香。是真香!

接下来,就跟着小鱼,一起fileinput,对,就是这个feel。

2.1 方法介绍

2.1.1 基本用法

先来看一下fileinput的基本功能:

    fileinput.filename():返回当前被读取的文件名。

        —>在第一行被读取之前,返回 None。  

    fileinput.fileno():返回以整数表示的当前文件“文件描述符”。

        —>当未打开文件时(处在第一行和文件之间),返回 -1。   

    fileinput.lineno():返回已被读取的累计行号。

        —>在第一行被读取之前,返回 0。在最后一个文件的最后一行被读取之后,返回该行的行号。

    fileinput.filelineno():返回当前文件中的行号。

        —>在第一行被读取之前,返回 0。

        —>在最后一个文件的最后一行被读取之后,返回此文件中该行的行号。

2.1.2 进阶用法

fileinput.isfirstline():如果刚读取的行是其所在文件的第一行则返回 True,否则返回 False。

fileinput.isstdin():如果最后读取的行来自 sys.stdin 则返回 True,否则返回 False。

fileinput.nextfile():关闭当前文件以使下次迭代将从下一个文件(如果存在)读取第一行;不是从该文件读取的行将不会被计入累计行数。直到下一个文件的第一行被读取之后文件名才会改变。

    —>在第一行被读取之前,此函数将不会生效;它不能被用来跳过第一个文件。

    —>在最后一个文件的最后一行被读取之后,此函数将不再生效。

fileinput.close():关闭序列。

2.2 默认读取

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : Carl_DJ
import fileinput
'当 Python 脚本没有传入任何参数时,fileinput 默认会以 stdin 作为输入源'
for line in fileinput.input():
    print(f'{line}')

运行结果

cc1312950dae4ba8bb163cbe88e76508.png

你输入的内容,程序都会读取并再输出。

俗称:复读机

2.3 处理一个文件

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
'files 输入打开文件的名称即可'
with fileinput.input(files=('output.txt',)) as file:
    for line in file:
        print(f'{fileinput.filename()} 第{fileinput.lineno()}行:{line}',end='')

行结果

2fe87b81be1f4e108719e874abab34f2.png

解析:

    fileinput 有且仅有这两种读取模式:‘r’,‘rb’;

    fileinput.input() 默认使用 mode=‘r’ 的模式读取文件,如果你的文件是二进制的,可以使用mode=‘rb’ 模式。

2.4 处理批量文件

2.4.1 多文件序号连续排序

调用方法

    fileinput.lineno()方法

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
'files 输入打开文件的名称即可'
with fileinput.input(files=('output.txt','input.txt')) as file:
    for line in file:
        #fileinput.lineno() 把两个文件的整合陈一个文件对象file,需要排序输出
        print(f'{fileinput.filename()} 第{fileinput.lineno()}行: {line}', end='')
        
        # fileinput.filelineno()两个文件单独读取,需要单独排序
        print(f'{fileinput.filename()} 第{fileinput.filelineno()}行: {line}', end='')

运行结果

f15838141b964610960ebee96f2604fa.png

2.4.2 多文件序号单独排序

调用方法

    fileinput.filelineno()方法

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
'files 输入打开文件的名称即可'
with fileinput.input(files=('test1.txt','test2.txt')) as file:
    for line in file:       
        # fileinput.filelineno()两个文件单独读取,需要单独排序
        print(f'{fileinput.filename()} 第{fileinput.filelineno()}行: {line}', end='')

运行结果

7bbd10d92c4c4b418a2129b393e75ba4.png

2.4.3 与glob配合用法

在颜值的时代,上面的输出样式,已经无法满足我们的需要了,

于是乎,我们就想到了glob。

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
import glob
#glob 匹配te开头的txt文件
for line in fileinput.input(glob.glob("te*.txt")):
    if fileinput.isfirstline():
        #输出读取文件
        print('='*10,f'读取文件{fileinput.filename()}','='*10)
        #fileinput.filelineno()方法读取
    print(str(fileinput.filelineno())+ ':'+line.upper(),end='')

运行结果

ccb72458c3fe459dbef8f4a4d018c9f1.png

就这颜值,哪个小姐姐能不喜欢呢。

2.5 读取与备份

调用方法

    fileinput.input 的backup 参数,可以指定备份的后缀名,比如 .bak

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
#触发backup的动作,源文件内容被修改,对源文件进行backup
with fileinput.input(files=("test1.txt",), backup=".bak",inplace=1) as file:
    for line in file:
        print(line.rstrip().replace('111111', '222222'))
        print(f'{fileinput.filename()} 第{fileinput.lineno()}行: {line}', end='')


运行结果2452e3809ae44f0d8b393b30ce960fe1.png

2.5 重定向替换

解析

    上面的例子, 用到了 inplace参数,表示是否将标准输出的结果写回文件,默认不取代

代码示例

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
#触发backup的动作,源文件内容被修改,对源文件进行backup
with fileinput.input(files=("test2.txt",), inplace=True) as file:
    print("[INFO] task is started...")
    for line in file:
        print(f'{fileinput.filename()} 第{fileinput.lineno()}行: {line}', end='')
    print("[INFO] task is closed...")

运行结果

112220390ec54c21a8021283da0f0eea.png

通过运行结果,可以看到:

    - 在 for 循环体内的 print 内容会写回到原文件中了。

    - 而在 for 循环体外的 print 则没有变化。

2.6 进阶

2.6.1 openhook含义解析

在 fileinput.input() 中有一个 openhook 的参数,它支持用户传入自定义的对象读取方法;

    -->如果没有传入任何勾子,fileinput 默认使用的是 open 函数;

2.6.2 方法介绍

fileinput 内置了两种:

1、fileinput.hook_compressed(filename, mode)

   使用 gzip 和 bz2 模块透明地打开 gzip 和 bzip2 压缩的文件(通过扩展名 ‘.gz’ 和 ‘.bz2’ 来识别);

   如果文件扩展名不是 ‘.gz’ 或 ‘.bz2’,文件会以正常方式打开(即使用 open() 并且不带任何解压操作);

    使用示例: fi = fileinput.FileInput(openhook=fileinput.hook_compressed)

2、fileinput.hook_encoded(encoding, errors=None)

    返回一个通过 open() 打开每个文件的钩子,使用给定的 encoding 和 errors 来读取文件。

    使用示例: fi = fileinput.FileInput(openhook=fileinput.hook_encoded(“utf-8”, “surrogateescape”))

2.6.3 示例实战

假如我想要使用 fileinput 来读取网络上的文件,思路:

先使用 requests 下载文件到本地

再使用 open 去读取它;

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
def online_open(url, mode):
    import requests
    r = requests.get(url) 
    filename = url.split("/")[-1]
    with open(filename,'w') as f1:
        f1.write(r.content.decode("utf-8"))
    f2 = open(filename,'r')
    return f2

直接将这个函数传给 openhook 即可:

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
import fileinput
file_url = 'https://www.csdn.net/robots.txt'
with fileinput.input(files=(file_url,), openhook=online_open) as file:
    for line in file:
        print(line, end="")

代码整合:

# -*- coding:utf-8 -*-
# @Time   : 2022-07-23
# @Author : carl_DJ
def online_open(url, mode):
    import requests
    r = requests.get(url)
    filename = url.split("/")[-1]
    with open(filename,'w') as f1:
        f1.write(r.content.decode("utf-8"))
    f2 = open(filename,'r')
    return f2
import fileinput
file_url = 'https://www.csdn.net/robots.txt'
with fileinput.input(files=(file_url,), openhook=online_open) as file:
    for line in file:
        print(line, end="")

运行结果

7faa48d8d4cb46eba3ff9a50aff9f10c.png

3、总结

看到这里,今天的分享差不多就要结束了。

关于fileinput的介绍,也就介绍到这里。

fileinput本身是对 open 函数的再次封装,所以在读取的部分,就比open显得更专业,更优雅,这也是仅限于读取的方面。

在写的方面,相对于open,就不是那么的强悍。

归根结底,fileinput还是一个不错的方法。值得你拥有。


最后,再唠叨一句:

我是奕然, 

    关注我,带你学习更多更有趣的python知识。


  • 【留下美好印记】
    赞赏支持
登录 后发表评论
+ 关注

热门文章

    最新讲堂

      • 推荐阅读
      • 换一换
          •   背景  随着分布式数据库的日渐成熟,在金融行业逐渐推行分布式数据库的使用,如何验证分布式数据库的高可用性是应用方所关注的。  本文针对主流的TDSQL分布式数据库,在测试环境模拟真实业务持续压测,通过人为制造数据节点故障,观测业务具体表现和赤免监控指标得出RTO数值。  相关概念  RTO:恢复时间目标,主要指的是所能容忍的业务停止服务的最长时间,也就是从灾难发生到业务系统恢复服务功能所需要的最短时间周期。  数据库恢复时间,指数据库停止对外服务到重新提供服务的时间。  Xmeter:一种性能测试发压工具,可以高效的模拟客户端发起高并发请求,同时统计测试结果。  分片:是把数据库横向扩展到...
            0 0 1283
            分享
          •   如果问软件测试有哪些前景,准确的说,应该是软件测试工程师的职业规划是什么样的。关于一些职业规划上的意见建议,我从下面几个方面梳理一下,供您参考。  1、软件测试行业的现状  2、目前业内人员的现状  3、软件测试工程师的职业规划  4、软件测试工程师的核心竞争力在哪  5、软件测试工程师如何不断提高自己的能力  大厂更加倾向于招收测试开发,更多的初级测试被外包吸收,只会手工测试的人员生存更加艰难。  在经历了18年的裁员潮之后,很多大厂及创业公司都在调整和优化自己的团队结构,特别是对于大厂来说,并不是简单地裁人,而是裁掉单一的手工测试,转而付出更高的薪酬来招收技术更好的测试开发人员。在这个...
            14 14 2222
            分享
          • 案例设计需求有一个ATM取款系统,现对于取款功能进行了需求变更:只能取面额是100元(如取500,输出5张100元)。现在功能修改为,可以取面额是10元、50元和100元的,其余功能不变,用户界面也没有任何变化,取款原则为“最优吐钞法”,有大额先吐大额,请根据需求变更进行案例设计。参考测试用例大额优先刚好整百的金额,吐出N张100元的刚好为50元,吐出1张50元刚好是50以内的整10元,吐出N张10元取款金额是N百,超过50元,吐出N张100元,1张50元,(M-5)张10元(例如380元,会吐出3张100元,1张50元,8-5张10元)如果是几十元,同时超过50元。那么会吐出1张50元,M-...
            0 0 1124
            分享
          • 几年的测试工作下来,除了日常的功能特性的测试,还完成了自己负责APP相关测试体系从零到一的建设,今天借由这个机会,做一个简单的梳理,将在这个过程中自己的一些思考,踩过的坑等做一个整理,分享给各位供参考。1.自动化测试自动化测试主要包括几个部分,UI功能的自动化测试、接口的自动化测试、其他专项的自动化测试。1.1UI功能自动化测试UI功能的自动化测试,也就是大家常说的自动化测试,主要是基于UI界面进行的自动化测试,通过脚本实现UI功能的点击,替代人工进行自动化测试。这个测试的优势在于对高度重复的界面特性功能测试的测试人力进行有效的释放,利用脚本的执行,实现功能的快速高效回归。但这种测试的不足之处...
            1 1 2770
            分享
          • 2018年最受人追捧的技术无疑是:人工智能,大数据,云计算,区块链。前三者对于IT圈内外的人还能知道是什么,能够运用到实际生活中哪些地方,而最后一个区块链技术,却是连大多数的程序员都说不清其概念,更不要提具体的应用场景,唯一能说出和区块链挂钩的名词也就是比特币等金融方面,那么区块链的深度就仅限于此吗?非也,本文就由易大师从目前投资最火热比特币等虚拟货币作为一个切入点,给大家讲讲区块链到底是什么?比特币又是什么? 先从大家最熟悉的比特币开始说起,自2008年金融危机后,全球人民都不再相信美国政府的公信力,由代号中本聪的存在创建比特币白皮书《比特币:一种点对点的电子现金系统》首次提出区块链...
            15 8 12436
            分享
      • 51testing软件测试圈微信