Java:日期+异常

一、日期

创建日期对象

package date;

//导入Date类
import java.util.Date;

public class TestDate {

	public static void main(String[] args) {

		// 当前时间
		Date d1 = new Date();
		System.out.println("当前时间:");
		System.out.println(d1);
		System.out.println();
		// 从1970年1月1日 早上8点0分0秒 开始经历的毫秒数
		Date d2 = new Date(5000);
		System.out.println("从1970年1月1日 早上8点0分0秒 开始经历了5秒的时间");
		System.out.println(d2);//Thu Jan 01 08:00:05 CST 1970

	}
}

(1)getTime得到一个long型的整数

这个整数代表 从1970.1.1 08:00:00:000 开始 每经历一毫秒,增加1
直接打印对象,会看到 “Tue Jan 05 09:51:48 CST 2016” 这样的格式

package date;
 
//
import java.util.Date;
 
public class TestDate {
 
    public static void main(String[] args) {
        //注意:是java.util.Date;
        //而非 java.sql.Date,此类是给数据库访问的时候使用的
        Date now= new Date();
        //打印当前时间
        System.out.println("当前时间:"+now.toString());
        //getTime() 得到一个long型的整数
        //这个整数代表 1970.1.1 08:00:00:000,每经历一毫秒,增加1
        //1464749767878
        System.out.println("当前时间getTime()返回的值是:"+now.getTime());
         
        Date zero = new Date(0);
        System.out.println("用0作为构造方法,得到的日期是时间原点:"+zero);
         
    }
}

(2)格式化 :format

package date;
 
//
import java.text.SimpleDateFormat;
import java.util.Date;
 
public class TestDate {
 
    public static void main(String[] args) {
         
        //y 代表年
        //M 代表月
        //d 代表日
        //H 代表24进制的小时
        //h 代表12进制的小时
        //m 代表分钟
        //s 代表秒
        //S 代表毫秒
        SimpleDateFormat sdf =new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS" );
        Date d= new Date();
        String str = sdf.format(d);
        System.out.println("当前时间通过 yyyy-MM-dd HH:mm:ss SSS 格式化后的输出: "+str);
        
        SimpleDateFormat sdf1 =new SimpleDateFormat("yyyy-MM-dd" );
        Date d1= new Date();
        String str1 = sdf1.format(d1);
        System.out.println("当前时间通过 yyyy-MM-dd 格式化后的输出: "+str1);
        
    }
}

(3) 模式字符串转日期parse

模式(yyyy/MM/dd HH:mm:ss)需要和字符串格式保持一致,如果不一样就会抛出解析异常ParseException

package date;
 
//
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
 
public class TestDate {
 
    public static void main(String[] args) {
        SimpleDateFormat sdf =new SimpleDateFormat("yyyy/MM/dd HH:mm:ss" );
 
        String str = "2016/1/5 12:12:12";
         
        try {
            Date d = sdf.parse(str);
            System.out.printf("字符串 %s 通过格式  yyyy/MM/dd HH:mm:ss %n转换为日期对象: %s",str,d.toString());
        } catch (ParseException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
         
    }
}

(4)★练习:待学

二、异常

Throwable是类,Exception和Error都继承了该类
所以在捕捉的时候,也可以使用Throwable进行捕捉
如图: 异常分Error和Exception
Exception里又分运行时异常和可查异常。
在这里插入图片描述
常见异常类型
在这里插入图片描述

(1)文件异常

比如要打开d盘的LOL.exe文件,这个文件是有可能不存在的
Java中通过 new FileInputStream(f) 试图打开某文件,就有可能抛出文件不存在异常FileNotFoundException (空指针异常、类型转换异常、数组越界异常、系统找不到指定文件异常)
如果不处理该异常,就会有编译错误

package exception;
 
import java.io.File;
import java.io.FileInputStream;
 
public class TestException {
 
    public static void main(String[] args) {
         
        File f= new File("d:/LOL.exe");
         
        //试图打开文件LOL.exe,会抛出FileNotFoundException,如果不处理该异常,就会有编译错误
        new FileInputStream(f);

(2)父类捕捉子类异常

FileNotFoundException是Exception的子类,使用Exception也可以catch住FileNotFoundException

package exception;
 
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
 
public class TestException {
 
    public static void main(String[] args) {
         
        File f= new File("d:/LOL.exe");
         
        try{
            System.out.println("试图打开 d:/LOL.exe");
            new FileInputStream(f);
            System.out.println("成功打开");
        }
        
        catch(Exception e){
            System.out.println("d:/LOL.exe不存在");
            e.printStackTrace();
        }
         
    }
}

(3)同时处理多异常

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class TestException {

	public static void main(String[] args) {

		File f = new File("d:/LOL.exe");

		try {
			System.out.println("试图打开 d:/LOL.exe");
			new FileInputStream(f);
			System.out.println("成功打开");
			SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
			Date d = sdf.parse("2016-06-03");
		} catch (FileNotFoundException e) {
			System.out.println("d:/LOL.exe不存在");
			e.printStackTrace();
		} catch (ParseException e) {
			System.out.println("日期格式解析错误");
			e.printStackTrace();
		}
	}
}

另一个种办法是把多个异常,放在一个catch里统一捕捉
这种方式从 JDK7开始支持,好处是捕捉的代码更紧凑,不足之处是,一旦发生异常,不能确定到底是哪种异常,需要通过instanceof 进行判断具体的异常类型

catch (FileNotFoundException | ParseException e) {
if (e instanceof FileNotFoundException)
                System.out.println("d:/LOL.exe不存在");
if (e instanceof ParseException)
                System.out.println("日期格式解析错误");
 
 e.printStackTrace();
}

(4)finally

无论有没有异常都会执行

finally{
            System.out.println("无论文件是否存在, 都会执行的代码");
        }

(5)throw与throws

考虑如下情况:
主方法调用method1
method1调用method2
method2中打开文件

method2中需要进行异常处理
但是method2不打算处理,而是把这个异常通过throws抛出去
那么method1就会接到该异常。 处理办法也是两种,要么是try catch处理掉,要么也是抛出去。
method1选择本地try catch住 一旦try catch住了,就相当于把这个异常消化掉了,主方法在调用method1的时候,就不需要进行异常处理了

package exception;

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

public class TestException {

	public static void main(String[] args) {
		method1();

	}

	private static void method1() {
		try {
			method2();
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

	}

	private static void method2() throws FileNotFoundException {

		File f = new File("d:/LOL.exe");

		System.out.println("试图打开 d:/LOL.exe");
		new FileInputStream(f);
		System.out.println("成功打开");

	}
}

throws与throw这两个关键字接近,不过意义不一样,有如下区别:

  1. throws 出现在方法声明上,而throw通常都出现在方法体内。
  2. throws 表示出现异常的一种可能性,并不一定会发生这些异常;throw则是抛出了异常,执行throw则一定抛出了某个异常对象。
  3. 如果保护代码中发生异常,异常被抛给第一个 catch 块。
    如果抛出异常的数据类型与 ExceptionType1 匹配,它在这里就会被捕获。
    如果不匹配,它会被传递给第二个 catch 块。
    如此,直到异常被捕获或者通过所有的 catch 块。
    也就是说当它被捕获了就不会在抛出去了。

(6)可查异常 CheckedException

可查异常: CheckedException
可查异常即必须进行处理的异常,要么try catch住,要么往外抛,谁调用,谁处理,比如 FileNotFoundException
如果不处理,编译器,就不让你通过

(7)运行时异常RuntimeException

运行时异常RuntimeException指: 不是必须进行try catch的异常
常见运行时异常:
除数不能为0异常:ArithmeticException
下标越界异常:ArrayIndexOutOfBoundsException
空指针异常:NullPointerException
在编写代码的时候,依然可以使用try catch throws进行处理,与可查异常不同之处在于,即便不进行try catch,也不会有编译错误
Java之所以会设计运行时异常的原因之一,是因为下标越界,空指针这些运行时异常太过于普遍,如果都需要进行捕捉,代码的可读性就会变得很糟糕。

package exception;
 
public class TestException {
 
    public static void main(String[] args) {
    	
    	//任何除数不能为0:ArithmeticException
    	int k = 5/0; 
    	
    	//下标越界异常:ArrayIndexOutOfBoundsException
    	int j[] = new int[5];
    	j[10] = 10;
    	
    	//空指针异常:NullPointerException
    	String str = null;
    	str.length();
   }
}

(8)错误Error

指的是系统级别的异常,通常是内存用光了
在默认设置下,一般java程序启动的时候,最大可以使用16m的内存
如例不停的给StringBuffer追加字符,很快就把内存使用光了。抛出OutOfMemoryError
与运行时异常一样,错误也是不要求强制捕捉的

package exception;
 
public class TestException {
 
    public static void main(String[] args) {
    
    	StringBuffer sb =new StringBuffer();
    	
    	for (int i = 0; i < Integer.MAX_VALUE; i++) {
			sb.append('a');
		}
    	
    }

(9)错误能否捕获、运行时异常与非运行时异常的区别

RuntimeException如果使用try和catch异常可以被捕捉。
错误Error 不能够被捕捉
运行时异常与非运行时异常的区别:

  • Throwable 是所有 Java 程序中错误处理的父类 ,有两种子类: Error 和 Exception 。

  • Error :表示由 JVM 所侦测到的无法预期的错误,由于这是属于 JVM 层次的严重错误 ,导致 JVM 无法继续执行,因此,这是不可捕捉到的,无法采取任何恢复的操作,顶多只能显示错误信息。

  • Exception :表示可恢复的例外,这是可捕捉到的。

  • Java 提供了两类主要的异常 :runtime exception 和 checked exception 。

  • checked 异常也就是我们经常遇到的 IO 异常,以及 SQL 异常都是这种异常。 对于这种异常, JAVA 编译器强制要求我们必需对出现的这些异常进行 catch 。所以,面对这种异常不管我们是否愿意,只能自己去写一大堆 catch
    块去处理可能的异常。

  • 但是另外一种异常: runtime exception ,也称运行时异常

我们可以不写catch处理,但运行后出现异常程序或线程终止。当出现这样的异常时,总是由虚拟机
接管。比如:我们从来没有人去处理过 NullPointerException 异常,它就是运行时异常,并且这种异常还是最常见的异常之一。
出现运行时异常后,系统会把异常一直往上层抛,一直遇到处理代码。如果没有处理块,到最上层,如果是多线程就由 Thread.run() 抛出 ,如果是单线程就被 main() 抛出 。抛出之后,如果是线程,这个线程也就退出了。如果是主程序抛出的异常,那么这整个程序也就退出了(所以才会编译器报错的提示)。

  • 运行时异常是

Exception 的子类,也有一般异常的特点,是可以被 Catch
块处理的。只不过往往我们不对他处理罢了。也就是说,你如果不对运行时异常进行处理,那么出现运行时异常之后,要么是线程中止,要么是主程序终止。

如果不想终止,则必须扑捉所有的运行时异常,决不让这个处理线程退出。队列里面出现异常数据了,正常的处理应该是把异常数据舍弃,然后记录日志。不应该由于异常数据而影响下面对正常数据的处理。
在这个场景这样处理可能是一个比较好的应用,但并不代表在所有的场景你都应该如此。如果在其它场景,遇到了一些错误,如果退出程序比较好,这时你就可以不太理会运行时异常
,或者是通过对异常的处理显式的控制程序退出。

(10)可不可以抛出 Throwable 类

package exception;
 
import java.io.FileNotFoundException;
 
public class demo1 {
    public static void main(String[] args){
 
        try {
            throwCheck(2);//调用方法,抛出Exception(是Throwable子类,所以是允许的的)
        }catch (Throwable e){// 处理了Exception异常
            e.printStackTrace();
        }
        throwRun(-3);// 调用方法,抛出运行时异常,可以不处理
 
    }
 
    public static void throwCheck(int a) throws Throwable{
        if(a > 0){
            throw new Exception("a");
        }
    }
 
    public static void throwRun(int a){
        if(a > -1){
            throw new RuntimeException("a > -1");
        }
    }
}

(11)

package day01;
import java.util.*;


class Test1{
	  static int sum(int n) throws Exception{
		int result=0;
		if(n<0)		
		{throw new Exception("n不能为零");}
		while(n>0) {
			result=result+n;
			n--;
		}
		
		return result;
		
	}	

	public static void main(String[] args) {// 因为main方法是静态方法,自然调用的sum方法也要是静态的
		int a;
		int s;
		System.out.println("输入n的值:");
		Scanner c= new Scanner(System.in);
		a = c.nextInt();
		try {
		s=sum(a);
		System.out.println("和为:"+s);
		}catch(Exception e) {
			System.out.println(e.getMessage());
		}
	}	
	
}	                        


(12)

package ch02;  //在JAVA中,要花1/3的精力,代码来处理异常  健壮性,安全性
class Division{   //除法类
	private int num1;
	private int num2;
	public Division() {	}
	public Division(int n1,int n2) {
		num1 = n1;		num2 = n2;
	}
	public void setNum(int n1,int n2) {   //修改2个成员变量的值 
		num1 = n1;		num2 = n2;
	}
	public int div() throws Exception{    //定义除法方法    方法名后写 throws抛出异常
		if(num2 == 2) {   //用if判断条件
			throw new Exception("这个程序规定,分母不能为2");  //在if中用throw抛出异常
		}
		return num1 / num2;	}
}
//当方法中如果是出现系统规定的异常时,系统会自动抛出异常。例如:当分母为0时,系统自动抛出异常。
//假如当方法中用户自定义的某个条件下要抛出异常,则要用throws,throw抛出。
//比如:当分母为2时,抛出异常:分母不能为2   则在div方法中用throws,throw关键字抛出异常
public class UesDivision {
	private Division d;   //d没有分配内存空间
	public void doDiv() {   //求三次 两个数相除    
	 	d = new Division(2,4);   // 求2/4   
	 	try {
		System.out.println("第一次相除:"+d.div());   
	 	}catch(Exception ex) {
	 		ex.printStackTrace();
	 	}
		d.setNum(2, 0);     
		try { //试一试,尝试的意思。  将可能出现异常的语句放在try{   }的大括号中
		     System.out.println("第二次相除:"+d.div());// 求2/0 div()会抛出两个异常:一个是分为2,一个是分母为0
		}catch(ArithmeticException ex) { //捕捉的意思 catch语句要捕捉try中出现的异常,  捕捉以异常后
			//要对异常进行处理    捕捉分母为0的异常
			System.out.println(ex.getMessage()); //处理异常, 将异常信息输出
		}   //try 后面一定要有catch() 匹配。\
		catch(Exception ex) { //捕捉分母为2的异常
			System.out.println(ex.getMessage());
		} //
		d.setNum(4, 2);   
		try {
			System.out.println("第三次相除:"+d.div()); // 求4/2 
		}catch(Exception ex) { 
			System.out.println(ex.toString());
		}
		System.out.println("后面还有很多语句要运行......");
	}
	public static void main(String[] args) {
		UesDivision ud = new UesDivision();   //创建对象
		ud.doDiv();                           //调用doDiv()执行除法
	}
}

三、自定义异常

package charactor;
  
public class Student {
    public String name;
    protected int age;
 
    public void attackHero(Student a1) throws EnemyHeroIsDeadException{
        if(a1.age== 0){
            throw new EnemyHeroIsDeadException(" 自定义异常 " );
        }
    }

     //自定义异常
    class EnemyHeroIsDeadException extends Exception{
         
        public EnemyHeroIsDeadException(){
             
        }
        public EnemyHeroIsDeadException(String msg){
            super(msg);
        }
    }
      
    public static void main(String[] args) {
         
        Student a1 =  new Student();
 		a1.name="小明";
 		a1.age=45;
 
        Student a2 =  new Student();
 		a2.name="小红";
 		a2.age = 0;
        try {
            a1.attackHero(a2);
             
        } catch (EnemyHeroIsDeadException e) {
            // TODO Auto-generated catch block
            System.out.println("异常的具体原因:"+e.getMessage());
            e.printStackTrace();
        }
         
    }
}

设置一个可以判断年龄的功能的Person类,当年龄大于100或小于0需要进行异常处理
(了解获得当前时间的类Calendar)


/*
自定义异常类:int n, String msg;   给msg赋值的方法,获得msg值的方法
Person类:int birthYear  计算年龄方法  -- 抛出异常(自定义异常)
应用程序类:定义一个Person对象,调用计算年龄的方法
*/
import java.util.Calendar;
class UserException extends Exception{
       private int n;
       private String msg;
       public UserException(){    }
       public UserException(int n){
                this.n = n;
                setMsg();
       }
       public void setMsg(){
              if (n > 100){
                       msg = "年龄超过100岁了,输入的年份不正确。";
               }
               if (n < 0){
                        msg = "输入的年份大于当前时间。";
                }
       }
       public void displayMsg(){
               System.out.println(msg);
       }
}
class Person{
        private int birthYear;
        public Person(){
               
         }
         public Person(int year){
                  birthYear = year;
          }
          public void setYear(int year){
                 birthYear =year;
          }
          public int calAge()throws UserException{
                   Calendar c =Calendar.getInstance();//获得当前时间
                   //get方法的参数是一个静态常量,YEAR,则返回年,MONTH则返回月,建议看API文档
                   int age = c.get(Calendar.YEAR) - birthYear;   //用当前的年-出生的年
                   if (age > 100 || age < 0) throw new UserException(age);
                   return age;
          }
}
public class UsePerson{
       public static void main(String[] ar){
             Person p = new Person(1995);
             try {
                   System.out.println(p.calAge() +"岁");
             }
              catch(UserException ex){
                      ex.displayMsg();
              }
              p.setYear(2030);
              try {
                   System.out.println(p.calAge() +"岁");
             }
              catch(UserException ex){
                      ex.displayMsg();
              }
              p.setYear(1900);
              try {
                   System.out.println(p.calAge() +"岁");
             }
              catch(UserException ex){
                      ex.displayMsg();
              }
       }
}

习题

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值