Java---实现运行任意目录下class中加了@MyTest的空参方法(实现图形界面)

说明:

因为上个代码,总是要输入完整的绝对路径,比较麻烦,于是,就写了这个小程序,直接进入文件对话框选择需要运行的class文件。

只需要提前输入完整的类名。

注意:加的MyTest必须打个包,加上:
import cn.hncu.myJuniitApp.vo.MyTest;
不然不是同一个注解呢。

测试的类:

package cn.hncu.myJuniitApp;

import cn.hncu.myJuniitApp.vo.MyTest;

public class MyJunitTest {

    public void run1(){
        System.out.println("run1().....");
    }

    @MyTest
    public void run2(){
        System.out.println("run2().....含有MyTest");
    }

    public void run3(){
        System.out.println("run3().....");
    }

    @MyTest
    public void run4(){
        System.out.println("run4().....含有MyTest");
    }

    public void run5(){
        System.out.println("run5().....");
    }

}   

然后是注解类:

package cn.hncu.myJuniitApp.vo;

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

//运行时也存在的,如果想看到运行结果,必须定义这个注解的保持性为运行时
@Retention (RetentionPolicy.RUNTIME)
@Target (ElementType.METHOD)//限制这个注解只能用在方法上面
public @interface MyTest {
}

数据层:

接口:

package cn.hncu.myJuniitApp.dao.dao;

/**
 * 数据层接口
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public interface JunitDao {
    public Class<?> findClass(String name,String className);
}

实现类:

package cn.hncu.myJuniitApp.dao.impl;

import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

import cn.hncu.myJuniitApp.dao.dao.JunitDao;
/**
 * 数据层的实现类
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public class JunitDaoImpl extends ClassLoader implements JunitDao{

    @Override
    public Class<?> findClass(String name, String className) {
        byte buf[] = loadClassData(name);
        Class c = defineClass(className, buf, 0, buf.length);
        return c;
    }

    private byte[] loadClassData(String name) {
        byte buf[]=null;
        try {
            FileInputStream in  = new FileInputStream(name);

            ByteArrayOutputStream bout = new ByteArrayOutputStream();
            int len=0;
            byte[] b = new byte[1024];
            while((len=in.read(b))!=-1){
                bout.write(b, 0, len);
            }
            in.close();
            bout.close();
            buf=bout.toByteArray();

        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return buf;
    }

}

工厂方法:

package cn.hncu.myJuniitApp.dao.factory;

import cn.hncu.myJuniitApp.dao.dao.JunitDao;
import cn.hncu.myJuniitApp.dao.impl.JunitDaoImpl;

/**
 * 数据层的工厂方法
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public class JunitDaoFactory {
    public static JunitDao getJunitDao(){
        return new JunitDaoImpl();
    }
}

逻辑层:

接口:

package cn.hncu.myJuniitApp.business.ebi;

/**
 * 逻辑层接口
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public interface JunitEbi {
    public void run(String name,String className);
}

实现类:

package cn.hncu.myJuniitApp.business.ebo;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import cn.hncu.myJuniitApp.business.ebi.JunitEbi;
import cn.hncu.myJuniitApp.dao.factory.JunitDaoFactory;
import cn.hncu.myJuniitApp.vo.MyTest;
/**
 * 逻辑层的实现类
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public class JunitEbo implements JunitEbi{

    //通过数据层获得对象的Class
    @Override
    public void run(String name, String className) {
        Class c = JunitDaoFactory.getJunitDao().findClass(name, className);
        try {
            Object obj = c.newInstance();
            Method ms[] = c.getDeclaredMethods();//获得当前类的所有声明方法,包括私有的
            for(Method m : ms){//增强for循环遍历
                if(m.isAnnotationPresent(MyTest.class)){//如果这个方法有这个注解,就运行
                    m.invoke(obj, null);//现在知道为什么要无参了吧,这样方便很多,直接就可以调用了。
                }
            }
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }
}

工厂方法:

package cn.hncu.myJuniitApp.business.factory;

import cn.hncu.myJuniitApp.business.ebi.JunitEbi;
import cn.hncu.myJuniitApp.business.ebo.JunitEbo;

/**
 * 逻辑层的工厂方法
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public class JunitEbiFactory {
    public static JunitEbi getJunitEbi(){
        return new JunitEbo();
    }
}

界面和main方法:

package cn.hncu.myJuniitApp;

import java.awt.Dimension;
import java.awt.Toolkit;
import java.io.File;

import javax.swing.JFileChooser;
import javax.swing.JOptionPane;

import cn.hncu.myJuniitApp.business.factory.JunitEbiFactory;

/**
 * 
 * @author 陈浩翔
 * @version 1.0  2016-5-6
 */
public class myJunit extends javax.swing.JFrame {

    public myJunit() {
        initComponents();
    }

    private void initComponents() {
        jLabel1 = new javax.swing.JLabel();
        tfdClassName = new javax.swing.JTextField();
        jLabel2 = new javax.swing.JLabel();
        btnFileName = new javax.swing.JButton();
        btnRun = new javax.swing.JButton();

        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);

        //获得系统屏幕分辨率
        Toolkit t = Toolkit.getDefaultToolkit() ;
        Dimension size=t.getScreenSize();  

        setBounds((int)size.getWidth()/4,(int)size.getHeight()/4,400,400);
        getContentPane().setLayout(null);

        jLabel1.setFont(new java.awt.Font("Dialog", 1, 30));
        jLabel1.setForeground(new java.awt.Color(204, 0, 0));
        jLabel1.setText("MyJunit-\u8fd0\u884c\u4efb\u610f\u7a7a\u53c2\u65b9\u6cd5");
        getContentPane().add(jLabel1);
        jLabel1.setBounds(10, 20, 380, 80);

        tfdClassName.setFont(new java.awt.Font("Dialog", 1, 18));
        tfdClassName.setForeground(new java.awt.Color(255, 0, 0));
        getContentPane().add(tfdClassName);
        tfdClassName.setBounds(30, 180, 350, 40);

        jLabel2.setFont(new java.awt.Font("Dialog", 1, 18));
        jLabel2.setForeground(new java.awt.Color(0, 0, 255));
        jLabel2.setText("请先输入完整类名(再选择文件)");
        getContentPane().add(jLabel2);
        jLabel2.setBounds(30, 120, 310, 80);

        btnFileName.setFont(new java.awt.Font("Dialog", 1, 18));
        btnFileName.setText("\u9009\u62e9\u6587\u4ef6");
        btnFileName.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                btnFileNameActionPerformed(evt);
            }
        });
        getContentPane().add(btnFileName);
        btnFileName.setBounds(40, 270, 110, 50);

        btnRun.setFont(new java.awt.Font("Dialog", 1, 18));
        btnRun.setText("\u8fd0\u884c");
        btnRun.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                btnRunActionPerformed(evt);
            }
        });
        getContentPane().add(btnRun);
        btnRun.setBounds(240, 270, 110, 50);
    }

    private void btnRunActionPerformed(java.awt.event.ActionEvent evt) {
        if(fileName==null){
            JOptionPane.showMessageDialog(this, "请先选择文件!");
        }
        JunitEbiFactory.getJunitEbi().run(fileName, className);
    }

    private void btnFileNameActionPerformed(java.awt.event.ActionEvent evt) {
        JFileChooser jfc = new JFileChooser();
        //打开文件选择对话框
        int result = jfc.showOpenDialog(this);
        File file=null;
        if(result==JFileChooser.APPROVE_OPTION){//选择了文件
            file=jfc.getSelectedFile();//获得选择的文件
            //System.out.println(file.getParent());
            //System.out.println(file.getName());
            fileName = file.getParent()+"\\"+file.getName();
            className = tfdClassName.getText().trim();
            if("".equals(className)){
                JOptionPane.showMessageDialog(this, "请输入正确的完整类名!!!");
                return ;
            }
            //System.out.println(fileName);
            //System.out.println(className);
        }
    }

    public static void main(String args[]) {
        java.awt.EventQueue.invokeLater(new Runnable() {
            public void run() {
                new myJunit().setVisible(true);
            }
        });
    }

    private javax.swing.JButton btnFileName;//选择文件的按钮
    private javax.swing.JButton btnRun;//运行含有@MyTest的空参方法
    private javax.swing.JLabel jLabel1;
    private javax.swing.JLabel jLabel2;
    private javax.swing.JTextField tfdClassName;
    private String fileName =null;
    private String className =null;


}

下面看看一些程序的图片:

先看下运行结果吧:我把那个测试类移到了d盘去了的。

run2().....含有MyTest
run4().....含有MyTest

Java---实现运行任意目录下class中加了@MyTest的空参方法(实现图形界面)

也许有人认为一个类完全就可以解决这个问题了,你还写这么多的包,这么多类,这不是麻烦嘛。
我想说,虽然麻烦,但是可以让我们看起来结构清楚,谁调用谁也很明白,而且是面向接口编程。
这个写得更加规范,特别是以后到公司工作,基本上都是几个人开发同一个项目的,这个人写这里的,那个人写那里的,如果没有规范,怎么行呢。怎么合作开发同一个项目呢。

所以,分包是必须的。分逻辑层和数据层也是需要的。特别是一个层的铁三角,必须都要有。
接口,工厂方法,实现类,缺一不可!!!

上一篇:JAVA入门--目录


下一篇:第二章 IoC 构造器注入