19 异常&IO(File类)

19.01_异常(异常的概述和分类)

  • A:异常的概述
    • 异常就是Java程序在运行过程中出现的错误。
  • B:异常的分类
    • 通过API查看Throwable,异常最顶层的
    • Error
      • 服务器宕机,数据库崩溃等
    • Exception
      C:异常的继承体系
    • Throwable
      • Error
      • Exception
        • RuntimeException(运行时异常,特殊,其他都是编译时异常)

19.02_异常(JVM默认是如何处理异常的)

  • A:JVM默认是如何处理异常的
    • main函数收到这个问题时,有两种处理方式:
    • a:自己将该问题处理,然后继续运行
    • b:自己没有针对的处理方式,只有交给调用main的jvm来处理
    • jvm有一个默认的异常处理机制,就将该异常进行处理.
    • 并将该异常的名称,异常的信息.异常出现的位置打印在了控制台上,同时将程序停止运行
  • B:案例演示
    • JVM默认如何处理异常

19.03_异常(try…catch的方式处理异常1)

  • A:异常处理的两种方式
    • a:try…catch…finally
      • try catch
      • try catch finally
      • try finally
    • b:throws
  • B:try…catch处理异常的基本格式
    • try…catch…finally
  • C:案例演示
    • try…catch的方式处理1个异常

try:用来检测异常的
catch:用来捕获异常的
finally:释放资源

	public static void main(String[] args) {
		Demo2 d = new Demo2();
		try{
			int x = d.div(10, 0);
			System.out.println(x);
		}catch(ArithmeticException a) {		//可能出现的异常类型,ArithmeticException a = new ArithmeticException();
			System.out.println("出错了,除数为零了");
		}
		
		System.out.println("1111111111111111");//当通过trycatch将问题处理了,程序会继续执行
	}

}

class Demo2 {
	/*
	 * 除法运算
	 */
	public int div(int a,int b) {		//a = 10,b = 0
		return a / b;					// 10 / 0  被除数是10,除数是0当除数是0的时候违背了算数运算法则,抛出异常
										//new ArithmeticException("/ by zero");
	}

19.04_异常(try…catch的方式处理异常2)

  • A:案例演示
    • try…catch的方式处理多个异常
  • 安卓,客户端开发,如何处理异常?try{}catch(Exception e){}
    ee,服务端开发,一般都是底层开发,从底层向上抛,需要多写几个catch
    try后面如果跟多个catch,那么小的异常放前面,大的异常放后面,根据多态的原理,如果大的放前面,就会将所有的子类对象接收
    后面的catch就没有意义了
public static void demo1() {
		int a = 10;
		int b = 0;
		int[] arr = {11,22,33,44,55};
		
		try {
			System.out.println(a / b);
			System.out.println(arr[10]);
			arr = null;
			System.out.println(arr[0]);
		} catch (ArithmeticException e) {
			System.out.println("除数不能为零");
		} catch (ArrayIndexOutOfBoundsException e) {
			System.out.println("索引越界了");
		} catch (Exception e) {				//Exception e = new NullPointerException(); 父类引用指向子类对象,
			System.out.println("出错了");
		}
		
		System.out.println("over");
	}
  • JDK7以后处理多个异常的方式及注意事项
public static void main(String[] args) {
		int a = 10;
		int b = 0;
		int[] arr = {11,22,33,44,55};
		
		//JDK7如何处理多个异常
		try {
			System.out.println(a / b);
			System.out.println(arr[10]);
		} catch (ArithmeticException | ArrayIndexOutOfBoundsException e) {   //或    不好用,知道就行了
			System.out.println("出错了");
		} 

19.05_异常(编译期异常和运行期异常的区别)

  • A:编译期异常和运行期异常的区别
    • Java中的异常被分为两大类:编译时异常和运行时异常。

    • 所有的RuntimeException类及其子类的实例被称为运行时异常,其他的异常就是编译时异常

    • 编译时异常

      • Java程序必须显示处理,否则程序就会发生错误,无法通过编译
    • 运行时异常

      • 无需显示处理,也可以和编译时异常一样处理
  • B:案例演示
    • 编译期异常和运行期异常的区别

19.06_异常(Throwable的几个常见方法)

  • A:Throwable的几个常见方法
    • a:getMessage()
      • 获取异常信息,返回字符串。
    • b:toString()
      • 获取异常类名和异常信息,返回字符串。
    • c:printStackTrace()
      • 获取异常类名和异常信息,以及异常出现在程序中的位置。返回值void。
  • B:案例演示
    • Throwable的几个常见方法的基本使用
	public static void main(String[] args) {
		try {
			System.out.println(1/0);
		} catch (Exception e) {			//Exception e = new ArithmeticException("/ by zero");
			//System.out.println(e.getMessage());		//获取异常信息
			//System.out.println(e); 		//调用toString方法,打印异常类名和异常信息
			e.printStackTrace();		//jvm默认就用这种方式处理异常 
		}
	}

19.07_异常(throws的方式处理异常)

  • A:throws的方式处理异常
    • 定义功能方法时,需要把出现的问题暴露出来让调用者去处理。
    • 那么就通过throws在方法上标识。
  • B:案例演示
    • 举例分别演示编译时异常和运行时异常的抛出
    • 编译时异常的抛出必须对其进行处理
    • 运行时异常的抛出可以处理也可以不处理
public static void main(String[] args) throws Exception {  //主函数中也要进行声明
		Person p = new Person();
		p.setAge(-17);
		System.out.println(p.getAge());
	}

}

class Person {
	private String name;
	private int age;
	public Person() {
		super();
		
	}
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) throws Exception {    //抛出异常时要在方法上进行声明
		if(age >0 && age <= 150) {
			this.age = age;
		}else {
			throw new Exception("年龄非法");
		}
	}
	
	
}

19.08_异常(throw的概述以及和throws的区别)

  • A:throw的概述
    • 在功能方法内部出现某种情况,程序不能继续运行,需要进行跳转时,就用throw把异常对象抛出。
  • B:案例演示
    • 分别演示编译时异常对象和运行时异常对象的抛出
  • C:throws和throw的区别
    • a:throws
      • 用在方法声明后面,跟的是异常类名
      • 可以跟多个异常类名,用逗号隔开
      • 表示抛出异常,由该方法的调用者来处理
    • b:throw
      • 用在方法体内,跟的是异常对象名
      • 只能抛出一个异常对象名
      • 表示抛出异常,由方法体内的语句处理
	        //Exception e = new Exception("年龄非法");
			//throw e;
			
			throw new AgeOutOfBoundsException("年龄非法");

throw两种写法

19.09_异常(finally关键字的特点及作用)

  • A:finally的特点
    • 被finally控制的语句体一定会执行
    • 特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))
  • B:finally的作用
    • 用于释放资源,在IO流操作和数据库操作中会见到
  • C:案例演示
    • finally关键字的特点及作用

19.10_异常(finally关键字的面试题)

  • A:面试题1

    • final,finally和finalize的区别

    final可以修饰类,不能被继承
    修饰方法,不能被重写
    修饰变量,只能赋值一次

    finally是try语句中的一个语句体,不能单独使用,用来释放资源

    finalize是一个方法,当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。

  • B:面试题2

    • 如果catch里面有return语句,请问finally的代码还会执行吗?如果会,请问是在return前还是return后。
public static void main(String[] args) {
		Demo d = new Demo();
		System.out.println(d.method());//30
	}

}

class Demo {
	public int method() {
		int x = 10;
		try {
			x = 20;
			System.out.println(1/0);
			return x;
		} catch (Exception e) {
			x = 30;
			return x;    //执行return时已经建立好了返回路径。即使执行了finally,x的返回值不会再改变
		} finally {
			x = 40;
			//return x;					千万不要在finally里面写返回语句,因为finally的作用是为了释放资源,是肯定会执行的
										//如果在这里面写返回语句,那么try和catch的返回值都会被改变
		}
	}

先执行finally再执行return

19.11_异常(自定义异常概述和基本使用)

  • A:为什么需要自定义异常(更准确的定位到错误)
    • 举例:人的年龄
  • B:自定义异常概述
    • 继承自Exception
    • 继承自RuntimeException
  • C:案例演示
    • 自定义异常的基本使用
class AgeOutOfBoundsException extends Exception {

	public AgeOutOfBoundsException() {
		super();
		
	}

	public AgeOutOfBoundsException(String message) {
		super(message);
		
	}
public void setAge(int age) throws AgeOutOfBoundsException {
		if(age >0 && age <= 150) {
			this.age = age;
		}else {
			throw new AgeOutOfBoundsException("年龄非法");
		}
	}

19.12_异常(异常的注意事项及如何使用异常处理)

  • A:异常注意事项
    • a:子类重写父类方法时,子类的方法必须抛出相同的异常或父类异常的子类。(父亲坏了,儿子不能比父亲更坏)
    • b:如果父类抛出了多个异常,子类重写父类时,只能抛出相同的异常或者是他的子集,子类不能抛出父类没有的异常
    • c:如果被重写的方法没有异常抛出,那么子类的方法绝对不可以抛出异常,如果子类方法内有异常发生,那么子类只能try,不能throws
  • B:如何使用异常处理
    • 原则:如果该功能内部可以将问题处理,用try,如果处理不了,交由调用者处理,这是用throws

    • 区别:

      • 后续程序需要继续运行就try
      • 后续程序不需要继续运行就throws
    • 如果JDK没有提供对应的异常,需要自定义异常。

19.13_异常(练习)

  • 键盘录入一个int类型的整数,对其求二进制表现形式
    • 如果录入的整数过大,给予提示,录入的整数过大请重新录入一个整数BigInteger
    • 如果录入的是小数,给予提示,录入的是小数,请重新录入一个整数
    • 如果录入的是其他字符,给予提示,录入的是非法字符,请重新录入一个整数
 * 分析:
	 * 1,创建键盘录入对象
	 * 2,将键盘录入的结果存储在String类型的字符串中,存储int类型中如果有不符合条件的直接报错,无法进行后续判断
	 * 3,键盘录入的结果转换成int类型的数据,是正确的还是错误的
	 * 4,正确的直接转换
	 * 5,错误的要进行对应判断
	 
	public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		System.out.println("请输入一个整数:");
		
		
		while(true) {
			String line = sc.nextLine();					//将键盘录入的结果存储在line中
			try {
				int num = Integer.parseInt(line);				//将字符串转换为整数
				System.out.println(Integer.toBinaryString(num));//将整数转换为二进制
				break;											//跳出循环
			}catch(Exception e) {
				try {
					new BigInteger(line);
					System.out.println("录入错误,您录入的是一个过大整数,请重新输入一个整数:");
				}catch (Exception e2) {							//alt + shif + z (try catch快捷键)
					try {
						new BigDecimal(line);
						System.out.println("录入错误,您录入的是一个小数,请重新输入一个整数:");
					} catch (Exception e1) {
						System.out.println("录入错误,您录入的是非法字符,请重新输入一个整数:");
					}
				}
				
			}
		}
		
		
	}

19.14_File类(File类的概述和构造方法)

  • A:File类的概述
    • File更应该叫做一个路径
      • 文件路径或者文件夹路径
      • 路径分为绝对路径和相对路径
      • 绝对路径是一个固定的路径,从盘符开始
      • 相对路径相对于某个位置,在eclipse下是指当前项目下,在dos下
    • 查看API指的是当前路径
    • 文件和目录路径名的抽象表示形式
  • B:构造方法
    • File(String pathname):根据一个路径得到File对象
public static void demo1() {
		File file = new File("F:\\双元课堂\\day19\\video\\001_今日内容.avi");//要加后缀
		System.out.println(file.exists());//true
		
		File file2 = new File("xxx.txt");
		System.out.println(file2.exists());//true
		
		File file3 = new File("yyy.txt");
		System.out.println(file3.exists());//false
	}
  • File(String parent, String child):根据一个目录和一个子文件/目录得到File对象
public static void demo2() {
		String parent = "F:\\双元课堂\\day19\\video";
		String child = "001_今日内容.avi";
		File file = new File(parent,child);
		System.out.println(file.exists());//true
	}
  • File(File parent, String child):根据一个父File对象和一个子文件/目录得到File对象
public static void main(String[] args) {
		File parent = new File("F:\\双元课堂\\day19\\video");//将父级路径封装成对象
		String child = "001_今日内容.avi";
		File file = new File(parent, child);
		System.out.println(file.exists());//true
		System.out.println(parent.exists());//true
	}
  • C:案例演示
    • File类的构造方法

19.15_File类(File类的创建功能)

  • A:创建功能
    • public boolean createNewFile():创建文件 如果存在这样的文件,就不创建了
    • public boolean mkdir():创建文件夹 如果存在这样的文件夹,就不创建了
    • public boolean mkdirs():创建文件夹,如果父文件夹不存在,会帮你创建出来
  • B:案例演示
    • File类的创建功能

    • 注意事项:

      • 如果你创建文件或者文件夹忘了写盘符路径,那么,默认在项目路径下。
public static void main(String[] args) throws IOException {
		//demo1();
		File dir1 = new File("aaa");   //文件夹创建在项目下
		System.out.println(dir1.mkdir());  //创建成功返回true
		
		File dir2 = new File("bbb.txt");				//这样写是可以的,文件夹也是可以有后缀的
		System.out.println(dir2.mkdir());
		
		File dir3 = new File("ccc\\ddd");  
		System.out.println(dir3.mkdirs()); 				//创建多级目录
	}

	public static void demo1() throws IOException {
		File file = new File("yyy.txt");
		System.out.println(file.createNewFile());			//如果没有就创建,返回true,如果有就不创建,返回false
		
		File file2 = new File("zzz");
		System.out.println(file2.createNewFile());//不加后缀也能创建
	}

19.16_File类(File类的重命名和删除功能)

  • A:重命名和删除功能
    • public boolean renameTo(File dest):把文件重命名为指定的文件路径
public static void demo1() {
		File file1 = new File("ooo.txt");
		File file2 = new File("D:\\xxx.txt");
		System.out.println(file1.renameTo(file2));
	}
  • public boolean delete():删除文件或者文件夹
public static void main(String[] args) {
		File file1 = new File("yyy.txt");
		System.out.println(file1.delete());
		
		File file2 = new File("aaa");
		System.out.println(file2.delete());
		
		File file3 = new File("ccc");					//如果删除一个文件夹,那么文件夹必须是空的
		System.out.println(file3.delete());
	}
  • B:重命名注意事项
    • 如果路径名相同,就是改名。
    • 如果路径名不同,就是改名并剪切。
  • C:删除注意事项:
    • Java中的删除不走回收站。
    • 要删除一个文件夹,请注意该文件夹内不能包含文件或者文件夹

19.17_File类(File类的判断功能)

  • A:判断功能
    • public boolean isDirectory():判断是否是目录
    • public boolean isFile():判断是否是文件
    • public boolean exists():判断是否存在
public static void demo1() {
		File dir1 = new File("ccc");
		System.out.println(dir1.isDirectory());			//判断是否是文件夹
		
		File dir2 = new File("zzz");
		System.out.println(dir2.isDirectory());
		
		System.out.println(dir1.isFile());				//判断是否是文件
		System.out.println(dir2.isFile());
	}
  • public boolean canRead():判断是否可读
  • public boolean canWrite():判断是否可写
  • public boolean isHidden():判断是否隐藏
public static void main(String[] args) {
		//demo1();
		File file = new File("zzz");
		file.setReadable(false);  //设置可读性,但是windows系统没意义
		System.out.println(file.canRead()); 			//windows系统认为所有的文件都是可读的
		file.setWritable(true);  //设置可写性
		System.out.println(file.canWrite()); 			//windows系统可以设置为不可写
		
		File file2 = new File("aaa.txt");
		System.out.println(file2.isHidden());			//判断是否是隐藏文件
		System.out.println(file.isHidden());
	}
  • B:案例演示
    • File类的判断功能

19.18_File类(File类的获取功能)

  • A:获取功能
    • public String getAbsolutePath():获取绝对路径
    • public String getPath():获取构造方法中传入的路径
    • public String getName():获取文件或文件夹名称
    • public long length():获取文件总内容的长度。字节数
    • public long lastModified():获取最后一次的修改时间,毫秒值
    • public String[] list():获取指定目录下的所有文件或者文件夹的名称数组
    • public File[] listFiles():获取指定目录下的所有文件或者文件夹的File数组
  • B:案例演示
    • File类的获取功能
public static void main(String[] args) {
		//demo1();
		File dir = new File("F:/双元课堂/day19/video");
		String[] arr = dir.list();						//仅为了获取文件名
		
		for (String string : arr) {
			System.out.println(string);
		}
		
		File[] subFiles = dir.listFiles();
		
		for (File file : subFiles) {					//获取文件对象
			System.out.println(file);
		}
 	}

	public static void demo1() {
		File file1 = new File("ccc.txt");
		File file2 = new File("D:\\双元课堂\\day19\\ccc.txt");
		//System.out.println(file1.getAbsolutePath());			//获取绝对路径
		//System.out.println(file2.getAbsolutePath());
		
		//System.out.println(file1.getPath());					//获取构造方法中传入路径
		//System.out.println(file2.getPath());
		
//		System.out.println(file1.getName());					//获取文件或者文件的名字
//		System.out.println(file2.getName());
//		
//		System.out.println(file1.length());
		
		Date d = new Date(file1.lastModified());				//文件的最后修改时间
		SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
		System.out.println(sdf.format(d));
	}

19.19_File类(输出指定目录下指定后缀的文件名)

  • A:案例演示
    • 需求:判断E盘目录下是否有后缀名为.jpg的文件,如果有,就输出该文件名称
public static void main(String[] args) {
		File dir = new File("E:\\");
		
//		String[] arr = dir.list();					//获取e盘下所有的文件或文件夹
//		for (String string : arr) {
//			if(string.endsWith(".jpg")) {
//				System.out.println(string);
//			}
//		}
		
		File[] subFiles = dir.listFiles();			//获取e盘下所有的文件或文件夹对象
		
		for (File subFile : subFiles) {
			if(subFile.isFile() && subFile.getName().endsWith(".jpg")) {
				System.out.println(subFile);
			}
		}
	}

19.20_File类(文件名称过滤器的概述及使用)

  • A:文件名称过滤器的概述
    • public String[] list(FilenameFilter filter)
    • public File[] listFiles(FileFilter filter)
  • B:文件名称过滤器的使用
    • 需求:判断E盘目录下是否有后缀名为.jpg的文件,如果有,就输出该文件名称
  • C:源码分析
    • 带文件名称过滤器的list()方法的源码
	public static void main(String[] args) {
		File dir = new File("E:\\");
		String[] arr = dir.list(new FilenameFilter() {
			
			@Override
			public boolean accept(File dir, String name) {
				//System.out.println(dir);
				//System.out.println(name);
				File file = new File(dir, name);
				return file.isFile() && file.getName().endsWith(".jpg");
			}
		});
		
		for (String string : arr) {
			System.out.println(string);
		}
	}

猜你喜欢

转载自blog.csdn.net/chenchaozzu/article/details/84111460