Dart的 IO 庫包含了文件讀寫的相關(guān)類,它屬于 Dart 語法標準的一部分,所以通過 Dart IO 庫,無論是 Dart VM 下的腳本還是 Flutter,都是通過 Dart IO 庫來操作文件的,不過和 Dart VM 相比,F(xiàn)lutter 有一個重要差異是文件系統(tǒng)路徑不同,這是因為Dart VM 是運行在 PC 或服務(wù)器操作系統(tǒng)下,而 Flutter 是運行在移動操作系統(tǒng)中,他們的文件系統(tǒng)會有一些差異。
1. APP目錄
Android 和 iOS 的應用存儲目錄不同,PathProvider插件提供了一種平臺透明的方式來訪問設(shè)備文件系統(tǒng)上的常用位置。該類當前支持訪問兩個文件系統(tǒng)位置:
臨時目錄: 可以使用
getTemporaryDirectory()來獲取臨時目錄; 系統(tǒng)可隨時清除的臨時目錄(緩存)。在 iOS 上,這對應于NSTemporaryDirectory()返回的值。在 Android上,這是getCacheDir()返回的值。文檔目錄: 可以使用
getApplicationDocumentsDirectory()來獲取應用程序的文檔目錄,該目錄用于存儲只有自己可以訪問的文件。只有當應用程序被卸載時,系統(tǒng)才會清除該目錄。在 iOS 上,這對應于NSDocumentDirectory。在 Android 上,這是AppData目錄。外部存儲目錄:可以使用
getExternalStorageDirectory()來獲取外部存儲目錄,如 SD 卡;由于 iOS不支持外部目錄,所以在 iOS 下調(diào)用該方法會拋出UnsupportedError異常,而在 Android 下結(jié)果是Android SDK 中getExternalStorageDirectory的返回值。
2. 文件操作基本使用
2.1 獲取文件操作對象File
File代表一個整體的文件,他有三個構(gòu)造函數(shù),分別是:
factory File(String path)
factory File.fromUri(Uri uri)
factory File.fromRawPath(Uint8List rawPath)
var file = File('file.txt');
2.2 讀取文件所有內(nèi)容
文件讀取本身有兩種形式,一種是文本,一種是二進制。
2.2.1 讀取文本內(nèi)容
如果是文本文件,F(xiàn)ile提供了readAsString、readAsLines、readAsStringSync、readAsLinesSync方法,讀取文本內(nèi)容
readAsString 一次性讀取所有文本
Future<String> readAsString({Encoding encoding: utf8});
var stringContents = await file.readAsString();
readAsLines 一行行的讀取文本
Future<List<String>> readAsLines({Encoding encoding: utf8});
結(jié)果返回的是一個List,list中表示文件每行的內(nèi)容
var lines = await file.readAsLines();
readAsStringSync、readAsLinesSync同步讀取文本
String readAsStringSync({Encoding encoding: utf8});
List<String> readAsLinesSync({Encoding encoding: utf8});
2.2.2 讀取二進制內(nèi)容
如果文件是二進制,那么可以使用readAsBytes或者同步的方法readAsBytesSync:
Future<Uint8List> readAsBytes();
Uint8List readAsBytesSync();
dart中表示二進制有一個專門的類型叫做Uint8List,他實際上表示的是一個int的List。
2.3 以流的形式讀取文件
上面提到的讀取方式,都是一次性讀取整個文件,缺點就是如果文件太大的話,可能造成內(nèi)存空間的壓力。
所以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);
}
}
2.4 隨機訪問
dart提供了open和openSync兩個方法來進行隨機文件讀寫:
Future<RandomAccessFile> open({FileMode mode: FileMode.read});
RandomAccessFile openSync({FileMode mode: FileMode.read});
- FileMode.read 只讀
- FileMode.write 可讀可寫,如果文件存在覆蓋,如果文件不存在創(chuàng)建
- FileMode.append 可讀可寫,如果文件存在在末尾追加,如果文件不存在創(chuàng)建
- FileMode.writeOnly 只寫,如果文件存在覆蓋,如果文件不存在創(chuàng)建
- FileModel.writeOnlyAppend 只寫,如果文件存在在末尾追加,如果文件不存在創(chuàng)建
2.5 文件的寫入
寫入和文件讀取一樣,可以一次性寫入或者獲得一個寫入句柄,然后再寫入。
一次性寫入的方法有四種,分別對應字符串和二進制
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});
句柄形式可以調(diào)用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);
2.6 處理異常
雖然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);
}
}
3. 示例
我們還是以計數(shù)器為例,實現(xiàn)在應用退出重啟后可以恢復點擊次數(shù)。 這里,我們使用文件來保存數(shù)據(jù):
1.引入PathProvider插件;在pubspec.yaml文件中添加如下聲明:
執(zhí)行 flutter pub get
2.實現(xiàn)如下
class MSFileDemo extends StatefulWidget {
const MSFileDemo({
Key? key,
}) : super(key: key);
@override
State<MSFileDemo> createState() => _MSFileDemoState();
}
class _MSFileDemoState extends State<MSFileDemo> {
var _counter = 0;
@override
void initState() {
super.initState();
//從文件讀取點擊次數(shù)
_readCounter().then((value) {
setState(() {
_counter = value;
});
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
body: Center(
child: Text(
"$_counter",
style: TextStyle(
fontSize: 20, color: Colors.red, fontWeight: FontWeight.bold),
),
),
floatingActionButton: FloatingActionButton(
child: Icon(Icons.add),
onPressed: () {
setState(() {
_counter++;
});
_writeCounter();
},
),
);
}
// 獲取本地文件路徑
Future<String> _getLocalFileDir() async {
Directory tempDir = await getTemporaryDirectory();
return tempDir.path;
}
// 獲取文件
Future<File> _getLocalFile() async {
String dir = await _getLocalFileDir();
return File("$dir/counter.txt");
}
// 讀取內(nèi)容
Future<int> _readCounter() async {
try {
File file = await _getLocalFile();
// 讀取文本內(nèi)容
String counterString = file.readAsStringSync();
return int.parse(counterString);
} on FileSystemException {
return 0;
}
}
// 寫入內(nèi)容
void _writeCounter() async {
File file = await _getLocalFile();
file.writeAsString("$_counter"); // 覆蓋寫入
}
}
參考:http://m.itdecent.cn/p/92b09aaecf17
https://book.flutterchina.club/chapter11/file_operation.html