JAVAプレミアムプロパティ-注記

4246 ワード

注記は、コードに挿入されたツール処理のラベルです.これらのラベルは、ソース階層で操作するか、コンパイラを処理して注釈クラスファイルに組み込むことができます.
注記プログラムのコンパイル方法は変更されません.Javaコンパイラは、注釈を含むコードと注釈を含まないコードに対して同じ仮想マシン命令を生成します.
Javaでは、注釈は通常@Annotationで表され、注釈は通常フレームワークの設計に用いられ、反射に合わせてコード量を減らし、ビジネスロジックを明確にする.
我々の最も一般的な注釈は@Overrideと@Testという注釈であり、前者はあるインタフェースを実現するための方法を明示するために用いられ、後者はJUNITによく用いられる.
注記のインスタンス定義を次に示します.
@Target(ElementType.Method)

@Retention(RetentionPolicy.RUNTIME)

public @interface Test

{

    long timeout() default 0L;

}


@Interface宣言は、本格的なjavaインタフェースを作成しました.注記を処理するツールは、インタフェースを実装したオブジェクトを受け入れます.
注記TargetとRetentionはメタ注記であり、Test注記がメソッドでのみ使用できる注釈であることを示し、クラスファイルが仮想マシンにロードされたときも保持できます.
注記の使用は通常、反射に合わせて使用されます.次に例を示します.
package com.inspur.jiyq.corejava.annotation;



import java.awt.Color;



import javax.swing.JButton;

import javax.swing.JFrame;

import javax.swing.JPanel;



public class ButtonFrame extends JFrame{

	private static final long serialVersionUID = 1L;

	

	public static final int DEFAULT_WIDTH = 300;

	public static final int DEFAULT_HEIGHT = 200;

	

	private JPanel panel;

	private JButton yellowButton;

	private JButton blueButton;

	private JButton redButton;

	

	public ButtonFrame()

	{

		setTitle("ButtonTest");

		setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT);

		

		panel = new JPanel();

		add(panel);

		

		yellowButton = new JButton("Yellow");

		blueButton = new JButton("Blue");

		redButton = new JButton("Red");

		

		panel.add(yellowButton);

		panel.add(blueButton);

		panel.add(redButton);

		

		ActionListenerInstaller.processAnnotations(this);

	}

	

	@ActionListenerFor(source = "yellowButton")

	public void yellowBackground()

	{

		panel.setBackground(Color.YELLOW);

	}

	

	@ActionListenerFor(source = "blueButton")

	public void blueBackground()

	{

		panel.setBackground(Color.BLUE);

	}

	

	@ActionListenerFor(source = "redButton")

	public void redBackground()

	{

		panel.setBackground(Color.RED);

	}



	public static void main(String[] args)

	{

		ButtonFrame frame = new ButtonFrame();

		frame.setVisible(true);

	}

}


  
package com.inspur.jiyq.corejava.annotation;



import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;



@Target(ElementType.METHOD)

@Retention(RetentionPolicy.RUNTIME)

public @interface ActionListenerFor {

	String source();

}


  
package com.inspur.jiyq.corejava.annotation;



import java.awt.event.ActionListener;

import java.lang.reflect.Field;

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.InvocationTargetException;

import java.lang.reflect.Method;

import java.lang.reflect.Proxy;



/**

 *  

 */

public class ActionListenerInstaller {

	/**

	 * Process all ActionListenerFor annotations in the given object

	 * 

	 * @param obj

	 *            an object whose methods may have ActionListnerFor annotations

	 */

	public static void processAnnotations(Object obj) {

		try {

			Class<?> cl = obj.getClass();

			for (Method m : cl.getDeclaredMethods()) {

				ActionListenerFor a = m.getAnnotation(ActionListenerFor.class);

				if (a != null) {

					Field f = cl.getDeclaredField(a.source());

					f.setAccessible(true);

					addListener(f.get(obj), obj, m);

				}

			}

		} catch (Exception e) {

			e.printStackTrace();

		}

	}



	public static void addListener(Object source, final Object param,

			final Method m) throws NoSuchMethodException, SecurityException,

			IllegalAccessException, IllegalArgumentException,

			InvocationTargetException {

		InvocationHandler handler = new InvocationHandler() {

			@Override

			public Object invoke(Object proxy, Method method, Object[] args)

					throws Throwable {

				return m.invoke(param);

			}

		};



		Object listener = Proxy.newProxyInstance(null,

				new Class[] { java.awt.event.ActionListener.class }, handler);

		Method adder = source.getClass().getMethod("addActionListener",

				ActionListener.class);



		adder.invoke(source, listener);

	}

}