• 0
  • 0
分享
  • 天花板级别的Python读取文件的方法——软件测试圈
  • 曼倩诙谐 2023-03-08 10:50:25 字数 5278 阅读 856 收藏 0

  序言

  哈喽兄弟们,今天咱们来了解一下 fileinput 。

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

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

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

  不止是香。是真香!

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

  正文

  1、方法介绍

  基本用法

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

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

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

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

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

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

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

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

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

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

  进阶用法

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

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

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

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

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

  fileinput.close():关闭序列。

  2、 默认读取

  代码示例:

  import fileinput
  '当 Python 脚本没有传入任何参数时,fileinput 默认会以 stdin 作为输入源'
  for line in fileinput.input():
      print(f'{line}')

  运行结果:

1-1.png


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

  俗称:复读机

  3、处理一个文件

  代码示例:

  import fileinput
  'files 输入打开文件的名称即可'
  with fileinput.input(files=('output.txt',)) as file:
      for line in file:
          print(f'{fileinput.filename()} 第{fileinput.lineno()}行:{line}',end='')

  运行结果:

1-2.png

  解析:

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

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

  4、处理批量文件

  多文件序号连续排序

  调用方法

  ·fileinput.lineno()方法

  代码示例:

  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='')

  运行结果:

1-3.png

  多文件序号单独排序

  调用方法

  ·fileinput.filelineno()方法

  代码示例:

  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='')

  运行结果:

1-4.png

  与glob配合用法

  在颜值的时代,上面的输出样式,已经无法满足我们的需要了,于是乎,我们就想到了glob。

  代码示例:

  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='')

  运行结果:

1-5.png

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

  5、读取与备份

  调用方法

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

  代码示例:

  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='')

  运行结果:

1-6.png

  6、重定向替换

  解析

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

  代码示例:

  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...")

  运行结果:

1-7.png

  注

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

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

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

  7、进阶

  openhook含义解析

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

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

  方法介绍

  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”))

  示例实战

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

  先使用 requests 下载文件到本地;

  再使用 open 去读取它。

  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="")

  运行结果:

1-8.png

  总结

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

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

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

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


作者:佚名    

来源:http://www.51testing.com/html/50/n-7794050.html

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

热门文章

    最新讲堂

      • 推荐阅读
      • 换一换
          •   前言  UI测试是测试应用中的各种交互是否达到了实现的效果。常用的UI测试框架有Espresso和UIAutomator。  今天给大家分享5个可能不曾听过的新自动化测试框架。  1.Kaspresso  Kaspresso是一个基于Espresso和UIAutomator构建的测试自动化框架。并针对Espresso的一些已知缺点进行优化:  ·解决flakiness问题;  · 解决Espresso不支持adb问题;  · 优化代码可读性;  以如下示例说明代码可读性:  Espresso测试示例写法:  @Testfunlogout(){   onView(with...
            0 0 1649
            分享
          •   想做程序员,想学软件测试,是自学还是培训好?学习的关键是什么呢?  大家纠结的点在于,我是应该去参加培训班还是选择自学,我自身的话没有参加过培训班,所以我觉得没有什么发言权来给大家什么建议。不过我后来去咨询了几个朋友,有一个是在培训机构当老师,还有两个曾经也接受过培训。我根据他们说的,还有结合我自己的一些想法来给大家做个分享。当然如果你有什么补充的,可以在评论区留言!  关于自学还是选择培训有几点——  第一:  你是零基础的小白,自学约等于浪费时间。这个公式在多数情况下都成立,当然,如果你有自信的话可以去试一下。  第二:  如果你有一到两年的开发/测试的经验,想在技术上有所提升,那这个...
            0 0 764
            分享
          •   这篇文章会产生,是因为刚好有一个项目,同时用到了这两款工具,而正是工具的使用让我解放了双手,感受到了什么是高效、智能化。  故事背景  某个功能模块的测试数据无法通过界面操作,为了模拟出真实的测试场景,开发人员提供了一个内部的接口文档,希望通过接口调用复现真实使用环境。  这里我大概讲一下测试流程及顺序,首先普及一下什么是Swagger UI。  Swagger UI  Swagger是一组开源项目,接口的文档可以在线自动生成,不需要频繁更新接口文档,保证接口文档与代码的一致性。  这里的实时更新是它的一大优点。试想一下,有些公司基本上使用word维护接口文档,假如出现涉及到接口变更情况,...
            12 13 3209
            分享
          • 读者提问:『作为测试经理,需要做测试执行吗,如果需要的话,还有时间做管理相关的事务吗 ?』阿常回答:测试经理也是需要做测试执行的。因为对于软件测试管理者来说,管理不会是纯粹的管理。软件测试是一门需要不断实践和总结的学科,所以身为测试经理,也是会参与到实际的测试任务之中的。那如果既需要做事,又需要做管理,会不会因为陷入具体的测试任务之中,而没办法做好管理事务呢?作为测试管理者,不应该让自己投入到,非常复杂的需要投入大量时间的测试任务之中,如果是这样的话,肯定没有时间兼顾到管理事务。其他测试同学都投入到具体的测试任务之中,陷入到细节之中,测试管理者必须把控好全局,所以测试管理者可以选择一...
            0 0 423
            分享
          • 1、添加线程组2、配置http信息头管理;使用json格式传递数据时,必须配置此项;3、添加httpcookie管理器,根据具体情况配置cookie信息,本例中未配置4、配置http请求默认值;同一个项目的接口请求放在一个线程组中,可配置请求默认值,这样无需在各接口请求中单独配置;5、配置http请求,就是具体请求的接口;本例中参数格式为json格式,使用bodydata;6、添加查看结果树,可查看接口执行情况整体的可看聚合报告Label:每个JMeter的element(例如HTTPRequest)都有一个Name属性,这里显示的就是Name属性的值#Samples:表示你这次测试中一共发出...
            13 13 1737
            分享
      • 51testing软件测试圈微信