文件

1. 文件的一些常用方法

package com.ljw.study;

import java.io.File;
import java.util.Arrays;
import java.util.Date;

public class FileDemo {

	public static void main(String[] args) {
		// 构造文件对象
		// 绝对路径,构造文件夹对象
		File f1 = new File("d:/work/code/ljw/Test");
		System.out.println(f1.exists()); // 文件是否存在

		// 相对路径,如果是eclipse的话,就是相对于项目目录
		File f2 = new File("employee.ser");
		System.out.println("f2的绝对路径:" + f2.getAbsolutePath());
		System.out.println(f2.exists());

		// 以f1为父目录创建文件对象
		File f3 = new File(f1,"LOL.exe");
		System.out.println("f3的绝对路径:" + f3.getAbsolutePath());

		System.out.println(f2.isDirectory()); // 是否是文件夹

		System.out.println(f2.isFile()); // 是否为文件(非文件夹)

		System.out.println(f2.length()); // 文件长度

		System.out.println(new Date(f2.lastModified())); // 最后的修改时间

//		System.out.println(f2.renameTo(new File("employee.ser")));// 文件重命名

		System.out.println(Arrays.toString(f1.list())); // 以字符串数组的形式,返回当前文件夹下的所有文件(不包含子文件及子文件夹)

		System.out.println(Arrays.toString(f1.listFiles())); // 以文件数组的形式,返回当前文件夹下的所有文件(不包含子文件及子文件夹)

		// 以字符串形式返回获取所在文件夹
        System.out.println(f1.getParent());

        // 以文件形式返回获取所在文件夹
        f2.getParentFile();
        // 创建文件夹,如果父文件夹skin不存在,创建就无效
        f1.mkdir();

        // 创建文件夹,如果父文件夹skin不存在,就会创建父文件夹
        f1.mkdirs();

        // 创建一个空文件,如果父文件夹skin不存在,就会抛出异常
        f1.createNewFile();
        // 所以创建一个空文件之前,通常都会创建父目录
        f1.getParentFile().mkdirs();

        // 列出所有的盘符c: d: e: 等等
        f1.listRoots();

        // 刪除文件
        f.delete();

        // JVM结束的时候,刪除文件,常用于临时文件的删除
        f.deleteOnExit()
    }
}

2. 读取和写入文件数据

2.1 以字节流方式

2.1.1 读取数据

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

public class TestStream {

    public static void main(String[] args) {
        try {
            //准备文件lol.txt其中的内容是AB,对应的ASCII分别是65 66
            File f =new File("d:/lol.txt");
            //创建基于文件的输入流
            FileInputStream fis =new FileInputStream(f);
            //创建字节数组,其长度就是文件的长度
            byte[] all =new byte[(int) f.length()];
            //以字节流的形式读取文件所有内容
            fis.read(all);
            for (byte b : all) {
                //打印出来是65 66
                System.out.println(b);
            }

            //每次使用完流,都应该进行关闭
            fis.close();

        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

2.1.2 写入数据

package com.ljw.practice;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		File f = new File("d:/xyz/abc/def/lol2.txt");
		f.getParentFile().mkdirs();

		try {
			FileOutputStream outputFile = new FileOutputStream(f);
			String s = "秋天,多了一些思念";
			byte[] b = s.getBytes(); // 将字符串转换为字节数组byte[]
			outputFile.write(b);
			outputFile.close();
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

	}
}

2.2 以字符流方式

2.2.1 读取数据


package com.ljw.practice;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
		File f = new File("d:/lol.txt");

		try(FileReader fr = new FileReader(f)){ // 数据流写在try()里面,系统会帮我们自动关闭数据流
			char[] c = new char[(int) f.length()];
			fr.read(c);
			for(char i:c) {
				System.out.print(i);
			}
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

2.2.2 写入数据


package com.ljw.practice;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
		File f = new File("d:/lol.txt");
		File f2 = new File("d:/lol2.txt");

		try(FileReader fr = new FileReader(f);FileWriter fw = new FileWriter(f2)){
			char[] c = new char[(int) f.length()];
			fr.read(c); // 读出数据到字符数组
			fw.write(c); // 写入数据从数组到文件
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

2.3 以缓存流方式

  • 缓存流必须建立在一个存在的流的基础上
  • 以介质是硬盘为例,字节流和字符流的弊端:在每一次读写的时候,都会访问硬盘。 如果读写的频率比较高的时候,其性能表现不佳。
  • 为了解决以上弊端,采用缓存流。缓存流在读取的时候,会一次性读较多的数据到缓存中,以后每一次的读取,都是在缓存中访问,直到缓存中的数据读取完毕,再到硬盘中读取。
  • 如果需要立即将缓存中的数据写入硬盘,可以使用flush()方法

2.3.1 读取数据


package stream;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;

public class TestStream {
    public static void main(String[] args) {
        File f = new File("d:/lol.txt");
        // 创建文件字符流
        // 缓存流必须建立在一个存在的流的基础上
        try (
                FileReader fr = new FileReader(f);
                BufferedReader br = new BufferedReader(fr);
            )
        {
            while (true) {
                // 一次读一行
                String line = br.readLine();
                if (null == line)
                    break;
                System.out.println(line);
            }
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

    }
}

2.3.2 写入数据

package com.ljw.practice;

import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;

public class FileDemo {

	public static void main(String[] args) {
		File f = new File("d:/lol2.txt");

        try (
                // 创建文件字符流
                FileWriter fw = new FileWriter(f);
                // 缓存流必须建立在一个存在的流的基础上
                PrintWriter pw = new PrintWriter(fw);
        ) {
            pw.println("garen kill teemo");
            pw.println("teemo revive after 1 minutes");
            pw.println("teemo try to garen, but killed again");
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
	}
}

2.4 以数据流方式

  • 使用数据流的writeUTF()和readUTF() 可以进行数据的格式化顺序读写
  • 要用DataInputStream 读取一个文件,这个文件必须是由DataOutputStream 写出的,否则会出现EOFException,因为DataOutputStream 在写出的时候会做一些特殊标记,只有DataInputStream 才能成功的读取。

2.4.1 实例

  • 以数据流的方式读写数据
package com.ljw.practice;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
        write();
        read();
	}

	private static void read() {
		File f = new File("d:/lol2.txt");
		try(
				FileInputStream fi = new FileInputStream(f);
				DataInputStream di = new DataInputStream(fi); // 创建一个数据流对象
			)
		{
			boolean b = di.readBoolean();
			int i = di.readInt();
			String s = di.readUTF();
			System.out.println(b);
			System.out.println(i);
			System.out.println(s);

		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e1) {
			// TODO Auto-generated catch block
			e1.printStackTrace();
		}

	}

	private static void write() {
		File f = new File("d:/lol2.txt");

		try(
				FileOutputStream fo = new FileOutputStream(f);
				DataOutputStream dos = new DataOutputStream(fo);
			)
		{
			dos.writeBoolean(false);
			dos.writeInt(200);
			dos.writeUTF("中国");

		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

2.5 以对象流(序列化对象)

  • 把一个对象序列化有一个前提是:这个对象的类,必须实现了Serializable接口
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

import charactor.Hero;

public class TestStream {

    public static void main(String[] args) {
        //创建一个Hero garen
        //要把Hero对象直接保存在文件上,务必让Hero类实现Serializable接口
        Hero h = new Hero();
        h.name = "garen";
        h.hp = 616;

        //准备一个文件用于保存该对象
        File f =new File("d:/garen.lol");

        try(
            //创建对象输出流
            FileOutputStream fos = new FileOutputStream(f);
            ObjectOutputStream oos =new ObjectOutputStream(fos);
            //创建对象输入流
            FileInputStream fis = new FileInputStream(f);
            ObjectInputStream ois =new ObjectInputStream(fis);
        ) {
            oos.writeObject(h);
            Hero h2 = (Hero) ois.readObject();
            System.out.println(h2.name);
            System.out.println(h2.hp);

        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

3. 拆分文件 和 合并文件

3.1 拆分文件

package com.ljw.practice;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		File file = new File("d:/F.docx");

		try {
			FileInputStream inFile = new FileInputStream(file);
			int i = 1;
			while(true) {
				byte[] b = new byte[1024*100];
				int n = inFile.read(b);
				if(n==-1) {  // 没有读到数据返回 -1
					break;
				}
				File f1 = new File(String.format("d:/F-%d.docx",i)); // 占位符
				FileOutputStream outFile = new FileOutputStream(f1);
				outFile.write(b);
				i++;
			}

		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

3.2 合并文件

package com.ljw.practice;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		File newF = new File("d:/newF.docx");
		try {
			FileOutputStream outNewF = new FileOutputStream(newF);
			for(int i=1;i<5;i++) {
				File f = new File(String.format("d:/F-%d.docx", i));
				byte[] b = new byte[(int) f.length()];
				FileInputStream inf = new FileInputStream(f);
				inf.read(b);
				outNewF.write(b);
				inf.close();
			}
			outNewF.close();
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

4. 中文问题

package com.ljw.practice;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

public class FileDemo {

	public static void main(String[] args) {
		File f = new File("d:/lol.txt");
        try (FileInputStream fis = new FileInputStream(f);) {
            byte[] all = new byte[(int) f.length()];
            fis.read(all);

            //文件中读出来的数据是
            System.out.println("文件中读出来的数据是:");
            for (byte b : all)
            {
                int i = b&0xff;  //只取16进制的后两位
                System.out.println(Integer.toHexString(i));
            }
            System.out.println("把这个数字,放在GBK的棋盘上去:");
            String str = new String(all,"GBK");
            System.out.println(str);
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
	}
}

综合练习

  • 复制文件或文件夹
  • 查找指定的内容
package com.ljw.practice;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.IOException;

public class FilePractice {

	public static void main(String[] args) {
		//----------复制文件--------------
		File srcFile = new File("d:/lol.txt");
		File destFile = new File("d:/lol3.txt");
		copyFile(srcFile,destFile);

		//----------复制文件夹------------
		File srcFolder = new File("d:/xyz");
		File destFolder = new File("d:/xyz-副本");
		copyFolder(srcFolder,destFolder);

		//----------查找---------------
		String s = "hello";
		findFileContent(s, srcFolder);

	}

	/**
	 * 复制文件
	 * @param srcFile:源文件
	 * @param destFile:目标文件
	 * @return void
	 */
	public static void copyFile(File srcFile, File destFile) {
		if(srcFile.exists()) {
			try(
					// 创建读取文件流对象
					FileInputStream fi = new FileInputStream(srcFile);
					// 创建写入文件流对象
					FileOutputStream fo = new FileOutputStream(destFile);

				)
			{
				byte[] b = new byte[10];
				int i;
				while(true) {
					i = fi.read(b);
					if(-1==i) {
						break;
					}
					fo.write(b,0,i);
				}
			} catch (FileNotFoundException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				} catch (IOException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
		}else {
			System.out.println("文件不存在");
		}
	}


	/**
	 * 复制文件夹
	 * @param srcFile
	 * @param destFile
	 */
	public static void copyFolder(File srcFile, File destFile) {
		if(srcFile.isDirectory()) {
			File[] files = srcFile.listFiles();
			destFile.mkdir();
			for(File f:files) {
				if(f.isDirectory()) {
					String newFolderPath = destFile.getAbsolutePath()+"\\"+f.getName();
					File newFolder = new File(newFolderPath);
					newFolder.mkdir();
					System.out.println("新文件夹的位置是:"+newFolderPath);
					copyFolder(f,newFolder);
				}else {
					String newFilePath = destFile.getAbsolutePath()+"\\"+f.getName();
					System.out.println("新文件的位置是:"+newFilePath);
					File newF = new File(newFilePath);
					copyFile(f,newF);
				}
			}
		}else {
			copyFile(srcFile, destFile);
		}
	}

	/**
	 * 查找指定的文件内容
	 * @param s:要查找的字符串
	 * @param file:要查找的文件对象
	 */
	public static void findFileContent(String s,File file) {
		if(file.isDirectory()) {
			File[] files = file.listFiles();
			for(File f:files) {
				findFileContent(s,f);
			}
		}else {
			try(
					FileReader fr = new FileReader(file);
					BufferedReader br = new BufferedReader(fr);
				) {

				while(true) {
					String str = br.readLine();
					if(null == str) {break;}
					if(str.contains(s)) {
						System.out.println("找到目标字符串"+s+",在文件:"+file.getAbsolutePath());
						break;
					}
				}

			} catch (FileNotFoundException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} catch (IOException e1) {
				// TODO Auto-generated catch block
				e1.printStackTrace();
			}
		}
	}
}

11-09 07:38