1. 介绍
1.1 介绍
今天福哥带着大家来学习Java读写文件的方法,虽然现阶段做项目需要用到文件操作的情况不多了,但是免不了在特殊情况下还是需要用到这个技术的。
今天福哥还会给大家讲解Java创建、删除、授权文件夹的方法,这个技术在控制台程序里面用途是非常多的。
Java创建、删除、授权文件夹以及读写文件是依靠File对象来实现的,而读写文件有依赖一些Reader和Writer对象,我们来逐个介绍一下。
文件夹操作就是目录操作,在Windows系统里面文件夹叫folder,翻译过来就是文件夹,在Linux系统里面文件夹叫directory,翻译过来就是目录。所以创建、删除、授权文件夹就是创建、删除、授权目录。
2. 基本原则
读写文件有一些常识需要大家先了解一下。
- 读写文件可以是本地电脑上面的文件,也可以是远程网络上面的文件,只要授权了就可以操作。
- 文件夹操作可以是本地电脑上面的文件夹,也可以是远程网络上面的文件夹,只要授权了就可以操作。
- 要创建文件需要对创建文件的文件夹有写权限。
- 读写已经存在的文件只需要对文件有权限。
- 文件内容分为普通模式和二进制模式,普通模式通过字符串操作,二进制模式通过字节操作。
- 写文件分为重置写入和追加写入,前者会清空已有内容,后者不会。
- 通过文件指针可以精确控制读写文件内容的具体位置,但是写入只会覆盖已有内容而不会像编辑器一样插入内容。
- 当前文件夹通过“.”表示,上一级文件夹通过“..”表示。
- 任何文件夹都会有“当前文件夹”和“上一级文件夹”。
3. 文件夹
3.1 遍历
遍历文件夹用到File对象的listFiles方法,可以通过File对象的isFile方法和isDirectory方法判断文件夹下面的项目是文件还是文件夹,可以通过File对象的canRead判断文件/文件夹是否可读,可以通过File对象的canWrite判断文件/文件夹是否可写。
String currDir = Sy("u;); File dir = new File(currDir); File[] files = dir.listFiles(); for(int i=0;i<;i++){ String r, w; r = (files[i].canRead()) ? "可读" : "不可读"; w = (files[i].canWrite()) ? "可写" : "不可写"; if(files[i].isFile()){ out += "文件:[" + r + "][" + w + "]" + files[i].getAbsolutePath() + "<br />"; } else if(files[i].isDirectory()){ out += "文件夹:[" + r + "][" + w + "]" + files[i].getAbsolutePath() + "<br />"; } }
3.2 递归遍历
如果需要遍历一个文件夹下面的所有子级文件和文件夹需要用到递归遍历,就是无论是子级还孙级都要遍历出来。
递归遍历很简单,只要在普通遍历的基础之上稍加改动即可实现,首先需要将普通遍历程序封装为一个方法,然后在普通遍历到的项目为文件夹的时候调用一下方法自己,就可以了。
private String listFolder(String currDir){ String out = ""; File dir = new File(currDir); File[] files = dir.listFiles(); for(int i=0;i<;i++){ String r, w; r = (files[i].canRead()) ? "可读" : "不可读"; w = (files[i].canWrite()) ? "可写" : "不可写"; if(files[i].isFile()){ out += "文件:[" + r + "][" + w + "]" + files[i].getAbsolutePath() + "<br />"; } else if(files[i].isDirectory()){ out += "文件夹:[" + r + "][" + w + "]" + files[i].getAbsolutePath() + "<br />"; out = listFolder(files[i].getAbsolutePath()); } } return out; } @RequestMapping(";) public String response( ){ String out = ""; String currDir = Sy("u;); out += listFolder(currDir); return out; }
3.3 创建
创建文件夹使用File的mkdir方法,创建之后可以通过File的setReadable、setWritable、setExecutable方法授权。通过布尔返回值判断创建文件夹是否成功。
String currDir = Sy("u;); String newDir = currDir + "/newDir"; File dir = new File(newDir); Boolean ret = dir.mkdir(); if(ret){ dir.setReadable(true, false); dir.setWritable(true, false); dir.setExecutable(true, false); }
3.4 删除
删除文件夹使用File的delete方法。
String currDir = Sy("u;); String newDir = currDir + "/newDir"; File dir = new File(newDir); Boolean ret = dir.delete();
3.5 递归删除
如果文件夹含有子级文件/文件夹则需要先删除子级项目才能删除文件夹,这个时候就需要用到递归遍历技术了。
private void removeFolder(String currDir){ File dir = new File(currDir); File[] files = dir.listFiles(); for(int i=0;i<;i++){ if(files[i].isFile()){ files[i].delete(); } else if(files[i].isDirectory()){ removeFolder(files[i].getAbsolutePath()); } } dir.delete(); } @RequestMapping(";) public String response( ){ String out = ""; String currDir = Sy("u;); String newDir = currDir + "/newDir"; removeFolder(newDir); return out; }
4. 文件
4.1 创建
Java创建文件可以通过BufferedWriter对象配合FileWriter对象来实现,也可以通过RandomAccessFile对象实现,RandomAccessFile对象更加像PHP的标准方式fopen。
4.1.1 BufferedWriter
使用BufferedWriter创建文件非常容易,虽然不是一行代码可以搞定的(Java似乎很少有一行代码能搞定的事情)。
String currDir = Sy("u;); String newFile = currDir + ";; try { BufferedWriter file = new BufferedWriter(new FileWriter(newFile)); ("这是福哥新创建的文件"); (); } catch (IOException e){ e.printStackTrace(); }
4.1.2 RandomAccessFile
使用File创建文件需要了解更多知识,首先我们需要指定“文件打开方式”这个概念,使用RandomAccessFile实际上知识打开了一个文件,需要我们指定一个打开文件想要干什么的模式,这个就是文件打开方式。
文件打开方式包括:
- r,读模式
- rw,可读可写模式
- rws,可读可写模式,更新文件内容到设备
- rwd,可读可写模式,更新文件内容到设备
String currDir = Sy("u;); String newFile = currDir + ";; try { RandomAccessFile file = new RandomAccessFile(newFile, "rw"); ((new String("这是福哥新创建的文件")).getBytes()); (); } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.2 删除
删除文件使用File的delete方法。
String currDir = Sy("u;); String newFile = currDir + ";; File file = new File(newFile); ();
4.3 读文件
读文件就是以读模式或者读写模式打开文件。
4.3.1 BufferedReader
使用BufferedReader读取文件内容并没有非常简便的方法。
String currDir = Sy("u;); String newFile = currDir + ";; try { BufferedReader file = new BufferedReader(new FileReader(newFile)); String fc = "", line; while((line = ()) != null){ fc += line; } (); out += fc; } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.3.2 RandomAccessFile
使用RandomAccessFile读文件和BufferedReader差不多,也是需要用到while语句通过readline循环读取,只不过得到的内容需要自己转换一下编码。
String currDir = Sy("u;); String newFile = currDir + ";; try { RandomAccessFile file = new RandomAccessFile(newFile, "r"); String fc = "", line; while((line = ()) != null){ fc += new String("8859_1")); } (); out += fc; } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.4 写文件
写文件就是以写模式或者读写模式打开文件。
4.4.1 BufferedWriter
使用BufferedWriter写入内容到文件并没有非常简便的方法。
try { BufferedWriter fileWriter = new BufferedWriter(new FileWriter(newFile)); BufferedReader fileReader = new BufferedReader(new FileReader(newFile)); ("福哥说这是全部内容了"); (); String fc = "", line; while((line = ()) != null){ fc += line; } (); out += fc; } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.4.2 RandomAccessFile
使用RandomAccessFIle写文件和BufferedWriter差不多,基本上就是从传入字符串改成了传入字节数组。
try { RandomAccessFile fileWriter = new RandomAccessFile(newFile, "rw"); RandomAccessFile fileReader = new RandomAccessFile(newFile, "r"); ((new String("福哥说这是全部内容了")).getBytes()); (); String fc = "", line; while((line = ()) != null){ fc += new String("8859_1")); } (); out += fc; } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.5 追加写文件
有时候我们不想将文件内容全部重置了,需要在文件现有内容后面追加内容怎么办?
4.5.1 BufferedWriter
使用BufferedWriter对象比较简单,可以通过构造器的第二个参数来控制是否开启追加模式。
try { BufferedWriter fileWriter = new BufferedWriter(new FileWriter(newFile, true)); BufferedReader fileReader = new BufferedReader(new FileReader(newFile)); ("福哥又说话了"); (); String fc = "", line; while((line = ()) != null){ fc += line; } (); out += fc; } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.5.2 RandomAccessFile
使用RandomAccessFile对象就是在打开文件后通过seek方法将文件指针移动到文件最后。不知道为什么RandomAccessFile就不设计一个打开方式是追加方式的,真是无语啊~~
try { RandomAccessFile fileWriter = new RandomAccessFile(newFile, "rw"); RandomAccessFile fileReader = new RandomAccessFile(newFile, "r"); ()); ((new String("福哥又说话了")).getBytes()); String fc = "", line; while((line = ()) != null){ fc += new String("8859_1")); } (); out += fc; } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); }
4.6 文件指针
我们以可读可写模式打开文件后,通过seek移动文件指针,在指定位置进行读写操作,这种方式可以避免将文件全部内容加载到内存当中就可以完成很多情况的读写操作,可以解决针对超大文件内容的格式化数据的编辑的难题。
4.6.1 文件数据库
举例:我们制作一个文件数据库,将用户名和密码存储到一个文件里面,用户名和密码格式如下
[用户名(最长50个字符)][密码(取MD5哈希串,所以固定32个字符)][换行]
然后我们只要保证每一组用户信息都是一样的固定长度就可以通过简单计算知道某一个用户ID对应的用户信息在文件的哪个位置上面,进而通过fseek就可以快速定位并读取用户信息了。
因为我们规定了文件内的用户数据格式,那么通过操控文件指针就可以实现这个文本数据库的增、删、改、查功能了,是不是觉得很神奇?
4.6.2 文本数据库定义
福哥在使用RandomAccessFile模拟文本数据库的时候总是有问题,这个等福哥解决了这个问题之后再给大家分享吧~~
4.6.3 文本数据库测试
福哥在使用RandomAccessFile模拟文本数据库的时候总是有问题,这个等福哥解决了这个问题之后再给大家分享吧~~
5. 超大文件读写
超大文件读写需要避免将文件里面的内容全部读出来放到变量里这种行为,因为变量都是存在于内存当中的,如果变量里面的数据太多会把内存用光,就会导致系统瘫痪,这个太可怕了~~
5. 将一个10G文件里的“福哥”换成“鬼谷子叔叔”
这个时候我们可以通过开启两个文件指针,一个文件指针负责读现有文件,一个文件指针负责写新的临时文件,完成后删除现有文件,再将临时文件重命名为原来的文件,就搞定了~~
try{ String currDir = Sy("u;); String newFile = currDir + ";; String tmpFile = currDir + ";; RandomAccessFile foNew = new RandomAccessFile(newFile, "r"); RandomAccessFile foTmp = new RandomAccessFile(tmpFile, "rw"); if(foNew != null && foTmp != null){ String tmpLine; while((tmpLine = ()) != null){ tmpLine = new String("8859_1")); tmpLine = ("福哥", "鬼谷子叔叔"); ()); } (); (); File fiNew = new File(newFile); (); File fiTmp = new File(tmpFile); (fiNew); } } catch (IOException e){ e.printStackTrace(); }
总结
好了,今天童鞋们跟着福哥系统地将Java语言操作文件夹、操作文件的方法学习了一遍,有了这些技术之后,今后在项目当中处理各种有关文件夹/文件的问题就不会发怵了!
要注意一点哦,文件夹操作、文件操作属于IO操作,是有一定风险的,一定不要把文件夹/文件的路径搞错了,要不把系统文件或者程序文件写坏了,系统有可能就完蛋了~~