从上图可以开出,java的读写操作(输入输出)可以用“流”这个概念来表示,总体而言,java的读写操作又分为两种:字符流和字节流。
流是一个抽象的概念。当Java程序需要从数据源读取数据时,会开启一个到数据源的流。数据源可以是文件,内存或者网络等。同样,当程序需要输出数据到目的地时也一样会开启一个流,数据目的地也可以是文件、内存或者网络等。流的创建是为了更方便地处理数据的输入输出。
1.字节流和字符流的区别
1.字节流也称为原始数据,需要用户读入后进行相应的编码转换。而字符流的实现是基于自动转换的,读取数据时会把数据按照JVM的默认编码自动转换成字符。
2.字符流处理的单元为2个字节的Unicode字符,分别操作字符、字符数组或字符串,而字节流处理单元为1个字节,操作字节和字节数组。
所以字符流是由Java虚拟机将字节转化为2个字节的Unicode字符为单位的字符而成的。
3.字节流可用于任何类型的对象,包括二进制对象,而字符流只能处理字符或者字符串,字节流提供了处理任何类型的IO操作的功能,但它不能直接处理Unicode字符,而字符流就可以;
2.字符流字节流如何选择
如果是音频文件、图片、歌曲,就用字节流好点;如果是中文(文本)的,用字符流更好;
3.使用步骤:
1 使用File类打开一个文件
2 通过字节流或字符流的子类,指定输出的位置,注,
3 进行读/写操作
4 关闭输入/输出
4。注意事项:
1.我们很容易搞混,或者弄不清inputstream和outpustream到底哪个是读数据,哪个是写数据,这里要说明的是,“读和写”是相对于程序本身而言的,主要记清楚一点即可,那就是凡是需要提供给程序处理的数据就是输入数据,当然就是inputstream,这里的in是要in到程序里面去,那么数据从哪里来呢,自然是从外界(网络,内存或者文件),那么针对文件而言,inputstream就是读文件了。反之,凡是程序已经处理过的数据,当然要流出程序啦,那就是out咯,所以outputstream就是输出的,那么从程序中流出,只能到外界(网络、内存或者文件),针对文件而言,就是写操作啦!(同样的道理可以针对字符流的操作)
需求描述:
需要获取手机里面一个txt文件的内容,里面为JSON数据。
于是乎,我写出了下面的代码
第一步:通过方法获取txt里面的数据:
private String getHardversionInformation(String key) {
FileInputStream fis = null;
String hardwareInfoPath = "/firmware/verinfo/ver_info.txt";
byte[] bytes = new byte[1024];//一次读取1024字节的数据
int count;
String allVersionInfo = "";
String hardVersionInfo = "";
try {
fis = new FileInputStream(hardwareInfoPath);//通过字节流获取
count = fis.read(bytes);//开始读取
fis.close();//关闭字节
allVersionInfo = new String(bytes, 0, count);//转化为字符串
JSONObject jsonObject = new JSONObject(allVersionInfo);//将字符串转化为json对象
JSONObject infoJsonObject = jsonObject.getJSONObject("Image_Build_IDs");//获取json对象
hardVersionInfo = infoJsonObject.getString(key);获取json对象中的具体值
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException ie) {
ie.printStackTrace();
} catch (JSONException e) {
e.printStackTrace();
} finally {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return hardVersionInfo;
}
2.调用方法显示出来:
String gpsVersion = getHardversionInformation("modem");//获取key为modem的数据
String wifiVersion = getHardversionInformation("wcnss");//获取key为wcnss的数据
if (!TextUtils.isEmpty(gpsVersion)) {
setStringSummary(KEY_GPS_VERSION, gpsVersion);//显示出来
}
if (!TextUtils.isEmpty(wifiVersion)) {
setStringSummary(KEY_WIFI_VERSION, wifiVersion);//显示出来
setStringSummary(KEY_BLUETOOTH_VERSION, wifiVersion);
}
这一顿操作也算是完成了任务。
但是大佬就只说了一句话,这个代码太low了。
针对上面的low代码,有下面几个地方有问题:
1.每次getHardversionInformation()是不是都是io操作,很耗时间呢
2.数据要是大于1kb怎么办?
3.如果 fis = new FileInputStream(hardwareInfoPath);//通过字节流获取 这里发生异常,fis将为空,
finally {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
这里面的fis.close()方法将会出现空指针
4.如果这里异常JSONObject jsonObject = new JSONObject(allVersionInfo);//将字符串转化为json对象,是不是fis会出现重复关闭呢
针对以上问题,给出正确的打开方式!
1.获取json对象,这样只需要一次操作
private JSONObject getJsonObject() {
FileInputStream fis = null;
String hardwareInfoPath = "/firmware/verinfo/ver_info.txt";
byte[] bytes = new byte[1024];
int count;
String allVersionInfo = "";
String hardVersionInfo = "";
try {
fis = new FileInputStream(hardwareInfoPath);
StringBuilder stringBuilder = new StringBuilder();//StringBuilder效率比buffer高,如果多线程需要用buffer
while ((count = fis.read(bytes)) != -1) {
stringBuilder.append(new String(bytes, 0, count));
}
JSONObject jsonObject = new JSONObject(stringBuilder.toString());
JSONObject infoJsonObject = jsonObject.getJSONObject("Image_Build_IDs");
return infoJsonObject;//正常情况下,走到这里就退出,此时finally代码也会执行,但是不会改变返回值。
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException ie) {
ie.printStackTrace();
} catch (JSONException e) {
e.printStackTrace();
} finally {
try {
if(fis != null) {
fis.close();//在finally里面关闭流
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
2.获取
不贴代码!
1、不管有没有异常,finally中的代码都会执行
2、当try、catch中有return时,finally中的代码依然会继续执行
3、finally是在return后面的表达式运算之后执行的,此时并没有返回运算之后的值,而是把值保存起来,不管finally对该值做任何的改变,返回的值都不会改变,依然返回保存起来的值。也就是说方法的返回值是在finally运算之前就确定了的。
4、finally代码中最好不要包含return,程序会提前退出,也就是说返回的值不是try或catch中的值
参考自:
2017java文件操作(读写操作)