当前位置: 首页 > 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 天了,我真的把握好时间了吗?


相关文章:

  • 7.3.隐私合规
  • [春秋云镜] Brute4Road 仿真场景
  • 使用JMETER中的JSON提取器实现接口关联
  • ASP.NET中Tailspin Travel的UI层奥秘分析
  • 电机密集型工厂环境下的无线通信技术选型与优化策略
  • C++:书架
  • SPDK NVMe of RDMA 部署
  • 软考速通攻略之邪修提醒篇
  • 命名视图~
  • Node.js数据抓取技术实战示例
  • BERT模型讲解
  • AI汽车时代的全面赋能者:德赛西威全栈能力再升级
  • 【Python】超全常用 conda 命令整理
  • C++中static关键字详解:不同情况下的使用方式
  • 浏览器节能机制导致Websocket断连的坑
  • CentOS 7 系统下安装 OpenSSL 1.0.2k 依赖问题的处理
  • [sklearn] 特征工程
  • 支付宝沙盒模式商家转账经常出现 响应异常: 解包错误
  • 嵌入式裸机模块——软定时器
  • 数据结构-堆
  • “苏河超级管”调研:桥下公园“留白”很好,指引差点
  • 工程院院士葛世荣获聘任为江西理工大学校长
  • 2025年上海市模范集体、劳动模范和先进工作者名单揭晓
  • 心相印回应官方旗舰店客服辱骂消费者:正排查
  • 19岁女生注射头孢离世后续:院方道歉,医生停职,监管介入
  • 硅料巨亏后弘元绿能割肉求生:逾12亿元内蒙古公司股权转让协鑫