基本覆盖java IO的全部内容

案例1】创建一个新文件

1
2
3
4
5
6
7
8
9
10
11
import java.io.*;
class hello{
  public static void main(String[] args) {
    File f= new File( "D:\\hello.txt" );
  try {
    f.createNewFile();
  } catch (Exception e) {
    e.printStackTrace();
  }
  }
}


【运行结果】:

程序运行之后,在d盘下会有一个名字为hello.txt的文件。

案例2】File类的两个常量

1
2
3
4
5
6
7
import java.io.*;
class hello{
  public static void main(String[] args) {
  System.out.println(File.separator);
  System.out.println(File.pathSeparator);
  }
}

【运行结果】:

\

;

此处多说几句:有些同学可能认为,我直接在windows下使用\进行分割不行吗?当然是可以的。但是在linux下就不是\了。所以,要想使得我们的代码跨平台,更加健壮,所以,大家都采用这两个常量吧,其实也多写不了几行。

现在我们使用File类中的常量改写上面的代码:

1
2
3
4
5
6
7
8
9
10
11
12
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  try {
  f.createNewFile();
  } catch (Exception e) {
  e.printStackTrace();
  }
  }
}

你看,没有多写多少吧,呵呵。所以建议使用File类中的常量。

删除一个文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/**
  * 删除一个文件
  * */
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  if (f.exists()){
  f.delete();
  } else {
  System.out.println( "文件不存在" );
  }
 
  }
}

创建一个文件夹

1
2
3
4
5
6
7
8
9
10
11
/**
  * 创建一个文件夹
  * */
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator+ "hello" ;
  File f= new File(fileName);
  f.mkdir();
  }
}

【运行结果】:

D盘下多了一个hello文件夹

列出指定目录的全部文件(包括隐藏文件):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
/**
  * 使用list列出指定目录的全部文件
  * */
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator;
  File f= new File(fileName);
  String[] str=f.list();
  for ( int i = 0 ; i < str.length; i++) {
  System.out.println(str[i]);
  }
  }
}

【运行结果】:

$RECYCLE.BIN

360

360Downloads

360Rec

360SoftMove

Config.Msi

da

Downloads

DriversBackup

(你的运行结果应该和这个不一样的)

但是使用list返回的是String数组,。而且列出的不是完整路径,如果想列出完整路径的话,需要使用listFiles.他返回的是File的数组

列出指定目录的全部文件(包括隐藏文件):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/**
  * 使用listFiles列出指定目录的全部文件
  * listFiles输出的是完整路径
  * */
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator;
  File f= new File(fileName);
  File[] str=f.listFiles();
  for ( int i = 0 ; i < str.length; i++) {
  System.out.println(str[i]);
  }
  }
}

【运行结果】:

D:\$RECYCLE.BIN

D:\360

D:\360Downloads

D:\360Rec

D:\360SoftMove

D:\Config.Msi

D:\da

D:\Downloads

D:\DriversBackup

通过比较可以指定,使用listFiles更加方便、

判断一个指定的路径是否为目录

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/**
  * 使用isDirectory判断一个指定的路径是否为目录
  * */
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator;
  File f= new File(fileName);
  if (f.isDirectory()){
  System.out.println( "YES" );
  } else {
  System.out.println( "NO" );
  }
  }
}

【运行结果】:YES

搜索指定目录的全部内容

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
/**
  * 列出指定目录的全部内容
  * */
import java.io.*;
class hello{
  public static void main(String[] args) {
  String fileName= "D:" +File.separator;
  File f= new File(fileName);
  print(f);
  }
  public static void print(File f){
  if (f!= null ){
  if (f.isDirectory()){
  File[] fileArray=f.listFiles();
  if (fileArray!= null ){
  for ( int i = 0 ; i < fileArray.length; i++) {
  //递归调用
  print(fileArray[i]);
  }
  }
  }
  else {
  System.out.println(f);
  }
  }
  }
}

【运行结果】:

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\framepages\web4welcome_jsp.java

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\help_005fhome_jsp.class

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\help_005fhome_jsp.java

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\home_jsp.class

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\home_jsp.java

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\index_jsp.class

D:\Tomcat6\work\Catalina\localhost\nevel\org\apache\jsp\index_jsp.java

......

【使用RandomAccessFile写入文件】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
  * 使用RandomAccessFile写入文件
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  RandomAccessFile demo= new RandomAccessFile(f, "rw" );
  demo.writeBytes( "asdsad" );
  demo.writeInt( 12 );
  demo.writeBoolean( true );
  demo.writeChar( 'A' );
  demo.writeFloat( 1 .21f);
  demo.writeDouble( 12.123 );
  demo.close();
  }
}

如果你此时打开hello。txt查看的话,会发现那是乱码。

字节流

【向文件中写入字符串】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/**
  * 字节流
  * 向文件中写入字符串
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  OutputStream out = new FileOutputStream(f);
  String str= "你好" ;
  byte [] b=str.getBytes();
  out.write(b);
  out.close();
  }
}

查看hello.txt会看到“你好”

当然也可以一个字节一个字节的写。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
  * 字节流
  * 向文件中一个字节一个字节的写入字符串
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  OutputStream out = new FileOutputStream(f);
  String str= "你好" ;
  byte [] b=str.getBytes();
  for ( int i = 0 ; i < b.length; i++) {
  out.write(b[i]);
  }
  out.close();
  }
}

结果还是:“你好”

向文件中追加新内容:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
/**
  * 字节流
  * 向文件中追加新内容:
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  OutputStream out = new FileOutputStream(f, true );
  String str= "Rollen" ;
  //String str="\r\nRollen"; 可以换行
  byte [] b=str.getBytes();
  for ( int i = 0 ; i < b.length; i++) {
  out.write(b[i]);
  }
  out.close();
  }
}

【运行结果】:

你好Rollen

读取文件内容】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/**
  * 字节流
  * 读文件内容
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  InputStream in= new FileInputStream(f);
  byte [] b= new byte [ 1024 ];
  in.read(b);
  in.close();
  System.out.println( new String(b));
  }
}

【运行结果】

你好Rollen

Rollen_

但是这个例子读取出来会有大量的空格,我们可以利用in.read(b);的返回值来设计程序。如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
  * 字节流
  * 读文件内容
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  InputStream in= new FileInputStream(f);
  byte [] b= new byte [ 1024 ];
  int len=in.read(b);
  in.close();
  System.out.println( "读入长度为:" +len);
  System.out.println( new String(b, 0 ,len));
  }
}

【运行结果】:

读入长度为:18

你好Rollen

Rollen

读者观察上面的例子可以看出,我们预先申请了一个指定大小的空间,但是有时候这个空间可能太小,有时候可能太大,我们需要准确的大小,这样节省空间,那么我们可以这样干:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
  * 字节流
  * 读文件内容,节省空间
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  InputStream in= new FileInputStream(f);
  byte [] b= new byte [( int )f.length()];
  in.read(b);
  System.out.println( "文件长度为:" +f.length());
  in.close();
  System.out.println( new String(b));
  }
}

文件长度为:18

你好Rollen

Rollen

将上面的例子改为一个一个读:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
  * 字节流
  * 读文件内容,节省空间
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  InputStream in= new FileInputStream(f);
  byte [] b= new byte [( int )f.length()];
  for ( int i = 0 ; i < b.length; i++) {
  b[i]=( byte )in.read();
  }
  in.close();
  System.out.println( new String(b));
  }
}

输出的结果和上面的一样。

细心的读者可能会发现,上面的几个例子都是在知道文件的内容多大,然后才展开的,有时候我们不知道文件有多大,这种情况下,我们需要判断是否独到文件的末尾。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/**
  * 字节流
  *读文件
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  InputStream in= new FileInputStream(f);
  byte [] b= new byte [ 1024 ];
  int count = 0 ;
  int temp= 0 ;
  while ((temp=in.read())!=(- 1 )){
  b[count++]=( byte )temp;
  }
  in.close();
  System.out.println( new String(b));
  }
}

【运行结果】

你好Rollen

Rollen_

提醒一下,当独到文件末尾的时候会返回-1.正常情况下是不会返回-1的

字符流


【向文件中写入数据】

现在我们使用字符流

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/**
  * 字符流
  * 写入数据
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  Writer out = new FileWriter(f);
  String str= "hello" ;
  out.write(str);
  out.close();
  }
}

当你打开hello。txt的时候,会看到hello

其实这个例子上之前的例子没什么区别,只是你可以直接输入字符串,而不需要你将字符串转化为字节数组。

当你如果想问文件中追加内容的时候,可以使用将上面的声明out的哪一行换为:

Writer out =new FileWriter(f,true);

这样,当你运行程序的时候,会发现文件内容变为:

hellohello如果想在文件中换行的话,需要使用“\r\n”

比如将str变为String str=”\r\nhello”;

这样文件追加的str的内容就会换行了。

从文件中读内容:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
  * 字符流
  * 从文件中读出内容
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  char [] ch= new char [ 100 ];
  Reader read= new FileReader(f);
  int count=read.read(ch);
  read.close();
  System.out.println( "读入的长度为:" +count);
  System.out.println( "内容为" + new String(ch, 0 ,count));
  }
}

【运行结果】:

读入的长度为:17

内容为hellohello

hello

当然最好采用循环读取的方式,因为我们有时候不知道文件到底有多大。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/**
  * 字符流
  * 从文件中读出内容
  * */
import java.io.*;
class hello{
  public static void main(String[] args) throws IOException {
  String fileName= "D:" +File.separator+ "hello.txt" ;
  File f= new File(fileName);
  char [] ch= new char [ 100 ];
  Reader read= new FileReader(f);
  int temp= 0 ;
  int count= 0 ;
  while ((temp=read.read())!=(- 1 )){
  ch[count++]=( char )temp;
  }
  read.close();
  System.out.println( "内容为" + new String(ch, 0 ,count));
  }
}

运行结果:

内容为hellohello

hello

关于字节流和字符流的区别

实际上字节流在操作的时候本身是不会用到缓冲区的,是文件本身的直接操作的,但是字符流在操作的 时候下后是会用到缓冲区的,是通过缓冲区来操作文件的。

读者可以试着将上面的字节流和字符流的程序的最后一行关闭文件的代码注释掉,然后运行程序看看。你就会发现使用字节流的话,文件中已经存在内容,但是使用字符流的时候,文件中还是没有内容的,这个时候就要刷新缓冲区。

使用字节流好还是字符流好呢?

答案是字节流。首先因为硬盘上的所有文件都是以字节的形式进行传输或者保存的,包括图片等内容。但是字符只是在内存中才会形成的,所以在开发中,字节流使用广泛。



你可能感兴趣的:(基本覆盖java IO的全部内容)