文件和文件流

2022-11-01 17:06:36 浏览数 (1)

大家好,又见面了,我是你们的朋友全栈君。

文件和文件流

1. File类

Java.io.File类可以获取文件以及文件夹的一些基本的属性 常用的方法 文件名称,路径,大小,判断是否存在,删除,创建

代码语言:javascript复制
// 创建一个文件对象(可以是文件,可以是文件夹)
		File file = new File("e:/java_text.txt");
		// 基本属性
		boolean canWriter = file.canWrite();
		System.out.println("是否可写:"   canWriter);
		boolean canRead = file.canRead();
		System.out.println("是否可读:"   canRead);
		long size = file.length(); // 常用
		System.out.println("文件大小:"   size);
		boolean isFile = file.isFile(); // 常用
		System.out.println("是否是文件:"   isFile);
		boolean isDirectory = file.isDirectory();
		System.out.println("是否是文件夹:"   isDirectory);
		String filename = file.getName(); // 常用
		System.out.println("文件的名称:"   filename);
		String absolutePath = file.getAbsolutePath(); // 常用
		System.out.println("文件的绝对路径:"   absolutePath);
		String filepath = file.getPath();
		System.out.println("文件的绝对路径:"   filepath);
		boolean isExists = file.exists(); // 常用
		System.out.println("是否存在:"   isExists);
		boolean isDelete = file.delete(); // 常用
		System.out.println("是否已经被删除:"   isDelete);
		boolean isCreate = file.createNewFile();
		System.out.println("创建了没有:"   isCreate);

File类的listFile获取文件夹下面的所有文件内容,可以通过递归调用的方法把某一个文件夹下的所有的文件查询出来

代码语言:javascript复制
// 测试文件目录的属性(递归遍历文件夹中所有的文件信息)
	public static void testDirectoryDeme(File file) {
		if (file.isDirectory()) {
			File[] files = file.listFiles();
			System.out.println("文件夹" file.getName() "有" files.length "个文件");
			// 利用for遍历所有的文件
			for (int i = 0; i < files.length; i  ) {
				File childFile = files[i];
				if (childFile.isFile()) {
					// 这是一个文件
					System.out.println("t这是一个文件:"   childFile.getName());
				} else {
					// 文件夹:继续递归调用
					testDirectoryDeme(childFile);
				}
			}
		} else {
			// 这是一个文件
			System.out.println("t这是一个文件:"   file.getName());
		}
	}

2. Files 和 paths是一个工具类,提供了对文件的基本功能的实现在java.nio包下面

代码语言:javascript复制
文件的创建,删除,判断是否存在,移动,拷贝. 因为提供了静态的方法,所以不需要创建对象直接调用方法即可
代码语言:javascript复制
// 如果文件不存在复制
if (!Files.exists(Paths.get("e:/a/cart1.jpg"))) {
	// java.nio.Files(文件的工具类) Paths(文件路径工具类)
	Files.copy(Paths.get("e:/cart1.jpg"), Paths.get("e:/a/cart1.jpg"));
}
Files.move(Paths.get("e:/a/cart1.jpg"), Paths.get("e:/a/b/cart1"   ((int) (Math.random() * 100))   ".jpg"));
Files.delete(Paths.get("e:/cart1.jpg"));

3. 文件流

文件流的分类: 根据功能分为:输入流(读取文件) 和 输出流(写入文件) 根据操作内容:字符流(读取字符数组) 和 字节流(读取字节数组) 字节输入流,字节输出流,字符输入流,字符输出流

使用字节流实现文件的读取

代码语言:javascript复制
//利用字节输入流实现文件的内容读取(inputStream 接口的  FileInputStream )
	public static void testInputStream() throws Exception{
		File file = new File("e:/a/file.txt");
		InputStream is = new FileInputStream(file);
		if(!file.exists()){
			System.out.println("文件不存在");
		}
		//开始读取文件
		byte[] temp_bytes = new byte[1024];
		int size = 0;  //用于记录读取文件的字节个数,如果没有读取任何的内容返回-1
		//因为文件不可能一次读取完毕,需要循环读取
		do{ 
			size =is.read(temp_bytes);
			if(size!=-1){
				String info = new String(temp_bytes,0,size,"GBK");
				System.out.println("读取的内容是:"   info);
			}
		}while(size !=-1); 
		//文件流是必须要关闭的(像水管子一样)
		is.close(); 

利用字节输出流实现文件的写入

代码语言:javascript复制
//利用字节输出流实现文件内容的写入(OutputStream 接口的FileOutputStream)
	public static void testOuputStream() throws Exception{
		File file = new File("e:/a/file_new.txt");
		if(file.exists()){
			file.createNewFile();
		}
		//文件写入
		String info = "这就是我们要写入文件的内容";
		//创建文件写入对象
		OutputStream os = new FileOutputStream(file);
		os.write(info.getBytes());
		//写入完毕后,关闭
		os.flush();  //清空缓存区
		os.close();
	}

利用字节输入流和字节输出流实现文件的拷贝

代码语言:javascript复制
//利用字节输入输入输出流,实现文件的复制,为了防止文件名称重复,需要对文件名称重命名
	public static void testCopy(String filepath) throws Exception
	{
		//创建文件对象
		File file = new File(filepath);
		//判断文件是否存在,且必须是一个文件而不能是一个文件夹
		if(!file.exists())
			throw new Exception("文件不存在");
		if(file.isDirectory())
			throw new Exception("只能拷贝文件,不能拷贝文件夹");
		//默认目标地址就是e:/a 文件夹
		
		//开始拷贝
		//创建一个文件输入流对象,读取文件的内容
		InputStream is = new FileInputStream(file);
		//创建一个文件输出流对象,写入文件的内容
		String filename = getFileName(file.getName());
		String targetpath="e:/a/" filename;
		OutputStream os = new FileOutputStream(targetpath);
		//利用循环,边读取内容,边写入内容即可
		byte[] temp_info = new byte[1024];  //利用临时数组保存读取的内容
		int size = 0; //保存读取的字节个数,如果没有读取到内容返回-1
		do{
			//先读取
			size = is.read(temp_info);
			//判断是否读取到了文件的内容
			if(size!=-1){
				//写入文件
				os.write(temp_info, 0, size);
			}
		}while(size!=-1);
		
		//关闭,先关闭输出流,后关闭输入流
		os.flush();
		os.close();
		is.close();
	}
	//根据原有的文件名称获取新的文件名称
	public static String getFileName(String fileName){
		SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddhhmmss");
		//abc.mp4  ---abc_20190805164520.mp4
		//根据.分别获取文件名称和扩展名
		String[] name_infos = fileName.split("\.");
		//获取当前日期的字符串
		Date date = new Date();
		String dateStr = sdf.format(date);
		 
		return name_infos[0] "_" dateStr "." name_infos[1];
	}

重点是熟练使用FileInputStream 和FileOutputStream 的使用

代码语言:javascript复制
// 利用字节流复制某一个文件夹中的所有文件
	public static void testCopyDirectory() throws Exception {
		File file = new File("e:/file_source");
		// 实现复制
		// 创建文件输入输出流对象
		InputStream is = null;
		OutputStream os = null;
		try {
			// 遍历这个文件夹下的所有的文件信息
			File[] files = file.listFiles();
			for (int i = 0; i < files.length; i  ) {
				File childFile = files[i];
				// 如果是一个文件就复制
				if (childFile.isFile()) {
					is = new FileInputStream(childFile);
					// 根据原有的文件名称获取新的文件名称
					String newFileName = getNewFileName(childFile.getName());
					os = new FileOutputStream("e:/a/b/"   newFileName);
					byte[] temp_info = new byte[1024];
					int size = -1;
					do {
						// 先读取
						size = is.read(temp_info);
						// 后写入(写入的内容多去取决于读取的内容多少)
						if (size != -1) {
							os.write(temp_info, 0, size);
						}
					} while (size != -1);
				}
			}
		} catch (Exception ex) {
			throw new Exception(ex);
		} finally {
			if (os != null)
				os.close();
			if (is != null)
				is.close();
		}
	}

	// 根据原有的文件名称,获取新的文件名称
	public static String getNewFileName(String oldFileName) {
		// oldname :上机作业.docx newnmae : 上机作业_20180222.docx;
		// 根据原有的文件名称获取文件名字和文件的类型
		int index = oldFileName.lastIndexOf(".");
		if (index == -1) {
			return oldFileName;
		}
		// 获取文件名: 上机作业
		String name = oldFileName.substring(0, index);
		// 获取文件类型: docx
		String type = oldFileName.substring(index   1);
		SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddhhmmss");
		Date date = new Date();
		String formatstr = sdf.format(date);
		// 上机作业_20180222.docx;
		return name   "_"   formatstr   "."   type;
	}

使用对象输入输出流实现对象的序列化和反序列化

代码语言:javascript复制
public static void testObjectInputStream() throws Exception{
		File file = new File("e:/a/stu.txt");
		InputStream is = new FileInputStream(file);
		ObjectInputStream ois = new ObjectInputStream(is);
		List<Student>  stuList= (List<Student>)ois.readObject(); 
		ois.close();
		is.close();
		
		for(Student stu: stuList){
			System.out.println(stu);
		}
	}
	//对象序列化(实现JAVA对象的保存和读取 ObjectInputStream)
	public static void testObjectOutputStream()  throws Exception{
		Student stu3 = new Student("赵丽丽","女","河北廊坊");
		stuList.add(stu3);
		//把集合保存到文件中 
		File file = new File("e:/a/stu.txt");
		OutputStream os = new FileOutputStream(file);
		ObjectOutputStream oos = new ObjectOutputStream(os);
		oos.writeObject(stuList);
		oos.flush();
		oos.close();
		os.close();
		
	}

字符流的读取和写入(只能读取字符文件的信息)

代码语言:javascript复制
//使用字符流读取文件
	public static void testReader() throws Exception{
		//创建文件对象
		File file = new File("e:/a/file.txt");
		//创建字符输入流对象
		Reader reader = new FileReader(file);
		//创建字符输出流对象
		File file2 = new File("e:/a/b/file_3.txt");
		Writer writer = new FileWriter(file2);
		char[] temp = new char[1000];
		int size =0;
		//通过循环边度编写
		do{
			size= reader.read(temp);  //读取字符内容
			if(size!=-1)
			{
				writer.write(temp, 0, size);
			}
		}while(size!=-1);
		//关闭
		writer.flush();
		writer.close();
		reader.close();
	}

释放资源的新方法

代码语言:javascript复制
//新的关闭资源的方法
		//try(  定义必须要关闭的对象; ){}catch(Exception ex){}; 
		// 创建输出流对象
		try(	OutputStream os = new FileOutputStream(FILE_PATH);
				ObjectOutputStream oos = new ObjectOutputStream(os); ) {

			oos.writeObject(stuList);
		} catch (Exception e) {
			throw new Exception(e);
		} 

4. 线程的状态

初始状态(创建了对象) 可运行状态(调用了start方法) 运行状态(调用了run方法) 阻塞状态 (调用了seleep,join,wait方法) 终结状态 (运行完毕)

5. 线程的常用方法

Start() 启动方法 表示线程进入了可运行状态 Seleep(int) 随眠方法,当前线程进入阻塞状态,在一定时间以后苏醒 Join() 方法 被调用者优先执行,执行完毕后当前线程在执行(阻塞的的是当前线程) Wait() 方法 当前线程进入阻塞状态,一直到对方唤醒自己才可以继续执行 notifyAll()唤醒被阻塞的线程 需要注意:wait(),notify()两个方法是Object类提供的

代码语言:javascript复制
package com.xja.sxnd.filedemo;
public class ThreadDemo {

	public static void main(String[] args) {
		MyThread mt = new MyThread();
		MyRunable mr = new MyRunable();
		Thread thread = new Thread(mr);
		mt.start();
		thread.start();
		for(int i = 0;i< 50;i  ){
			if(i == 20){
				try{
					//主线程执行到20的时候,先让第一个线程执行完毕,然后主线程在接着执行
				mt.join(); 
				}catch(Exception ex){
					ex.printStackTrace();
				}
			}
			System.out.println("主线程中的内容:"  i);
		}	
	}
}
class MyThread extends Thread
{ 
	@Override
	public void run() {
		 for(int i =0;i< 50;i  ){
			 try{
			 if(i==20)
				 Thread.sleep(500); //停止运行0.5秒,时间到了以后自动执行
			 }catch(Exception ex){
				 ex.printStackTrace();
			 }
			 System.out.println("第一个线程的对象:"  i);
		 }
	}
}
class MyRunable implements Runnable{
	@Override
	public void run() {
		 for(int i =0;i< 50;i  ){
			 System.out.println("第二个线程的对象:" i);
		 }
	}	
}

.

6. 线程的死锁

如果两个线程同时调用对方线程的JOIN方法,互相希望对方先运行就会出现死锁的问题

代码语言:javascript复制
public class MyRunnable implements Runnable {
	public Thread myThread;
	@Override
	public void run() {
		try {
			for (int i = 0; i < 100; i  ) { 
				Thread.sleep(10); 
				if(i == 5){
					System.out.println("MyRunnable运行到了5,也休息,让Thread先走");
					myThread.join();
				}
				System.out.println("第二个线程:"   i);
			}
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
	} 
}

public class MyThread extends Thread {
	public Thread myRun;
	@Override
	public void run() {
		try {
			for (int i = 0; i < 10; i  ) { 
				Thread.sleep(10); 
				
				if(i==5){
					System.out.println("Thread 运行到了5 就需要对方先走");
					myRun.join();
				}
				System.out.println("第一个线程的内容:"   i);
			}
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
	} 
}

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		try { 
			// 创建一个线程对象
			MyThread myThread = new MyThread();
			// 创建第二个线程
			MyRunnable myRunn = new MyRunnable(); 
			Thread thread = new Thread(myRunn); 
			// 启动线程 启动线程不能直接调用run 只能调用start 
			myThread.myRun = thread;
			myRunn.myThread = myThread;
			 
			myThread.start();
			thread.start();
}

7. 线程的同步

StringBuilder和StringBuffer ArrayList 和vector HashMap 和HashTable

当两个线程同时操作同一个对象的时候,因为两个线程互相影响对方的结果,导致数据不统一,这种现象称之为线程不同步 例如:银行卡存取钱的问题。(两种方式操作一个账户)

代码语言:javascript复制
public class Card {
	private int balance =500;
	public void takeMoney(int money){
		if(balance>money){
			System.out.println("取钱之前的余额:"   balance); 
			int temp_balance = balance - money;
			balance = temp_balance;
			System.out.println("取钱之后的余额:"   balance);
		}
	}
	public void saveMoney(int money){
		System.out.println("存钱之前的余额:"   balance); 
		int temp_balance = balance   money;
		balance = temp_balance;
		System.out.println("存钱之后的余额:"   balance);	
	}
}
 
public class ThreadDemo {
	public static void main(String[] args) {
		Card card = new Card();
		MyThread mt = new MyThread(card);
		MyRunable mr = new MyRunable(card);
		Thread thread = new Thread(mr);
		mt.start();
		thread.start();
		System.out.println("程序结束");
	}
}
class MyThread extends Thread
{ 
	private Card card;
	public MyThread(Card card){
		this.card = card;
	}
	@Override
	public void run() { 
		 for(int i =0;i< 10;i  ){ 
			 card.takeMoney(50);
		 } 
	} 
}
class MyRunable implements Runnable{
	private Card card;
	public MyRunable(Card card){
		this.card = card;
	}
	@Override
	public void run() { 
		 for(int i =0;i< 10;i  ){ 
			card.saveMoney(50);
		 } 
	}	
}

通过synchronized关键字修饰方法就是线程同步的方法,线程同步的方法要求同一个对象同时只能调用一个方法

代码语言:javascript复制
public class Card {
	private int balance = 500;
	public void takeMoney(int money) {
		synchronized (this) {
			if (balance > money) {
				System.out.println("取钱之前的余额:"   balance);
				int temp_balance = balance - money;
				balance = temp_balance;
				System.out.println("取钱之后的余额:"   balance);
			}
		}
	}

	public void saveMoney(int money) {
		synchronized (this) {
			System.out.println("存钱之前的余额:"   balance);
			int temp_balance = balance   money;
			balance = temp_balance;
			System.out.println("存钱之后的余额:"   balance);
		}
	}
}

8. 线程的经典案例(生产者和消费者)

代码语言:javascript复制
public class Card {
	private int balance = 500;

	public synchronized void takeMoney(int money) throws Exception {
		if (balance -money<0) {
			System.out.println("家里没有钱了,需要减少消费");
			this.wait();
			return;
		}
		System.out.println("取钱之前的余额:"   balance);
		int temp_balance = balance - money;
		balance = temp_balance;
		System.out.println("取钱之后的余额:"   balance);

		// 一旦我花费了钱,就可以让生成者继续生成
		this.notify();
	}

	public synchronized void saveMoney(int money) throws Exception {
		// 如果挣钱的足够,则需要休息,让消费者消费
		if (balance > 700) {
			System.out.println("钱以够多了,可以休息一下");
			this.wait();
			return;
		}
		System.out.println("存钱之前的余额:"   balance);
		int temp_balance = balance   money;
		balance = temp_balance;
		System.out.println("存钱之后的余额:"   balance);
		// 一旦有钱就可以让消费者消费
		this.notifyAll();
	}
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

0 人点赞