当前位置: 首页 > news >正文

学习记录:DAY28

DispatcherController 功能完善与接口文档编写

前言


没什么动力说废话了。
今天来完善 DispatcherController 的功能,然后写写接口文档。


日程


  • 早上:本来只有早八,但是早上摸鱼了,罪过罪过。
  • 下午:把 DispatcherController 完善得比较充足了(我认为的哈)。
  • 晚上 10 点:现在的时间,deepseek 卡爆了,先来写写 blog。
  • 晚上 10 点半:摆烂了,开摆!

学习内容


省流

  1. DispatcherController 完善

1. DispatcherController 完善(有点水了)

主要是兼容各种询问参数,注解如下:

@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface KatPathVariable {String value() default "";
}@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface KatRequestBody {boolean required() default true;
}@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface KatRequestParam {String value() default "";  // 参数名boolean required() default true;  // 是否必须String defaultValue() default "";  // 默认值
}
1)路径参数 KatPathVariable
private Object resolvePathVariable(KatPathVariable annotation,java.lang.reflect.Parameter parameter,Class<?> paramType,HttpServletRequest req) {Map<String, String> pathVariables = (Map<String, String>) req.getAttribute("pathVariables");// 获取参数名,优先使用注解值,其次使用参数名String paramName = annotation.value().isEmpty()? parameter.getName(): annotation.value();String value = pathVariables.get(paramName);if (value == null) {throw new IllegalArgumentException("Path variable '" + paramName + "' not found");}try {// 使用 TypeConverter 进行类型转换return TypeConverter.convertValue(value, paramType);} catch (IllegalArgumentException e) {throw new IllegalArgumentException(String.format("Failed to convert path variable '%s' value '%s' to type %s",paramName, value, paramType.getName()),e);}
}

TypeConverter 是从原来 SimpleMapper 类中拆出来的方法:

public static Object convertValue(Object value, Class<?> targetType) {if (value == null) {if (targetType == Number.class) { // Number类型检查return 0;}return null;}if (targetType.isInstance(value)) {return value;}// 数值类型转换if (value instanceof Number number) {if (targetType == Double.class || targetType == double.class) {return number.doubleValue();}if (targetType == Float.class || targetType == float.class) {return number.floatValue();}if (targetType == Integer.class || targetType == int.class) {return number.intValue();}// 其他数值类型省略...}// 字符串到其他类型的转换if (value instanceof String strValue) {try {if (targetType == Integer.class || targetType == int.class) {return Integer.parseInt(strValue);}if (targetType == Long.class || targetType == long.class) {return Long.parseLong(strValue);}// 其他类型省略...} catch (NumberFormatException e) {throw new IllegalArgumentException("Failed to convert string '" + strValue +"' to type " + targetType.getName(), e);}}// 日期类型转换if (value instanceof java.sql.Date sqlDate) {if (targetType == LocalDate.class) {return sqlDate.toLocalDate();}if (targetType == LocalDateTime.class) {return sqlDate.toLocalDate().atStartOfDay();}}// 布尔类型转换、时间戳转换等省略...log.warn("Cannot convert value '{}' of type {} to target type {}",value, value.getClass().getName(), targetType.getName());throw new IllegalArgumentException("Cannot convert value '" + value +"' of type " + value.getClass().getName() +" to target type " + targetType.getName());
}
2)询问参数 KatRequestParam

也是差不多的逻辑:

private Object resolveRequestParam(KatRequestParam annotation,java.lang.reflect.Parameter parameter,Class<?> paramType,HttpServletRequest req) {// 获取参数名,优先使用注解值,其次使用参数名String paramName = annotation.value().isEmpty()? parameter.getName(): annotation.value();String paramValue = req.getParameter(paramName);// 处理参数缺失情况if (paramValue == null || paramValue.isEmpty()) {if (annotation.required()) {throw new IllegalArgumentException("Required request parameter '" + paramName + "' is not present");}if (!annotation.defaultValue().isEmpty()) {paramValue = annotation.defaultValue();} else {return null; // 非必需且无默认值,返回null}}try {// 使用 TypeConverter 进行类型转换return TypeConverter.convertValue(paramValue, paramType);} catch (IllegalArgumentException e) {throw new IllegalArgumentException(String.format("Failed to convert request parameter '%s' value '%s' to type %s",paramName, paramValue, paramType.getName()),e);}
}
3)请求体参数 KatRequestBody
private Object resolveRequestBody(Class<?> paramType,HttpServletRequest req,KatRequestBody annotation) throws IOException {// 检查请求体是否为空if (req.getContentLength() == 0) {if (annotation.required()) {throw new IllegalArgumentException("Required request body is missing");}return null;}try {String requestBody = ServletUtils.getRequestBody(req); // 如果是String类型,直接返回if (paramType.equals(String.class)) {return requestBody;}// 构建对象return JsonUtils.parseJson(requestBody, paramType); //这里借助了Jakson工具} catch (Exception e) {throw new IllegalArgumentException("Failed to parse request body", e);}
}
4)对应代理方法的装配也有比较大的改动
private Object invokeHandlerMethod(HandlerMethod handler,HttpServletRequest req,HttpServletResponse resp) throws Exception {Method method = handler.method();Object[] args = new Object[method.getParameterCount()];Class<?>[] paramTypes = method.getParameterTypes();Annotation[][] paramAnnotations = method.getParameterAnnotations(); //一个for (int i = 0; i < paramTypes.length; i++) {// 处理 @KatPathVariable 注解参数KatPathVariable pathVar = findAnnotation(paramAnnotations[i],KatPathVariable.class);if (pathVar != null) {args[i] = resolvePathVariable(pathVar, method.getParameters()[i],paramTypes[i], req);continue;}// 处理 @KatRequestParam 注解参数KatRequestParam requestParam = findAnnotation(paramAnnotations[i], KatRequestParam.class);if (requestParam != null) {args[i] = resolveRequestParam(requestParam,method.getParameters()[i], paramTypes[i], req);continue;}// 处理 @KatRequestBody 注解参数KatRequestBody requestBody = findAnnotation(paramAnnotations[i], KatRequestBody.class);if (requestBody != null) {args[i] = resolveRequestBody(paramTypes[i], req, requestBody);continue;}// 处理 HttpServletRequest/HttpServletResponse 参数if (paramTypes[i].equals(HttpServletRequest.class)) {args[i] = req;} else if (paramTypes[i].equals(HttpServletResponse.class)) {args[i] = resp;}}return method.invoke(handler.controllerInstance(), args);
}

结语


不知不觉,不知不觉,已经 5 月 9 号了。
项目只剩下 12 天了,我真的把握好时间了吗?


http://www.xdnf.cn/news/355861.html

相关文章:

  • Xcode16.3配置越狱开发环境
  • 武汉火影数字|数字科技馆打造:开启科技探索新大门
  • 深入理解 Java 代理模式:从基础到实战​
  • BP神经网络
  • 【PmHub后端篇】PmHub整合TransmittableThreadLocal (TTL)缓存用户数据
  • Python代码编程基础
  • 使用JMETER中的JSON提取器实现接口关联
  • onResume()和 onPause()的触发条件
  • 7、三维机械设计、装配与运动仿真组件 - /设计与仿真组件/3d-mechanical-designer
  • c/c++的Libevent 和OpenSSL构建HTTPS客户端详解(附带源码)
  • 基于设备指纹识别的反爬虫技术:给设备办 “身份证”
  • 【MySQL】-- 事务
  • 机器学习之数据转换策略
  • Java 23种设计模式 - 结构型模式7种
  • 数据库故障排查指南
  • React+Taro选择日期组件封装
  • 51c自动驾驶~合集40
  • 新品:同等小体积通信距离翻一倍-RF3060F27通信模块
  • Vmware 最新下载教程和安装教程,外带免下载文件
  • project从入门到精通(四)
  • idea spring boot 打包成可执行的 JAR包
  • 使用docker安装Dinky
  • `timescale 1ns/1ps的意义
  • 【250GB空间不够用】
  • 发那科机器人4(编程实例)
  • [Unity]-[UI]-[Image] 关于UI精灵图资源导入设置的详细解释
  • 日语学习-日语知识点小记-构建基础-JLPT-N4阶段(17):「 」と言いました
  • 芋道框架 账号未登录、租户标识未传递
  • 云效 MCP Server:AI 驱动的研发协作新范式
  • # YOLOv2:目标检测的升级之作