Java基础语法回顾(二)

异常处理

异常就是运行时的代码序列引起非正常情况。Java提供的异常机制是通过异常对象来描述错误,当引起异常的情况时,就创建一个能够描述该异常的对象,并在引起错误的方法中将其抛出。针对异常情况,我们可以自己处理异常,也可以继续传递异常,但是在某一点都需要捕获并处理异常。
异常可以由Java运行时系统抛出(违反了Java语言规则),也可以是我们通过代码抛出(手动报告某些错误)。
异常关键字:try、catch、finally、throw和throws。

try{
    //可能发生异常的代码块
}catch(ExceptionType1 e1){
    //捕获该类异常,并且进行处理
}catch(ExceptionTyp2 e2){
    //捕获该类异常,并且进行处理
}finally{
    //无论异常是否发生,最后都会执行finally语句块
}

异常类型

Java中所有异常都是内置类Throwable的子类,所以Throwable是所有异常类的顶级父类。Throwable下面有两个子类,它们对应着Java两类不同的异常类型,分别是Exception和Error。
Exception是我们使用最多的异常类,我们可以使用它捕获异常,也可以通过继承它来实现自己的异常类。
Error异常一般是Java运行时系统使用,它表示运行环境本身出现了错误,但是这类异常我们是无法处理的,它们只是为了响应灾难性的失败而创建的。比如堆栈溢出就属于这类异常。

tyr-catch运行原理

如果当异常发生时,我们没有捕获所发生的异常,最终将会由默认处理程序进行处理,默认处理程序会显示一个错误描述字符串,输出异常发生点的堆栈跟踪信息并最终终止程序执行。
如果我们使用try-catch来捕获异常,它能给我们带来两点好处:

  • 允许我们根据业务修复错误。
  • 防止程序自动停止。
try{
    a = 1/0;
    System.out.println("1");
}catch(Exception e){
    System.out.println("2");
}
System.out.println("3");
//输出结果
1
3

当try中抛出异常时,程序会立即跳出try语句块,进入catch语句块,执行完catch语句块之后,会跳出整个try-catch语句,然后执行try-catch后面的语句。所以try语句块中发生异常点后面的语句是永远不会执行的。
Throwabl重写了Object类中的toString()方法,会返回包含描述异常信息的字符串,所以我们在catch中其实也可以打印出e。
当单个代码块出现多种异常时,可以使用多条catch语句块来捕获,当发生异常时会从上向下检索是否有对应的异常能够匹配,如果匹配到就会忽略下面的catch语句。所以我们要把所有子类异常放在前面,父类异常放在后面,因为父类异常能够捕获所有子类的异常。

在JDK7之后提供了使用一个catch捕获多个异常的使用方式:

try{
}catch(ArithmeticException | ArrayIndexOutOfBoundsException e){
    System.out.println(e)
}

throw和throws

我们不仅能够捕获Java系统抛出的异常,我们也可以在程序中显示的抛出异常。

throw ThrowableInstance;

ThrowInstance必须要是Throwable或其子类的实例对象。ThrowInstance对象一般通过new关键字创建,或者通过catch中的参数获取。
throw语句之后的语句都不会被执行,检查最近的try语句是否有与其匹配的catch语句,如果没有就继续向上抛。

throw new NullPointerException("demo");

大多数异常类都会至少有两个构造函数,无参构造函数和带一个参数的构造函数。第二种参数就是异常描述信息的字符串,可以通过直接打印异常获取,也可以通过getMessage()方法获取。

当我们抛出Error、RuntimeException以及子类时,如果方法本身不能处理这种异常,就需要使用throws关键字列出该方法可能抛出的异常列表。这样做的目标是使调用者能够对这种异常有准备。对于RuntimeException这类异常被称为未经检查的异常,因为编译器不检查方法是否被处理或抛出这类异常,所以不需要指定这类异常。

class A{
    static throwExcep() throws IllegalAccessException{
        throw new IllegalAccessException("a");
    }
    
    pulic static void main(String[] args){
        try{
            throwExcep();
        }catch(IllegalAccessException e){
            System.out.println(e)
        }
    }
}

除了在抛出方法上使用throws抛出异常类型,还需要调用者使用try-catch语句块来捕获这类异常。

自定义异常

我们也可以创建自己的异常类型,用来处理特定应用程序的情况。我们只需要继承Exception类或Throwable类,然后编写需要的构造方法即可,不需要实现任何方法。实际上Exception也没有实现任何方法,它的所有方法都是从Throwable继承来的。

class MyException extends Exception{
    private String msg;
    MyException(){
        super();
    }
    MyException(String msg){
        this.msg = msg;
    }
    
    public String toString(){
        return "MyException[" + msg + "]"
    }
}

链式异常

在JDK1.4之后Java增加了链式异常机制,也就是可以设置引起当前异常的异常是什么,这个链可以是任意层次的。我们可以通过构造方法添加链式异常,可以通过initCause(Throwable caseExc)来设置异常连,但是这个方法只能调用一次。

Class MyException extends Exception{
    MyException(Throwable throwExc){
        throwExc = new NullInportException("null");
    }
}

public class A{
    public staic void main(){
        b();
    }
    
    static void b(){
        NullPointerException e = new NullPointerException("top layer");
        e.initCause(new ArithmeticException("cause"));
        throw e;
    }
}

对于那些需要查看异常背后原因的情况,链式异常是完美的解决方案。

枚举

从JDK5之后Java提供对枚举的支持,枚举就是具有一系列名称的常量(可以使用final来代替使用)。枚举的本质是一种类类型,它可以有构造函数、实例变量和方法,但是不能使用new去实例化枚举。

public enum Number{
    ONE,TWO,THREE,FOUR,FIVE
}

枚举中定义的标识符ONE、TWO这些称为枚举常量,每个枚举常量被隐式的声明为Number的public、static和final的成员,也就说这些常量属于“自类型化的”,它们的类型就是定义它们的枚举。

Number num = Number.ONE;
if(num == Number.ONE){
    //判断两个枚举常量是否相等
}

枚举可以用在switch语句中,case语句是枚举常量值,不需要使用枚举类名标识,因为switch表达式已经隐式的指定了case常量的类型。

switch(num){
    case ONE:
        //...
    case TWO:
        //...
}

values()和valueOf()方法

枚举类中有两个常用的方法,就是values()和valueOf()。values()方法会以数据的形式将枚举类中的所有枚举常量值列出来。valuesOf(String str)会返回与str字符串对应的枚举常量。

public static enum-type[] values();
public static enum-type valueOf(String str);

枚举是类类型

枚举本身是类类型,所以可以为其创建构造方法、实例变量和方法,它甚至可以实现接口。

public enum Number{
    ONE(1),TWO(2),THREE(3),FOUR(4),FIVE(5)
    
    private int price;
    
    public Number(int price){
        this.price = price;
    }
    
    public int getPrice(){
        return price;
    }
}

Number num;
Number.ONE.getPrice();

枚举类虽然不能使用new关键字创建实例,但是每个枚举常量都是枚举类的实例,所以在创建每个枚举常量的时候都会调用构造方法。当我们创建一个枚举常量时,每个枚举常量值都会调用一次构造函数,因为每个枚举常量都是枚举类的一个对象,所以它们的实例变量都是一个个副本,它们之间并没有任何联系。

Enum类

枚举本身有两条限制:第一,枚举不能继承其它类;第二:枚举不能是超类。所以这就意味着枚举不能扩展。
虽然枚举不能继承其它类,但是所有枚举类都默认继承了超类java.lang.Enum,这个类中定义了所有枚举类可以使用的一些方法。比如ordinal()获取枚举的序列值(从0开始),compareTo()比较相同类型枚举值的序列号。

类型封装器和自动拆装箱

类型封装器

java出于性能考虑,将常用的类型使用基本类型来表示。基本类型不是对象层次的组成,所以它们不继承Object类。但是有许多场景需要以对象的形式来来表示基本类型,比如Java中定义的标准数据结构(比如集合类型)、不能通过引用类型为方法传递基本类型等。
为了解决这些场景,Java为八种基本类型分别定义了对应的类型封装器:Character、Boolean、Byte、Short、Integer、Long、Float和Double。
每种类型封装器都定义从基本类型到封装类型的构造函数,也定义从封装类型到基本类型的方法。

//Character(char ch)
Character ch1 = new Character('a');
char ch2 = ch1.charValue();

//Boolean(boolean bol)和Boolean(String bol)
Boolean bo1 = new Boolean(true);
boolean bo2 = bo1.booleanValue();

对于所有数值类型的类封装器都继承自Number抽象类,Number中定义了以不同数据类型返回基本类型的方法:

byte byteValue();
shot shortValue();
int intValue();
long longValue();
float floatValue();
double doubleValue();

数值类型的封装器也都包含两类构造方法:对应的基本类型和字符串。

Integer(int num)
Integer(String num)

由基本类型到类型封装器的过程称之为装箱,由类型封装器到基本的类型过程称之为拆箱。

//装箱
Interger num = new Integer(123);
//拆箱
int num2 = num.intValue();

自动拆装箱

由于频繁调用拆装箱方法用起来很费事,Java在JDK5提供了自动拆装箱的功能,就不需要我们频繁的创建类型封装器和调用类型封装器方法了。只要我们需要基本类型对应的对象,就会自动的将基本类型装箱到与之等价的类型封装器中,这个过程叫做自动装箱。当我们需要基本类型时,会自动将类型封装器转换成基本类型,这个过程叫做自动拆箱。

//有了自动拆装箱机制,就可以轻易使用了
Integer ob = 100;
int num = ob;

自动拆装箱不仅用于简单的赋值,在任何需要基本类型与封装类型转换的时候,它都会自动进行拆装箱转换,比如方法调用。

public class A{
    static int getNum(Integer num){
        //自动拆箱
        return num;
    }
    public static void main(String[] args){
        //自动装箱
        getNum(100);
    }
}

当在表达式中,数值类型对象会被自动拆箱,然后进行表达式计算。

Integer ob = 1;
//这个过程会先自动拆箱,然后加1,最后在自动装箱为Integer
++ob;
Integer num1 = 100;
Double num2 = 12.5;
//num1自动拆箱后进行了类型转换
num2 = num2 + num1;

自动拆装箱除了提供便利外,还能防止错误的发生。比如下面将Integer类型手动拆箱为byte类型,导致数据出错。

Integer num = 1000;
int n = num.byteValue();
//n=-24

自动拆装箱的警告

虽然Java提供了自动拆装箱机制,但是我们应该在只能需要用类型封装器的时候使用它,一般情况下不应该使用。因为自动拆装箱的过程会增加开销,导致同样的代码比基本类型效率低很多。比如下面:

Integer a,b,c;
a = 1;
b = 2;
c = a + b;
System.out.printn(c)

这时候完全没有必要使用类型封装器,这样的代码效率更低。我们应该在一般情况下限制类型封装器的使用。

泛型

泛型是JDK5引入的,泛型在两方面改变了Java:第一,定义了新的语法元素;第二,改变了许多核心API中的类和方法。使用泛型可以以类型安全的方式创建需要兼容各种类型的类、接口和方法,比如许多算法虽然操作类型不同,但是算法逻辑是相同的,通过使用泛型可以达到定义一次算法,可以独立于特定的数据类型的使用。
泛型又称为为参数化类型,因为类、接口或方法中的类型,是通过方法传递的方式传递过来的,使用泛型的类、接口和方法,称为泛型类、泛型接口和泛型方法。
在泛型之前,是通过使用Object引用来操作各种类型的对象,但是使用Object存在两个问题:第一,需要手动强制类型转换;第二,匹配类型是不安全的,容易产生运行时错误。而这两点正也是泛型的优点。

单类型参数的泛型

class Gen{
    T ob;
    Gen(T ob){
        this.ob = ob;
    }
    T getOb(){
        return ob;
    }
}

T就是类型参数的名称,它本身是一个占位符,当创建对象时,将需要传递参数传递给该类型参数名称。类型参数定义在<>中,整个类体都可以使用参数类型。

Gen ob = new Gen();

将Integer类型作为类型参数传递给Gen类,这样Gen类所操作的类型都是Integer类型了。当创建泛型类时,Java编译器实际上没有传递不同版本的Gen类,它在编译期间就将所有泛型信息移除了,将其替换成特定的类型,从而使代码看似创建了不同版本的Gen类,但实际上他们虽然都是Gen的对象,但是它们实际不是相同的类型。移除泛型信息的过程,称为擦除。

对于泛型类有两点需要注意:1. 泛型只能使用引用类型,所以对于基本类型,需要使用类型封装器。2. 基于不同类型参数的泛型类型是不同的,这也是类型安全的有效保证机制

之所以使用泛型能够保证类型安全,因为这个过程消除了手动输入类型转换以及运行时进行类型检查。比如使用Object,在编译过程中编译器并不知道使用的数据类型,所以在取数据时需要进行类型转换,并且在使用中如果类型不匹配直接在运行时跑错。而使用泛型,可以将运行错误提前到编译时错误,这就是泛型的主要优势。

带有多个类型参数的泛型

泛型可以使用多个类型参数,它们之间只需要使用逗号分隔。

class TwoGen{
    T ob1;
    V ob2;
    TwoGen(T ob1,V ob2){
        this.ob1 = ob1;
        this.ob2 = ob2;
    }
}
#创建对象
TwoGen gen = new TwoGen(123,"Hello");

创建泛型的一般形式:

class class-name {
#创建对象一般形式
class-name var-name = new class-name(cons-arg-list);

在JDK7开始,可以使用简短方式创建泛型类的对象了,不需要在创建对象时指定泛型参数,因为它可以通过赋值变量来判定。
TwoGen gen = new TwoGen<>(123,"hello")

有界类型

使用前面的泛型定义可以传递任意类型的参数,但是有时候需要对传递的类型参数进行限定,比如该泛型类只能操作数值类型,因为它定义了一些涉及数值类型的操作。

class Stats{
    T[] nums;
    Stats(T nums){
        this.nums = nums;
    }
    double average(){
        double sum = 0;
        for(T num: nums){
            sum += num.doubleValue();//Number类定义了该方法
        }
        return sum;
    }
}

T extends Number 这种定义就是有界类型,也就是定义了能够使用的类型必须是Number及其子类。

#定义有界类型

除了使用类作为边界外,还可以使用接口,并且可以指定多个接口,所以边界只能包括一个超类和多接口。比如下面T必须是MyClass类或其子类,并且实现了MyInterface接口。

class

使用通配符

类型安全虽然有用,但是有时候会影响可以接受的数据结构。比如我要比较Integer类型和Double类型的平均值,使用上面的泛型就很难做到,因为一个方法如果使用泛型参数,那么它只能接受一种类型,也就是你所创建的数据类型类。但使用通配符就可以解决这类问题:

boolean sameAvg(Stats ob){
    if(average() == ob.average)
        return true;
    else 
        return false;
}

通配符使用?标识,通配符不会影响能够创建什么类型的Stats对象,这是因为通配符只能简单匹配有效的数据类型,比如上面Status是Number类及其子类,那么通配符只能匹配这些类。

有界通配符

我们也可以为通配符进行界定,因为在很多时候我们不需要通配符来匹配任意类型。有界的通配符可以为类型参数指定上界或者下界,从而限制方法的操作(通配符只能在方法中定义)。

//定义上界,只能包含superclass及其子类

//定义下界,只能包含subclassd的超类

比如只能比较double类型:

boolean sameAvg(Stats ob){
    if(average() == ob.average)
        return true;
    else 
        return false;
}

创建泛型方法

上面的创建完泛型类之后,我们可以在方法中使用传递的参数类型。我们也可以独立于泛型类,创建一个或多个泛型方法,这些泛型方法可以在非泛型类中创建。

 return-type method-name(param-list){}

泛型方法实例,对泛型参数T进行了界定,只能是实现了Comparable接口的类型(可被排序的对象),泛型参数V是T类型或者T类型的子集。

public class GenMethDemo {
    static ,V extends T> boolean isIn(T x,V[] y){
        for(V v : y){
            if(x.equals(v))
                return true;
        }
        return false;
    }
}
//使用泛型方法
GenMethDemo.isIn(2,Arrays.asList(1,2,3));

调用泛型方法可以直接传递参数,不需要指定类型,因为参数类型能够自动辨别。当然,我们也可以显示的指定,但是没有意义。

GenMethDemo.isIn(2,Arrays.asList(1,2,3));

泛型方法可以是静态的,也可以不是,这个没有限定。

我们不仅可以定义普通泛型方法,还可以定义泛型的构造方法。

public classs GenCons{
    private double val;
     GenCons(T arg){
        val = arg.doubleValue();
    }
}

泛型接口

除了可以定义泛型泛型类和泛型方法外,还可以定义泛型接口。泛型接口的定义和泛型类相似。

interface MinMax>{
    T min();
    T max();
}

class MyClass> implements MinMax{
    ...
}

这里需要注意的是,泛型接口的实现类必须需要与接口指定相同的界限,一旦建立这个界限,就不需要再在implements子句中指定了。
如果类实现了泛型接口,那么类也必须是泛型化的,因为它至少需要带有被传递给接口的类型参数。除非类实现了某种具体的泛型接口。

class MyClass implements MinMax{
}

泛型接口有两个优势:一是可以针对不同类型的数据进行实现;二是为接口的数据类型设置限制条件。

泛型类的层次

泛型类也可以像普通类那样有层次关系,因此,泛型类可以作为超类或子类。如果超类是泛型类,那么所有子类都需要向上传递超类所需要的所有泛型参数(子类也可以定义自己的类型参数)。

class Gen{}
class Gen2 extends Gen{}

泛型类也可以继承非泛型类。
泛型类也可以使用instanceof运算符进行运行时比较:

Gen2 a = new Gen2<>();
if(a instanceof Gen) //指定使用通配符,运行时不知道泛型类型信息

当两个泛型实例的类型相互兼容切它们的类型参数相同时,可以使用强制类型转换,
子类可以重写泛型方法。

擦除

Java使用擦除来实现的泛型。擦除原理:编译Java代码时,所有泛型信息都会被移除(擦错)。使用它们的界定类型替换为类型参数,如果没有指定界定类型,就使用Object。然后使用类型转换,以保持和指定的类型参数兼容。
所以运行时并没有类型参数语法,它只是一种源代码的机制。
之所以这样做的原因是,要和之前的Java版本兼容。

使用泛型的一些限制

  1. 不能实例化泛型参数,比如 new T();
  2. 静态成员(包括静态变量和方法)不可以使用泛型参数,但可以为静态方法创建泛型方法。
  3. 创建泛型类数组时,只能使用通配符。Gen[] gens = new Gen<>[10];
  4. 泛型不能扩展Throwable,就是不能创建泛型异常类。

lambda表达式

对Java产生深远影响的两个功能就是JDK5增加的泛型和JDK8新增的lambda表达式。lambda作为新的语法元素,提升了java语言的表达能力,并且流线化了一些常用结构的实现方式。

什么是lambda表达式

lambda表达式本身是一个匿名方法,但是这个方法不能独立执行,它需要实现由函数式接口定义的方法。所以lambda表达式会产生一个匿名类。lambda表达式也常被称为闭包。
函数式接口是仅包含一个抽象方法的接口。这个方法指明了接口的用途,所以函数式接口通常表示单个动作。

lambda表达式使用操作符:->。左侧指定了lambda表达式所需要的所有参数(如果不需要参数,则使用空参数列表)。右侧指定了lambda体,就是要执行的动作,lambda体可以是单独的一个表达式,也可以是一个代码块。

() -> Math.random() * 100;
//n参数通过类型推断得到,也可以显示定义 (int n)
(n) - > n % 2 == 0

上面说了lambda表达式不是独立执行的(直接执行上面语句是错误的),因为lambda表达式是作为函数式接口中的抽象方法实现,这个函数式接口定义了lambda表达式的目标类型。我们只有在这个目标类型的上下文中(可以理解成接口定义的变量),才能使用lambda表达式

//只有一个方法的函数式接口
interface MyNumber{
    double getValue();
}
//自动创建实现了函数式接口的一个匿名类实例,而该接口的方法实现使用lambda表示
MyNumber number = () -> 100.01;
System.out.println(number)

函数式接口可以执行任何与其兼容的lambda表达式,也就说可以有多种实现该接口的方法。

interface NumberTest{
    boolean test(int x,int y);
}
NumberTest test = (n,d) -> n % d == 0;
System.out.println(test(2,3));

对于多个参数的lambda,如果想要显示声明一个参数的类型,那么必须为所有的参数类型显示声明。

NumberTest test = (int n,int d) -> n % d == 0;

lambda块体

lambda体可以包含单个表达式,称为表达式lambda;也可以是一个代码块,称为块lambda表达式。块lambda必须显示使用return语句返回值,因为块lambda不是一个单独表达式。

MyNumber sum = () -> {
    int s = 0;
    for(int i=0; i< 100;i++)
        s += i;
    return s;
}

泛型函数式接口

lambda表达式自身不能指定类型参数,但是可以定义泛型函数式接口,lambda表达式由泛型接口的引用类型来决定。

interface SomeFunc{
    T fun(T t);
}
SomeFunc reverse = (str) - >{
    ...
}

SomeFunc sum = (n) -> {
    ...
}

作为参数传递lambda表达式

lambda表达式的另一个常用方法就是作为参数传递给方法,这样就可以将可执行代码作为参数传递给方法,极大的增强了Java的表达力。
需要注意,将lambda表达式作为参数传递,那么接受lambda表达式的形参需要和该lambda表达式兼容的函数式接口。

interface StringFunc{
    String func(String sr);
}

class Test{
    //接受的是一个lambda表达式
    stattic String strOp(StringFun fun,String str){
        return fun.func(str);
    }
    public static void main(String[] args){
        StringFunc fun1 = (str) -> {
            ...
        }
        
        String opResult = strOp(fun1,"hello");
        String opResult2 = strOp((str) -> n.toUpperCase(),"hello")
    }
}

lambda表达式抛出异常

lambda表达式也可以抛出异常,但是需要注意的是,如果抛出的异常是经检查的异常,那么必须在函数式接口中通过throws定义了。

interface StringFunc{
    //抛出自定义的空异常
    String func(String sr) throws EmptyException;
}

String str = (str) ->{
...
throw new EmptyException();
}

lambda表达式变量捕获

lambda表达式能够访问外层作用域定义的变量,比如外层定义的实例或者静态变量。
但是lambda表达式在使用外层作用域中的局部变量时,不能修改局部变量,因为它被认为是final类型的变量了。不仅不能在lambda表达式中修改,也不能在lambda表达式外修改。

int sum = 0;
public void test() {
        TestFunc testFunc = () -> {
            //因为sum是全局变量,所以修改没有问题
            sum++;
        };
 }

public void test() {    
        int sum = 0;
        TestFunc testFunc = () -> {
            //不能修改局部变量
            //sum++;
        };
        sum++;//如果lambda表达式使用了该变量,这里也不能修改
 }

这种称为变量捕获,捕获的变量被默认设置为final类型了。

方法引用

lambda还有另一个重要的特性就是方法引用,方法引用提供了一种引用方法但是不执行方法的方式。这个特性需要和lambda一起使用,因为要引用的方法必须能够和函数式接口中的方法兼容。也就说任一普通方法,只要和函数式接口的方法兼容,就都可以作为引用方法传递给函数式类型的参数的方法。相当于我们将这个方法作为() -> {} lambda表达式使用了。
方法引用可以是静态方法引用或实例方法。

interface StringFun{
    String func(String n);
}
class StringOps{
    static String strReverse(String str) {
        ...
    }
}

class Demo{

    static String stringOp(StringFun fun,String str){
        fun.func(str);
    }

    public static void main(String[] args) {
        //将StringOps.strReverse()当做lambda使用
        String result = stringOp(StringOps::strReverse(),"Hello");
    }
}

也可以是实例方法

    objRef :: methodName

还有一种,就是直接使用类名和实例方法,这样我们要引用的方法需要比函数式接口中的方法参数少一个,因为默认会将这个引用对象作为参数传递过去。这里要和静态方法进行区别。

ClassName :: instanceMethodName

除了可以使用普通方法引用,还可以使用构造函数引用。

interface MyFunc{
    MyClass func(int n);
}
class MyClass{
    MyClass(){
        ...
    }
    MyClass(int n){
    }
}
class Main{
    public static void main(String[] args) {
        //构造方法应用 className :: new
        MyFunc myFunc = MyClass :: new
        myFunc.func(100);
    }
}

预定义的函数式接口

JDK8中包含了一个新包java.util.function,里面提供了一些预定义的函数式接口。
//TODO

注解/元数据

在JDK5上,Java支持了在源文件中嵌入补充信息,这些补充信息对程序运行没有任何改变。一般用于开发和部署期间,各种工具使用这类信息。比如源代码生成器可以处理注解信息。
注解通过基于接口机制创建:

@interface MyAnno{
    String str();
    int val();
}

@告诉编译器这里声明的是注解类型,注解实体中包含了两个方法,和接口一样,这里只是声明方法,不做具体实现。具体实现由使用注解的地方实现。
注解不能包含extends子句,但是所有字节都自动扩展了java.lang.annotation.Annotation接口。

在声明完注解后,我们就可以来注解声明了(使用注解声明)。在JDK8之前注解只能用于声明,在JDK8中添加了使用注解类型的功能。

使用注解

在所有类型的声明中都可以与注解进行关联,比如类、方法、域变量、参数以及枚举常量,都可以带有注解。

@MyAnno(str = "hello", val = 100)
class MyClass{

}

应用注解时,需要为注解成员提供值,从这里看这些更像是变量不像抽象方法。

注解保留策略

注解保留策略决定了在什么时候丢弃注解。java定义了三种策略,他们被封装到java.lang.annotation.RetentionPolicy中。

  • SOURCE: 只在源文件中保留,编译期间会被丢弃。
  • CLASS: 在编译时被存储到.class文件中,但是在运行时JVM不能得到这些注解。
  • RUNTIME: 编译时存储到.class文件中,并且运行时可以通过JVM获取到这些注解。RUNTIME提供了最永久的注解。

注意:局部变量的注解不能存储到.class文件中

保留策略通过内置注解@Retention指定:

@Retention(retention-policy)

retention-policy必须是RetentionPolicy中的常量,如果没有指定,默认就是CLASS。

@Retention(RetentionPolicy.RUNTIME)
@interface MyAnno{
    ...
}

运行时通过反射获取注解

我们除了在其它开发和部署工具上使用注解,对于RUNTIME保留策略的注解,可以在运行时通过反射来查询注解。反射能够获取运行时类相关信息,反射API位于java.lang.reflect包中。
使用反射获取注解分为三步:

  1. 获取Class对象,Class是在java.lang包中定义的,可以通过多种方式获取class。比如getClass()方法。
  2. 获取Class对象后,可以使用它的方法获取与类声明的相关的信息,包括注解。Class提供了getMethod()、getField()、以及getConstructor(),他们分别返回Method、Field和Constructor。
  3. 使用Method、Field或Constructor对象调用getAnnotation()方法,就可以获取与对象关联的注解了。
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnno{
    String str();
    int val();
}
class Test{
    @MyAnno(str = "hello", val = 100)
    public static myMethod(int num){
        Test test = new Test();
        Class clazz = test.getClass();
        //如果方法有参数,需要指定参数类型
        Method m = clazz.getMethod("myMethod",int.class);
        MyAnno myannno = m.getAnnotation(MyAnno.class);
        System.out.println(myanno.str())
    }
}

为注解使用默认值

在定义注解的时候,我们可以为注解指定默认值,当使用注解时候,如果没有为注解指定值,就会使用默认值。

@interface MyAnno{
    String str() default "hello";
    int val() default 100;
}
//使用
@MyAnno()
@MyAnno(str = "test")
@MyAnno(val = 1)
@MyAnno(str = "test",val = 1)

还有一类特殊使用注解,就是单成员注解。如果注解中只有一个成员,我们可以将成员名称定义为value,这时候在使用时候就可以不指定成员名称了。

@interface MyAnno{
    String value();
}
@MyAnno("hello")
class Test{

如果有多个成员,并且其它所有成员都有默认值的话,也可以这样使用。不过如果显示使用默认值的时候,这时候在调用的时候也要显示value=xxxx。

标记注解

标记注解时特殊类型的注解,他没有成员(定义完注解明后后就完事了)。标记注解的唯一目的就是标记。确定标记注解的最好方式就是使用反射java.lang.reflect.AnnotatedElement类中的isAnnotationPreset()方法判断。

@interface MyAnno{
}

内置注解

Java提供了许多内置注解,大部分都是专用注解,但是有9个用于一般目的。java.lang.annotation中的@Retention、@Documented、@Target、@Inherited,java.lang中的@Override、@Deprecated、@FunctionalInterface、@SafeVarargs和@SuppressWarnning。

  • @Retention:只能用于注解其它注解,用于指定保留策略。
  • @Documented:标记注解,只能注解其它注解,用于提示其它工具,被注解的注解将被文档化。
  • @Target:只能注解其它注解,用来表示注解的声明类型(它告诉你它标记的这个注解是用来干什么的,使用ElementType),如果声明该注解用于类的。
  • @Inherited:标记注解,只能注解其它注解。并且@Inherited只能影响类声明的注解,会将超类的注解被子类继承。
  • @Override:标记注解,只能用于方法。使用@Override注解的方法,必须重写超类中的方法。它用于确保超类方法被重写,而不是简单的重载。
  • @Deprecatd:标记注解,用于指示是过时的了。
  • @FunctionalInterface: JDK8新增注解的标记注解,用于接口。指出被注解的接口是函数式接口。
  • @SafeVarags:标记注解,只能用于方法和构造函数,指示没有发生于可变长度参数相关的不安全动作。
  • @SuppressWarnings:抑制一个或多个编译器可能会报告的警告,就是我们IDEA中黄色表示的部分。

类型注解

JDK8新增了使用注解的地方,就是类型注解,也就是说注解可以用于标识类型。比如返回类型、this、强制转换、被继承的类以及throws子句或者注解泛型。类型注解需要使用@Target(ElementType.TYPE_USE)标识。

class Test{
    //注解this
    int method(@TypeAnno Test this,int i,int j){
}

你可能感兴趣的:(Java基础语法回顾(二))