接触java一年多,一直在研究js,最近闲下来准备看看spring的基本原理,目前先研究下自定义注解

spring有个转发的dispatchServlet,所有的请求都是由它转发的,那么它是怎么工作的呢。。我没看源码,自己想了一种方式,我觉得spring应该也是这个原理

1.首先你也得在web.xml配置一个servlet  设置成初始化直接启动

	<servlet>
		<servlet-name>bookServlet</servlet-name>
		<servlet-class>book.ActionBookServlet</servlet-class>
		<load-on-startup>1</load-on-startup>
	</servlet>
	<servlet-mapping>
		<servlet-name>bookServlet</servlet-name>
		<url-pattern>*.do</url-pattern>
	</servlet-mapping>

2.自定义一个注解类

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.TYPE) // 类型上
public @interface MyAnnoation {
	String name() default "张三";
	String id() default "444";
}

3.定义一个自己写的action,并使用注解,注解中就是要拦截的地址,e

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import test.annotation.MyAnnoation;

@MyAnnoation(name="aa.do")
public class EntityBook {

	private int num =1;
	public String exceute(HttpServletRequest req,HttpServletResponse res){
		
		System.out.println("我是num"+num);
		num++;
		return "index.html";
	}
}
4.在servlet的init方法里扫描注解的类,并注册到缓存中

package book;

import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.io.PrintWriter;
import java.lang.reflect.Method;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import test.annotation.MyAnnoation;

import com.mxgraph.examples.swing.examlpeToWeb;

public class ActionBookServlet extends HttpServlet{
	Map<String,Method> methods = Collections.synchronizedMap(new HashMap<String,Method>());
	
	String packagename = "test.annotation";
	Map<String,Object> objs = Collections.synchronizedMap(new HashMap<String,Object>());
	@Override
	public void init() throws ServletException {
		try {
			Set<Class<?>> classes = getClasses(packagename);
			for(Class<?> c : classes){
				MyAnnoation annotation = c.getAnnotation(MyAnnoation.class);
				if(annotation!=null){//
					Method method = c.getMethod("execute", HttpServletRequest.class,HttpServletResponse.class);
					Object newInstance = c.newInstance();
					objs.put(annotation.name(), newInstance);
					methods.put(annotation.name(), method);
					System.out.println(annotation.name());
				}
			}
			
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		
		
		
		super.init();
	}
	
	protected void service(HttpServletRequest req, HttpServletResponse res) {
		String requestURI = req.getRequestURI();
		requestURI = requestURI.substring(requestURI.lastIndexOf("/")+1);
		System.out.println(requestURI);
		Object object = objs.get(requestURI);
		if(object != null){
			 Method method = methods.get(requestURI);
			 try{
				 String invoke = (String)method.invoke(object, req,res);
				 req.getRequestDispatcher(invoke).forward(req, res);
				 return;
			 }catch(Exception e){
				 e.printStackTrace();
			 }
		}
		res.setCharacterEncoding("utf-8");
		String action = req.getParameter("action");
		PrintWriter out=null;
		try {
			out = res.getWriter();
			if("query".equals(action)){
				req.getRequestDispatcher("hubei/result.jsp").forward(req, res);
			}else if("show".equals(action)){
				res.sendRedirect("hubei/book.jsp");
			}else if("mxg".equals(action)){
				out.println(userGraph());
			}
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
   private String userGraph() throws Exception{
  	 String xml = examlpeToWeb.start();
  	 xml = xml.replaceAll("noEdgeStyle=1", "");
//  	 draw2d d = new draw2d();
//  	 String xml = d.xml;
  	 return xml;
   }
   
   public static Set<Class<?>> getClasses(String pack) {  
  	  
     // 第一个class类的集合  
     Set<Class<?>> classes = new LinkedHashSet<Class<?>>();  
     // 是否循环迭代  
     boolean recursive = true;  
     // 获取包的名字 并进行替换  
     String packageName = pack;  
     String packageDirName = packageName.replace('.', '/');  
     // 定义一个枚举的集合 并进行循环来处理这个目录下的things  
     Enumeration<URL> dirs;  
     try {  
         dirs = Thread.currentThread().getContextClassLoader().getResources(  
                 packageDirName);  
         // 循环迭代下去  
         while (dirs.hasMoreElements()) {  
             // 获取下一个元素  
             URL url = dirs.nextElement();  
             // 得到协议的名称  
             String protocol = url.getProtocol();  
             // 如果是以文件的形式保存在服务器上  
             if ("file".equals(protocol)) {  
                 System.err.println("file类型的扫描");  
                 // 获取包的物理路径  
                 String filePath = URLDecoder.decode(url.getFile(), "UTF-8");  
                 // 以文件的方式扫描整个包下的文件 并添加到集合中  
                 findAndAddClassesInPackageByFile(packageName, filePath,  
                         recursive, classes);  
             } else if ("jar".equals(protocol)) {  
                 // 如果是jar包文件  
                 // 定义一个JarFile  
                 System.err.println("jar类型的扫描");  
                 JarFile jar;  
                 try {  
                     // 获取jar  
                     jar = ((JarURLConnection) url.openConnection())  
                             .getJarFile();  
                     // 从此jar包 得到一个枚举类  
                     Enumeration<JarEntry> entries = jar.entries();  
                     // 同样的进行循环迭代  
                     while (entries.hasMoreElements()) {  
                         // 获取jar里的一个实体 可以是目录 和一些jar包里的其他文件 如META-INF等文件  
                         JarEntry entry = entries.nextElement();  
                         String name = entry.getName();  
                         // 如果是以/开头的  
                         if (name.charAt(0) == '/') {  
                             // 获取后面的字符串  
                             name = name.substring(1);  
                         }  
                         // 如果前半部分和定义的包名相同  
                         if (name.startsWith(packageDirName)) {  
                             int idx = name.lastIndexOf('/');  
                             // 如果以"/"结尾 是一个包  
                             if (idx != -1) {  
                                 // 获取包名 把"/"替换成"."  
                                 packageName = name.substring(0, idx)  
                                         .replace('/', '.');  
                             }  
                             // 如果可以迭代下去 并且是一个包  
                             if ((idx != -1) || recursive) {  
                                 // 如果是一个.class文件 而且不是目录  
                                 if (name.endsWith(".class")  
                                         && !entry.isDirectory()) {  
                                     // 去掉后面的".class" 获取真正的类名  
                                     String className = name.substring(  
                                             packageName.length() + 1, name  
                                                     .length() - 6);  
                                     try {  
                                         // 添加到classes  
                                         classes.add(Class  
                                                 .forName(packageName + '.'  
                                                         + className));  
                                     } catch (ClassNotFoundException e) {  
                                         // log  
                                         // .error("添加用户自定义视图类错误 找不到此类的.class文件");  
                                         e.printStackTrace();  
                                     }  
                                 }  
                             }  
                         }  
                     }  
                 } catch (IOException e) {  
                     // log.error("在扫描用户定义视图时从jar包获取文件出错");  
                     e.printStackTrace();  
                 }  
             }  
         }  
     } catch (IOException e) {  
         e.printStackTrace();  
     }  

     return classes;  
 }  
   /** 
    * 以文件的形式来获取包下的所有Class 
    *  
    * @param packageName 
    * @param packagePath 
    * @param recursive 
    * @param classes 
    */  
   public static void findAndAddClassesInPackageByFile(String packageName,  
           String packagePath, final boolean recursive, Set<Class<?>> classes) {  
       // 获取此包的目录 建立一个File  
       File dir = new File(packagePath);  
       // 如果不存在或者 也不是目录就直接返回  
       if (!dir.exists() || !dir.isDirectory()) {  
           // log.warn("用户定义包名 " + packageName + " 下没有任何文件");  
           return;  
       }  
       // 如果存在 就获取包下的所有文件 包括目录  
       File[] dirfiles = dir.listFiles(new FileFilter() {  
           // 自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)  
           public boolean accept(File file) {  
               return (recursive && file.isDirectory())  
                       || (file.getName().endsWith(".class"));  
           }  
       });  
       // 循环所有文件  
       for (File file : dirfiles) {  
           // 如果是目录 则继续扫描  
           if (file.isDirectory()) {  
               findAndAddClassesInPackageByFile(packageName + "."  
                       + file.getName(), file.getAbsolutePath(), recursive,  
                       classes);  
           } else {  
               // 如果是java类文件 去掉后面的.class 只留下类名  
               String className = file.getName().substring(0,  
                       file.getName().length() - 6);  
               try {  
                   // 添加到集合中去  
                   //classes.add(Class.forName(packageName + '.' + className));  
                                        //经过回复同学的提醒,这里用forName有一些不好,会触发static方法,没有使用classLoader的load干净  
                                       classes.add(Thread.currentThread().getContextClassLoader().loadClass(packageName + '.' + className));    
                               } catch (ClassNotFoundException e) {  
                   // log.error("添加用户自定义视图类错误 找不到此类的.class文件");  
                   e.printStackTrace();  
               }  
           }  
       }  
   }  
}



Logo

CSDN联合极客时间,共同打造面向开发者的精品内容学习社区,助力成长!

更多推荐