public class WebServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {req.setCharacterEncoding("utf-8");Enumeration<String> parameterNames = req.getParameterNames();String contextPath = req.getServletPath();System.out.println(contextPath);Method method = Containers.mappingMethods.get(contextPath);if (method == null) {//请求路径不存在时 返回404 也可以重定向或转发到404页面resp.sendError(404, "页面不存在");} else {//获取请求类型String contentType = req.getContentType();String method1 = req.getMethod();System.out.println("method:" + method1 + " contentType:" + contentType);try {//获取方法所在的类并实例化Object o = method.getDeclaringClass().newInstance();//获取方法所有的对象Parameter[] parameters = method.getParameters();//存放方法参数解析注入后的值 参数列表ArrayList<Object> params = new ArrayList<>();//获取对应参数解析器,注入参数for (Parameter parameter : parameters) {Object o2 = null;//获取解析器List<BaseResolver> resolvers = Containers.resolvers;for (BaseResolver resolver : resolvers) {//判断是否符合预期类型if (resolver.supportsParameter(parameter)) {//符合 进行解析o2 = resolver.resolveArgument(parameter, req, resp);break;}}//保存解析后的值params.add(o2);}//根据方法所在类及所需参数通过反射调用方法Object invoke = method.invoke(o, params.toArray());if (invoke != null) {//判断方法是否标记了ResponseBody注解ResponseBody annotation = method.getAnnotation(ResponseBody.class);if (annotation != null) {//有 ResponseBody 注解并且是String类型 直接响应请求if (invoke instanceof String) {resp.setContentType("text/html");resp.setCharacterEncoding("UTF-8");// 获取PrintWriter对象PrintWriter out = resp.getWriter();out.print(invoke);// 释放PrintWriter对象out.flush();out.close();} else {//如果不是String类型则装换成JSON格式字符串在进行响应resp.setContentType("application/json; charset=utf-8");resp.setCharacterEncoding("UTF-8");// 获取PrintWriter对象PrintWriter out = resp.getWriter();out.print(JSON.toJSONString(invoke));// 释放PrintWriter对象out.flush();out.close();}} else {//没有标记注解 返回String类型 则转发到返回字符串所对应的jsp页面if (invoke instanceof String) {RequestDispatcher dispatcher = req.getRequestDispatcher("/WEB-INF/views/" + invoke + ".jsp");dispatcher.forward(req, resp);} else {new RuntimeException("返回值不是String类型 且没有@ResponseBody注解");}}}} catch (Exception e) {e.printStackTrace();}}}@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {doGet(req, resp);}5.所用到的注解标记
RequestBody
@Target({ElementType.TYPE, ElementType.PARAMETER})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface RequestBody {}RequestMapping
@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface RequestMapping {String value() default "";}ResponseBody
@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface ResponseBody {}6.所用到工具类
ClassUtils
public class ClassUtils {public static Set<Class<?>> getClasses(String pack) {// 第一个class类的集合Set<Class<?>> classes = new LinkedHashSet<Class<?>>();// 是否循环迭代boolean recursive = true;// 获取包的名字 并进行替换String packageName = pack;String packageDirName = packageName.replace('.', '/');// 定义一个枚举的集合 并进行循环来处理这个目录下的thingsEnumeration<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包文件// 定义一个JarFileSystem.err.println("jar类型的扫描");JarFile jar;try {// 获取jarjar = ((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 {// 添加到classesclasses.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;}public static void findAndAddClassesInPackageByFile(String packageName,String packagePath, final boolean recursive, Set<Class<?>> classes) {// 获取此包的目录 建立一个FileFile dir = new File(packagePath);// 如果不存在或者 也不是目录就直接返回if (!dir.exists() || !dir.isDirectory()) {// log.warn("用户定义包名 " + packageName + " 下没有任何文件");return;}// 如果存在 就获取包下的所有文件 包括目录File[] dirfiles = dir.listFiles(new FileFilter() {// 自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)@Overridepublic 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();}}}}}
推荐阅读
- 12 个优化 Docker 镜像安全性的技巧
- 热血传奇|热血传奇:战士一刀一个法师?后期法师就真的弱吗?
- 羊毛被怎么样哪个牌子比较好
- 取卵后小腹疼痛
- 手腕一样粗的阴茎?
- 龟头表皮褶皱
- 科学变美的100个基本|网红必读“教材”《科学变美的100个基本》,职场小白也用得上
- 如何轻松食用香螺
- 如何挑选慈菇
- 如何挑选空心菜
