大家一起写mvc(二)

上一篇已经看了,我想大家都明白了mvc的原理,今天我们来说一下要写自己mvc框架必须要会的技术。

mvc的目录是这样的

src目录是我们核心的mvc代码。这个代码明天讲,今天主要讲的代码都在test目录下。

在第一章我已经说了,写mvc技术需要用的就是java的反射原理,还有自定义注解。

现在我们先讲一下注解的用处,可能这个自定义注解听着挺高深的,等你看完这个就会明白,其实很简单,就是xml的另一种方法。

讲解我就都放在代码的注解里了,这样写比较方便。

annotation注解:

package com.test;

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

/**
 * jdk1.5以上
 * @Target 描述注解的使用范围
 * @Retention 注解生命周期
 * @Documented 文档相关
 * @Inherited 阐述了某个被标注的类型是被继承的
 * 关于注解的详细信息可以看下面这个博客。
 * http://www.cnblogs.com/peida/archive/2013/04/24/3036689.html
 * 这篇文章写的很好,我就不重复的叨逼叨了。。
 * 在mvc里 我们就用两个注解属性
 * @Target 标注使用范围,这个我们标注为method。即这个注解我们要用在方法级别上。
 * @Retention 生命周期,即运行时,Runtime
 * @author wanglong
 *
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TestAnno {
    public String ActionName() default "";
    public String Result() default "";
}

现在我们定义注解了,我们要用在哪呢,就用在Action的方法上,比如我们写一个Action

package com.test;

public class TestAction {
    
    @Anno(ActionName="test.action",Result="index.jsp")
    public void TestAnno(User user)
    {
        System.out.println(user.getUsername());
    }
}

这样我们就在TestAction里的TestAnno方法上加上了我们自定义的注解。

我们怎么获得注解里的值呢,就是通过反射。下面的代码是Junit4的一个testcase。

    public void testAnno() throws Exception{
        Class clazz = Class.forName("com.test.TestAction");
        Method[] methods = clazz.getDeclaredMethods();
        for (Method method : methods)
        {
            if(method.isAnnotationPresent(com.test.Anno.class))
            {
                Anno anno = method.getAnnotation(com.test.Anno.class);
                System.out.println(anno.ActionName()); //test.action
                System.out.println(anno.Result());     //index.jsp
            }
        }
    }

恩,这里我们已经获取到了,大家也知道该怎么用了吧,也知道在mvc里应该用在哪里了。注解我们就讲到这里。下面我们说反射

java反射

我也不知道大家会不会反射。。我这也不知道说到什么程度合适,所以我就把要在mvc中用到的反射方法说一下吧。

首页,我们有一个登录页面-login.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<form action="test.action" method="post">
    <input type="text" value="admin" name="user.userName">
    <input type="text" value="adminPassword" name="user.password">
    <input type="submit" value="登录"/>
</form>
</body>
</html>

然后我们还要有一个user类

package com.test;

public class User {
    private String username;
    private String password;
    
    public String getUsername()
    {
        return username;
    }
    public void setUsername(String username)
    {
        this.username = username;
    }
    public String getPassword()
    {
        return password;
    }
    public void setPassword(String password)
    {
        this.password = password;
    }
}

我们看到login.jsp请求action是test.action.里面的参数分别是user.username跟user.password,我们通过servlet拦截后,可以通过java的自定义注解也就是我们上面的Anno这个注解来获取到test.action注解的这个(TestAnno)方法,但是我们怎么把页面上的这两个参数传给action呢,所以我们就用到反射,通过反射,我们把页面上user封装,然后在通过反射来调用TestAnno方法。

首先,我们要通过反射获取到TestAnno这个方法的参数类型。

/*
     * 反射获取方法参数类型
     */
    @org.junit.Test
    public void testActionParamType() throws Exception{
        Class clazz = Class.forName("com.test.TestAction");
        Method[] methods = clazz.getDeclaredMethods();
        for (Method method : methods)
        {
            Class[] params = method.getParameterTypes();
            for (int i = 0; i < params.length; i++)
            {
                System.out.println(method.getName());//获取的是方法的名称 也就是 TestAnno
                System.out.println(params[i]);//方法参数类型 这里打印       class com.test.User 说明这个方法只有一个参数
            }
        }
    }

利用jdk的反射,我们获取不到方法的参数名称,也就是public void Test(String s){...}

现在我们能获取到String类型,但是如果想获取这个s,利用jdk的反射是获取不到的,所以我们借助第三方jar包 javassist,这个jar包是对java反射的增强。下载地址百度找吧,如果找不到,可以去这个地址找 https://jarfiles.pandaidea.com/ 这是个很不错网站 大家可以收藏~就不用到处去求包了。下面是利用javassist来获取方法参数名称。

    /**
     * 利用javassist获取方法参数名称
     * @throws Exception
     */
    @org.junit.Test
    public void testGetParam() throws Exception{
        Class clazz = Class.forName("com.test.TestAction");
        String methodName = "TestAnno";
        ClassPool pool=ClassPool.getDefault();
        CtClass ctClass = pool.get("com.test.TestAction");
        CtMethod ctMethod = ctClass.getDeclaredMethod(methodName);
        MethodInfo methodInfo = ctMethod.getMethodInfo();
        CodeAttribute codeAttr = methodInfo.getCodeAttribute();
        LocalVariableAttribute attr = (LocalVariableAttribute) codeAttr.getAttribute(LocalVariableAttribute.tag);
        if(attr==null){
            System.out.println("没有属性");
        }
        String[] paramsName = new String[ctMethod.getParameterTypes().length];
        int pos =Modifier.isStatic(ctMethod.getModifiers())?0:1;
        for (int i = 0; i < paramsName.length; i++)
        {
            paramsName[i] = attr.variableName(i+pos);
        }
        for (int i = 0; i < paramsName.length; i++)
        {
            System.out.println(paramsName[i]);
        }
    }

这样我们就能获取到方法所需要的一切了。然后通过反射的invoke来调用方法就可以了。下面演示一下怎么使用invoke来调用方法及传参。

首先我们新建一个测试类。我们来调用这个类的say方法

package com.test;

public class TestInvoke {
    public void say(String i ){
        System.out.println(i);
    }
}

然后我们在junit测试里添加测试方法。

/**
     * 测试方法调用
     * @throws Exception
     */
    @org.junit.Test
    public void testInvoke() throws Exception{
        Class clazz = Class.forName("com.test.TestInvoke");
        Object obj = clazz.newInstance();
        /**
         * getMethod(方法名,方法参数类型).invoke(调用类的实例,方法参数);
         */
        clazz.getMethod("say", String.class).invoke(obj, "this is a test");
    }

在控制台会打出this is a test。

在补充一个反射的用法。获取类的属性,属性的类型。

    /**
     * 获取类的field 和类型
     * @throws Exception
     */
    @org.junit.Test
    public void testFieldType() throws Exception
    {
        Class clazz = Class.forName("com.test.User");
        Object o = clazz.newInstance();
        Field[] fields = clazz.getDeclaredFields();
        for (Field f : fields)
        {
            System.out.println(f.getName());
            System.out.println(f.getType());
            /**
               控制台打印:
                username
                class java.lang.String
                password
                class java.lang.String

             */
        }
    }

 


总结一下,通过今天的学习我们都得到了什么。

1.通过annotation,我们可以获得注解的名称,以及其他属性,以及注解的方法

2.通过反射:我们可以获得一个类都有哪些方法,方法的参数类型,类的属性,类属性的类型。不知道这么说大家能不能明白。下面截个图给大家看一下。

可以看到 画方框的,我们通过反射(javassist)都可以获取到,既然我们都能获取到,那我们是不是就能封装,转发调用了呢?

一个简单的mvc就用到这些技术,所以上面这些看懂了的话,你也能写一个mvc了。

给大家留一个问题:

怎么通过反射把页面传过来的参数 user 封装起来。

如果你能封装起来,我想就能通过invoke来调用action参数了吧?

如果不明白没问题,我后面会教大家怎么做。如果会的话,就继续做用invoke来调用action方法吧。

提示:

1.反射
2.request.getParameterMap

 

写这么多实在太累了。。。

明天可能会有一更,周五至周末要去参加婚礼,所以没时间更了。。。

希望大家把代码都自己写一遍。不要copy。能记住才是自己的。。。

谢谢观看。。。

ps:有多少人看?

 

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。