Java自定義注解


Java自定義注解

 

前言:這兩天看了一下Java自定義注解的內容,然后按照我自己的理解寫了兩份代碼,還挺有趣的,本文包括三個部分:注解的基礎、通過注解進行賦值(結合了工廠方法模式)、通過注解進行校驗。

 

一、注解的基礎

1.注解的定義:Java文件叫做Annotation,用@interface表示。

2.元注解:@interface上面按需要注解上一些東西,包括@Retention、@Target、@Document、@Inherited四種。

3.注解的保留策略:

  @Retention(RetentionPolicy.SOURCE)   // 注解僅存在於源碼中,在class字節碼文件中不包含

  @Retention(RetentionPolicy.CLASS)     // 默認的保留策略,注解會在class字節碼文件中存在,但運行時無法獲得

  @Retention(RetentionPolicy.RUNTIME)  // 注解會在class字節碼文件中存在,在運行時可以通過反射獲取到

4.注解的作用目標:

  @Target(ElementType.TYPE)                      // 接口、類、枚舉、注解

  @Target(ElementType.FIELD)                     // 字段、枚舉的常量

  @Target(ElementType.METHOD)                 // 方法

  @Target(ElementType.PARAMETER)            // 方法參數

  @Target(ElementType.CONSTRUCTOR)       // 構造函數

  @Target(ElementType.LOCAL_VARIABLE)   // 局部變量

  @Target(ElementType.ANNOTATION_TYPE) // 注解

  @Target(ElementType.PACKAGE)               //

5.注解包含在javadoc中:

  @Documented

6.注解可以被繼承:

  @Inherited

7.注解解析器:用來解析自定義注解。

 

二、通過注解進行賦值(結合了工廠方法模式)

  1.自定義注解

package annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Documented
@Inherited
@Target({ ElementType.FIELD, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface Init
{
    public String value() default "";
}

  2.在數據模型使用注解

package model;

import annotation.Init;

public class User
{
    private String name;
    private String age;

    public String getName()
    {
        return name;
    }

    @Init(value = "liang")
    public void setName(String name)
    {
        this.name = name;
    }

    public String getAge()
    {
        return age;
    }

    @Init(value = "23")
    public void setAge(String age)
    {
        this.age = age;
    }
}

  3.用“構造工廠”充當“注解解析器”

package factory;

import java.lang.reflect.Method;

import annotation.Init;
import model.User;

public class UserFactory
{
    public static User create()
    {
        User user = new User();

        // 獲取User類中所有的方法(getDeclaredMethods也行)
        Method[] methods = User.class.getMethods();

        try
        {
            for (Method method : methods)
            {
                // 如果此方法有注解,就把注解里面的數據賦值到user對象
                if (method.isAnnotationPresent(Init.class))
                {
                    Init init = method.getAnnotation(Init.class);
                    method.invoke(user, init.value());
                }
            }
        }
        catch (Exception e)
        {
            e.printStackTrace();
            return null;
        }

        return user;
    }
}

  4.運行的代碼

package app;

import java.lang.reflect.InvocationTargetException;

import factory.UserFactory;
import model.User;

public class Test
{
    public static void main(String[] args) throws IllegalAccessException,
            IllegalArgumentException, InvocationTargetException
    {
        User user = UserFactory.create();

        System.out.println(user.getName());
        System.out.println(user.getAge());
    }
}

  5.運行結果

liang
23

 

三、通過注解進行校驗

  1.自定義注解

package annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Documented
@Inherited
@Target({ ElementType.FIELD, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface Validate
{
    public int min() default 1;

    public int max() default 10;

    public boolean isNotNull() default true;
}

  2.在數據模型使用注解

package model;

import annotation.Validate;

public class User
{
    @Validate(min = 2, max = 5)
    private String name;

    @Validate(isNotNull = false)
    private String age;

    public String getName()
    {
        return name;
    }

    public void setName(String name)
    {
        this.name = name;
    }

    public String getAge()
    {
        return age;
    }

    public void setAge(String age)
    {
        this.age = age;
    }
}

  3.注解解析器

package check;

import java.lang.reflect.Field;

import annotation.Validate;
import model.User;

public class UserCheck
{
    public static boolean check(User user)
    {
        if (user == null)
        {
            System.out.println("!!校驗對象為空!!");
            return false;
        }

        // 獲取User類的所有屬性(如果使用getFields,就無法獲取到private的屬性)
        Field[] fields = User.class.getDeclaredFields();

        for (Field field : fields)
        {
            // 如果屬性有注解,就進行校驗
            if (field.isAnnotationPresent(Validate.class))
            {
                Validate validate = field.getAnnotation(Validate.class);
                if (field.getName().equals("age"))
                {
                    if (user.getAge() == null)
                    {
                        if (validate.isNotNull())
                        {
                            System.out.println("!!年齡可空校驗不通過:不可為空!!");
                            return false;
                        }
                        else
                        {
                            System.out.println("年齡可空校驗通過:可以為空");
                            continue;
                        }
                    }
                    else
                    {
                        System.out.println("年齡可空校驗通過");
                    }

                    if (user.getAge().length() < validate.min())
                    {
                        System.out.println("!!年齡最小長度校驗不通過!!");
                        return false;
                    }
                    else
                    {
                        System.out.println("年齡最小長度校驗通過");
                    }

                    if (user.getAge().length() > validate.max())
                    {
                        System.out.println("!!年齡最大長度校驗不通過!!");
                        return false;
                    }
                    else
                    {
                        System.out.println("年齡最大長度校驗通過");
                    }
                }
                if (field.getName().equals("name"))
                {
                    if (user.getName() == null)
                    {
                        if (validate.isNotNull())
                        {
                            System.out.println("!!名字可空校驗不通過:不可為空!!");
                            return false;
                        }
                        else
                        {
                            System.out.println("名字可空校驗通過:可以為空");
                            continue;
                        }
                    }
                    else
                    {
                        System.out.println("名字可空校驗通過");
                    }

                    if (user.getName().length() < validate.min())
                    {
                        System.out.println("!!名字最小長度校驗不通過!!");
                        return false;
                    }
                    else
                    {
                        System.out.println("名字最小長度校驗通過");
                    }

                    if (user.getName().length() > validate.max())
                    {
                        System.out.println("!!名字最大長度校驗不通過!!");
                        return false;
                    }
                    else
                    {
                        System.out.println("名字最大長度校驗通過");
                    }
                }
            }
        }

        return true;
    }
}

  4.運行的代碼

package app;

import check.UserCheck;
import model.User;

public class Test
{
    public static void main(String[] args)
    {
        User user = new User();
        
        user.setName("liang");
        user.setAge("1");
        
        System.out.println(UserCheck.check(user));
    }
}

  5.運行結果

名字可空校驗通過
名字最小長度校驗通過
名字最大長度校驗通過
年齡可空校驗通過
年齡最小長度校驗通過
年齡最大長度校驗通過
true

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM