Java反射调用ashx

2023-10-31 05:52
文章标签 java 调用 反射 ashx

本文主要是介绍Java反射调用ashx,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这篇文章卡了大概一周,一个是没时间,只能带娃加锻炼间隙挤点时间,一个是碰到了问题卡住了。本篇实现反射调用ashx实现类的基础结构。

首先申明ashx的接口,所有的ashx实现类继承实现该接口的基类

package appcode;
import java.io.*;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
//http执行请求接口
public interface IBaseHttpHandler {/// <summary>/// 执行请求/// </summary>/// <param name="request">请求对象</param>/// <param name="response">响应对象</param>public void ProcessRequest(HttpServletRequest request, HttpServletResponse response);
}

然后实现第一个不需要会话的基类

package appcode;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.PrintWriter;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.lang.Class;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.jar.JarFile;
import java.util.jar.JarEntry;
import LIS.Core.MultiPlatform.LISContext;
import appcode.IBaseHttpHandler;
//实现不带会话的请求基类
public class BaseHttpHandlerNoSession implements IBaseHttpHandler {///请求对象public HttpServletRequest Request=null;///响应对象HttpServletResponse Response=null;///输出数据到前台private void Write(PrintWriter writer,String str){writer.println(str);writer.flush();writer.close();}/// <summary>/// 执行请求/// </summary>/// <param name="request">请求对象</param>/// <param name="response">响应对象</param>public void ProcessRequest(HttpServletRequest request, HttpServletResponse response){try {Request = request;Response = response;//前台传入的要调用的方法String method = LISContext.GetRequest(request, "Method");if (method != "") {//得到方法Method m = this.getClass().getMethod(method);if (m != null) {//执行Object ret = m.invoke(this);PrintWriter writer = response.getWriter();if (ret == null) {Write(writer, "");} else {Write(writer, (String) ret);}} else {//没有指定名称的方法}} else {//没按要求传调用方法}}catch (Exception ex){ex.printStackTrace();}}
}

然后实现主中间件中转调用实现的ashx

import appcode.IBaseHttpHandler;import java.io.*;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.util.concurrent.ConcurrentHashMap;
import java.net.URL;
import java.net.URLClassLoader;import java.util.*;@javax.servlet.annotation.WebServlet(name = "MianMiddleware")
public class MianMiddleware extends javax.servlet.http.HttpServlet {/// <summary>/// 缓存路径和类型,允许多线程读一个线程写/// </summary>private static ConcurrentHashMap<String, Class> hsType = new ConcurrentHashMap<>();///执行post请求@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.setContentType("text/html");request.setCharacterEncoding("UTF-8");response.setCharacterEncoding("UTF-8");String url=request.getRequestURI();//解析得到类名String className = url.split("\\.")[0];PrintWriter writer = response.getWriter();//反射得到类型Object objDeal = GetObjectByConfString(className,writer);//转换处理接口if(objDeal!=null){//转换成接口appcode.IBaseHttpHandler baseDeal=(appcode.IBaseHttpHandler)objDeal;baseDeal.ProcessRequest(request,response);}else{Write(writer,"为找到名称为:"+className+"的处理类");}}/// <summary>/// 通过配置得当对象/// </summary>/// <param name="confStr">配置/UI/sys/ashx/ashCommonNoSession</param>/// <returns></returns>public static Object GetObjectByConfString(String confStr,PrintWriter writer) {try {//取第一部分if (confStr.charAt(0) == '/') {confStr = confStr.substring(1);}if (!hsType.containsKey(confStr)) {URL urlBase = new File("D:\\Java业余\\jbase\\Lib\\appcode.jar").toURI().toURL();//自己生成jar包路径URL url = new File("D:\\Java业余\\jbase\\Lib\\AshDemo.jar").toURI().toURL();URL[] urls = new URL[]{urlBase,url};//加载程序集,这里很重要,一定要指定父加载器,否则加载的类和父加载器的类不认为是一个类URLClassLoader loader = new URLClassLoader(urls, MianMiddleware.class.getClassLoader());ClassLoader  loader1=Thread.currentThread().getContextClassLoader();//加载类Class c = loader.loadClass("AshDemo.ashDemo");//先写死,后面执行编译和从jar包反射hsType.put(confStr, c);}Class c = hsType.get(confStr);//创建对象Object o = c.newInstance();return o;}catch (Exception ex) {ex.printStackTrace();}return null;}//get直接走post的逻辑@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {doPost(request,response);}///输出数据到前台private static void Write(PrintWriter writer,String str){writer.println(str);writer.flush();writer.close();}}

在这里插入图片描述

然后实现一个ashx的demo
在这里插入图片描述

测试
在这里插入图片描述

在这里插入图片描述

本次碰到的卡壳问题就是通过URLClassLoader加载的类和创建的对象无法转换成接口类型。明明是一个类型,运行总是报错无法转换。各方资料查看加上自己尝试,发现要指定URLClassLoader类加载器的父加载器,这样父加载器加载的类就不会被重复加载导致类型不匹配了。

下个阶段就是实现文件变化监控和比对ashx实现类代码和jar包时间自动编译jar包供反射调用了。为下图的每个ashx的Java类生成jar包供反射使用而达到脚本化的目的,这样就可以实现业务脚本化和虚拟M层,用虚拟M来兼容打印和仪器接口等。

在这里插入图片描述

卡壳问题没人询问是真痛苦,卡过去了收获不少,有些事看着简单,动手做多多少少还是有些意外的

这篇关于Java反射调用ashx的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/313265

相关文章

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1