dart系列之:如丝滑般柔顺,操作文件和目录
简介
文件操作是IO中非常常见的一种操作,那么对应dart语言来说,操作文件是不是很简单呢?实际上dart提供了两种读取文件的方式,一种是一次性全部读取,一种是将文件读取为流。
一次性读取的缺点是需要将文件内容一次性全部载入到内存中,如果遇到文件比较大的情况,就会比较尴尬。所以还需要流式读取文件的方式。一起来看看dart中这两种文件的读取方式吧。
File
事实上dart中有很多地方都有File这个类,这里我们要讲解的File类是dart:io包中的。
读取整个文件
File代表一个整体的文件,他有三个构造函数,分别是:
factory File(String path) factory File.fromUri(Uri uri) factory File.fromRawPath(Uint8List rawPath)
其中最常用的就是第一个构造函数。
我们可以这样来构造一个文件:
var file = File('file.txt');
有了文件之后,就可以调用File中的各种读取方法。
文件读取本身有两种形式,一种是文本,一种是二进制。
如果是文本文件,File提供了readAsString的方法,将整个文件读取为字符串。
Future<String> readAsString({Encoding encoding: utf8});
我们可以这样使用:
var stringContents = await file.readAsString();
另外,我们还可以一行一行的对文件进行读取:
Future<List<String>> readAsLines({Encoding encoding: utf8});
结果返回的是一个List,list中表示文件每行的内容。
var lines = await file.readAsLines();
上面两个方法是异步的方法,File还提供了两个同步的方法:
String readAsStringSync({Encoding encoding: utf8}); List<String> readAsLinesSync({Encoding encoding: utf8});
如果文件是二进制,那么可以使用readAsBytes或者同步的方法readAsBytesSync:
Future<Uint8List> readAsBytes(); Uint8List readAsBytesSync();
dart中表示二进制有一个专门的类型叫做Uint8List,他实际上表示的是一个int的List。
还是刚刚的文件,我们看下怎么以二进制的形式进行读取:
var file = File('file.txt'); var contents = await file.readAsBytes();
以流的形式读取文件
上面我们讲到的读取方式,都是一次性读取整个文件,缺点就是如果文件太大的话,可能造成内存空间的压力。
所以File为我们提供了另外一种读取文件的方法,流的形式来读取文件.
相应的定义方法如下:
Stream<List<int>> openRead([int? start, int? end]);
我们看一个基本的使用:
import 'dart:io'; import 'dart:convert'; Future<void> main() async { var file = File('file.txt'); Stream<List<int>> inputStream = file.openRead(); var lines = utf8.decoder .bind(inputStream) .transform(const LineSplitter()); try { await for (final line in lines) { print('Got ${line.length} characters from stream'); } print('file is now closed'); } catch (e) { print(e); } }
随机访问
一般情况下文件是顺序访问的,但是有时候我们需要跳过某些前面的数据,直接跳转到目标地址,则需要对文件进行随机访问。
dart提供了open和openSync两个方法来进行随机文件读写:
Future<RandomAccessFile> open({FileMode mode: FileMode.read}); RandomAccessFile openSync({FileMode mode: FileMode.read});
RandomAccessFile提供了对文件的随机读写方法。非常好用。
文件的写入
写入和文件读取一样,可以一次性写入或者获得一个写入句柄,然后再写入。
一次性写入的方法有四种,分别对应字符串和二进制:
Future<File> writeAsBytes(List<int> bytes, {FileMode mode: FileMode.write, bool flush: false}); void writeAsBytesSync(List<int> bytes, {FileMode mode: FileMode.write, bool flush: false}); Future<File> writeAsString(String contents, {FileMode mode: FileMode.write, Encoding encoding: utf8, bool flush: false}); void writeAsStringSync(String contents, {FileMode mode: FileMode.write, Encoding encoding: utf8, bool flush: false});
句柄形式可以调用openWrite方法,返回一个IOSink对象,然后通过这个对象进行写入:
IOSink openWrite({FileMode mode: FileMode.write, Encoding encoding: utf8});
var logFile = File('log.txt'); var sink = logFile.openWrite(); sink.write('FILE ACCESSED ${DateTime.now()}\n'); await sink.flush(); await sink.close();
默认情况下写入是会覆盖整个文件的,但是可以通过下面的方式来更改写入模式:
var sink = logFile.openWrite(mode: FileMode.append);
处理异常
虽然dart中所有的异常都是运行时异常,但是和java一样,要想手动处理文件读写中的异常,则可以使用try,catch:
Future<void> main() async { var config = File('config.txt'); try { var contents = await config.readAsString(); print(contents); } catch (e) { print(e); } }
总结
以上就是dart中的文件操作了。
本文已收录于 http://www.flydean.com/23-dart-file/
最通俗的解读,最深刻的干货,最简洁的教程,众多你不知道的小技巧等你来发现!
欢迎关注我的公众号:「程序那些事」,懂技术,更懂你!

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Kubernetes 集群无损升级实践
一、背景 活跃的社区和广大的用户群,使 Kubernetes 仍然保持3个月一个版本的高频发布节奏。高频的版本发布带来了更多的新功能落地和 bug 及时修复,但是线上环境业务长期运行,任何变更出错都可能带来巨大的经济损失,升级对企业来说相对吃力,紧跟社区更是几乎不可能,因此高频发布和稳定生产之间的矛盾需要容器团队去衡量和取舍。 vivo 互联网团队建设大规模 Kubernetes 集群以来,部分集群较长时间一直使用 v1.10 版本,但是由于业务容器化比例越来越高,对大规模集群稳定性、应用发布的多样性等诉求日益攀升,集群升级迫在眉睫。集群升级后将解决如下问题: 高版本集群在大规模场景做了优化,升级可以解决一系列性能瓶颈问题。 高版本集群才能支持 OpenKruise 等 CNCF 项目,升级可以解决版本依赖问题。 高版本集群增加的新特性能够提高集群资源利用率,降低服务器成本同时提高集群效率。 公司内部维护多个不同版本集群,升级后减少集群版本碎片化,进一步降低运维成本。 这篇文章将会从0到1的介绍 vivo 互联网团队支撑在线业务的集群如何在不影响原有业务正常运行的情况下从 v1.10 ...
- 下一篇
说说hdfs是如何处理块副本多余和缺失的
上一文,我们讲了nn在内存中如何对元数据进行存储和管理的,文章最后也提到了nn内部如何保证块的副本数维持在指定个数,即对副本缺失的块触发块副本复制,对副本多余的块触发块副本删除。本文就来详细聊聊具体流程及细节。 【块副本复制的场景与处理逻辑】 哪些时候会出现需要块的副本数不够,需要进行块副本复制呢? 一种是客户端或管理员手动修改块的副本数,将其扩大;另一种则是块副本所在dn节点异常,包括网络异常,物理机异常,程序被kill导致的异常下线;又或者是块副本存储所在dn节点的磁盘异常,例如磁盘被拔掉,或者确实出现了坏盘。两种情况都会导致块的副本数不满足实际设置要求。 不管哪种情况,在nn内部最终处理时(处理dn的块汇报、处理dn的心跳超时,处理管理员或客户端的设置),都会将不满足副本数的块放到UnderReplicatedBlocks中。 接下来就是块副本复制的处理流程(看图说话) 1. 块副本监测线程(ReplicationMonitorThread)定时监测是否有不满足副本数的块,如果有则取出块信息。 2. 根据块的当前副本数情况,选择一个作为副本复制源的dn节点,同时结合块的存储要求,...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- CentOS7安装Docker,走上虚拟化容器引擎之路
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- Linux系统CentOS6、CentOS7手动修改IP地址
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS关闭SELinux安全模块
- SpringBoot2整合Thymeleaf,官方推荐html解决方案
- Docker安装Oracle12C,快速搭建Oracle学习环境
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- Docker快速安装Oracle11G,搭建oracle11g学习环境