自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-@RequestParam
😀前言
自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-@RequestParam
🏠个人主页:尘觉主页
🧑个人简介:大家好,我是尘觉,希望我的文章可以帮助到大家,您的满意是我的动力😉😉
在csdn获奖荣誉: 🏆csdn城市之星2名
💓Java全栈群星计划top前5
🤗 端午大礼包获得者
💕欢迎大家:这里是CSDN,我总结知识的地方,欢迎来到我的博客,感谢大家的观看🥰
如果文章有什么需要改进的地方还请大佬不吝赐教 先在次感谢啦😊
文章目录
- 💫实现任务阶段 6- 完成控制器方法获取参数-@RequestParam
- 功能说明:
- 完成任务说明
- - 后端 Handler 的目标方法
- 😄代码完成
- 🥰完成: 将方法的 HttpServletRequest 和 HttpServletResponse 参数封装到参数数组,进行反射调用
- 修改WyxDispatcherServlet.java类
- 完成测试
- ❤️🔥完成: 在方法参数 指定 @RequestParam 的参数封装到参数数组,进行反射调用
- - 测试页面
- - 后端 Handler 的目标方法
- 创建自定义注解RequestParam
- 修改WyxDispatcherServlet.java接口类
- 修改MonsterSer viceImpl.java实现类
- 修改MonsterController
- 修改WyxDispatcherServlet.java类
- getIndexRequestParameterIndex方法
- 完成测试 (Redeploy Tomcat 即 可 )
- 💞完成: 在方法参数 没有指定 @RequestParam ,按照默认参数名获取值, 进行反射调用
- 修改MonsterController
- 修改WyxDispatcherServlet.java类
- 得 到 控 制 器 方 法 的 参 数 名 ,
- 💖注意:
- 点击 maven 管理,clean 项目,在重启一下 tomcat ,完成测试
- 后台:
- 如果请求参数和方法参数不一致:
- 😄总结
💫实现任务阶段 6- 完成控制器方法获取参数-@RequestParam
功能说明:
自定义@RequestParam 和 方法参数名获取参数
完成任务说明
- 后端 Handler 的目标方法
@RequestMapping(value = “/monster/find”)
public void findMonstersByName(HttpServletRequest request,
HttpServletResponse response,
@RequestParam(value = “name”) String name) {
//代码…
}
😄代码完成
🥰完成: 将方法的 HttpServletRequest 和 HttpServletResponse 参数封装到参数数组,进行反射调用
● 代码实现, 说明,整个实现思路,就是参考 SpringMVC 规范
修改WyxDispatcherServlet.java类
private void executeDispatch(HttpServletRequest request,HttpServletResponse response) {WyxHandler wyxHandler = getWyxHandler(request);try {if (null == wyxHandler) {//说明用户请求的路径/资源不存在response.getWriter().print("<h1>404 NOT FOUND</h1>");} else {//匹配成功, 反射调用控制器的方法//目标将: HttpServletRequest 和 HttpServletResponse封装到参数数组//1. 得到目标方法的所有形参参数信息[对应的数组]Class<?>[] parameterTypes =wyxHandler.getMethod().getParameterTypes();//2. 创建一个参数数组[对应实参数组], 在后面反射调用目标方法时,会使用到Object[] params =new Object[parameterTypes.length];//3遍历parameterTypes形参数组,根据形参数组信息,将实参填充到实参数组for (int i = 0; i < parameterTypes.length; i++) {//取出每一个形参类型Class<?> parameterType = parameterTypes[i];//如果这个形参是HttpServletRequest, 将request填充到params//在原生SpringMVC中,是按照类型来进行匹配,这里简化使用名字来进行匹配if ("HttpServletRequest".equals(parameterType.getSimpleName())) {params[i] = request;} else if ("HttpServletResponse".equals(parameterType.getSimpleName())) {params[i] = response;}}//wyxHandler.getMethod()// .invoke(wyxHandler.getController(),request,response);//反射调用目标方法Object result = wyxHandler.getMethod().invoke(wyxHandler.getController(), params);
}
完成测试
(启动 tomcat), 浏览器输入 http://localhost:8080/monster/list , 仍然可以看 到正确的返回
❤️🔥完成: 在方法参数 指定 @RequestParam 的参数封装到参数数组,进行反射调用
- 测试页面
- 后端 Handler 的目标方法
@RequestMapping(value = “/monster/find”)
public void findMonstersByName(HttpServletRequest request,
HttpServletResponse response,
@RequestParam(value = “name”) String name) {
//代码… }
创建自定义注解RequestParam
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RequestParam {String value() default "";
}
修改WyxDispatcherServlet.java接口类
public interface MonsterService {public List<Monster> listMonsters();public List<Monster> findMonstersByName(String name);
}
修改MonsterSer viceImpl.java实现类
@Service
public class MonsterServiceImpl implements MonsterService {@Overridepublic List<Monster> listMonster() {//这里就模拟数据->DBList<Monster> monsters =new ArrayList<>();monsters.add(new Monster(100, "牛魔王", "芭蕉扇", 400));monsters.add(new Monster(200, "老猫妖怪", "抓老鼠", 200));return monsters;}@Overridepublic List<Monster> findMonsterByName(String name) {//这里就模拟数据->DBList<Monster> monsters =new ArrayList<>();monsters.add(new Monster(100, "牛魔王", "芭蕉扇", 400));monsters.add(new Monster(200, "老猫妖怪", "抓老鼠", 200));monsters.add(new Monster(300, "大象精", "运木头", 100));monsters.add(new Monster(400, "黄袍怪", "吐烟雾", 300));monsters.add(new Monster(500, "白骨精", "美人计", 800));//创建集合返回查询到的monster集合List<Monster> findMonsters =new ArrayList<>();//遍历monsters,返回满足条件for (Monster monster : monsters) {if (monster.getName().contains(name)) {findMonsters.add(monster);}}return findMonsters;}
修改MonsterController
@Controller
public class MonsterController {//@AutoWired表示要完成属性的装配.@AutoWiredprivate MonsterService monsterService;@RequestMapping(value = "/monster/find")public void findMonsterByName(HttpServletRequest request,HttpServletResponse response,@RequestParam(value="name") String name) {//设置编码和返回类型response.setContentType("text/html;charset=utf-8");System.out.println("--接收到的name---" + name);StringBuilder content = new StringBuilder("<h1>妖怪列表信息</h1>");//调用monsterServiceList<Monster> monsters = monsterService.findMonsterByName(name);content.append("<table border='1px' width='400px' style='border-collapse:collapse'>");for (Monster monster : monsters) {content.append("<tr><td>" + monster.getId()+ "</td><td>" + monster.getName() + "</td><td>"+ monster.getSkill() + "</td><td>"+ monster.getAge() + "</td></tr>");}content.append("</table>");//获取writer返回信息try {PrintWriter printWriter = response.getWriter();printWriter.write(content.toString());} catch (IOException e) {e.printStackTrace();}}
}
修改WyxDispatcherServlet.java类
private void executeDispatch(HttpServletRequest request,HttpServletResponse response) {WyxHandler wyxHandler = getWyxHandler(request);try {if (null == wyxHandler) {//说明用户请求的路径/资源不存在response.getWriter().print("<h1>404 NOT FOUND</h1>");} else {//匹配成功, 反射调用控制器的方法//目标将: HttpServletRequest 和 HttpServletResponse封装到参数数组//1. 得到目标方法的所有形参参数信息[对应的数组]Class<?>[] parameterTypes =wyxHandler.getMethod().getParameterTypes();//2. 创建一个参数数组[对应实参数组], 在后面反射调用目标方法时,会使用到Object[] params =new Object[parameterTypes.length];//3遍历parameterTypes形参数组,根据形参数组信息,将实参填充到实参数组for (int i = 0; i < parameterTypes.length; i++) {//取出每一个形参类型Class<?> parameterType = parameterTypes[i];//如果这个形参是HttpServletRequest, 将request填充到params//在原生SpringMVC中,是按照类型来进行匹配,老师这里简化使用名字来进行匹配if ("HttpServletRequest".equals(parameterType.getSimpleName())) {params[i] = request;} else if ("HttpServletResponse".equals(parameterType.getSimpleName())) {params[i] = response;}}//将http请求参数封装到params数组中, 提示,要注意填充实参的时候,顺序问题//1. 获取http请求的参数集合//解读//http://localhost:8080/monster/find?name=牛魔王&hobby=打篮球&hobby=喝酒//2. 返回的Map<String,String[]> String:表示http请求的参数名// String[]:表示http请求的参数值,为什么是数组////处理提交的数据中文乱码request.setCharacterEncoding("utf-8");Map<String, String[]> parameterMap =request.getParameterMap();//2. 遍历parameterMap 将请求参数,按照顺序填充到实参数组paramsfor (Map.Entry<String, String[]> entry : parameterMap.entrySet()) {//取出key,这name就是对应请求的参数名String name = entry.getKey();//说明:这里只考虑提交的参数是单值的情况,即不考虑类似checkbox提示的数据// 这里做了简化,如果考虑多值情况,也不难..String value = entry.getValue()[0];//我们得到请求的参数对应目标方法的第几个形参,然后将其填充//这里专门编写一个方法,得到请求的参数对应的是第几个形参int indexRequestParameterIndex =getIndexRequestParameterIndex(wyxHandler.getMethod(), name);if (indexRequestParameterIndex != -1) {//找到对应的位置params[indexRequestParameterIndex] = value;} else {//一会写}/*** 解读* 1. 下面这样写法,其实是针对目标方法是 m(HttpServletRequest request , HttpServletResponse response)* 2. 这里准备将需要传递给目标方法的 实参=>封装到参数数组=》然后以反射调用的方式传递给目标方法* 3. public Object invoke(Object obj, Object... args)..*///wyxHandler.getMethod()// .invoke(wyxHandler.getController(),request,response);//反射调用目标方法wyxHandler.getMethod().invoke(wyxHandler.getController(), params);} catch (Exception e) {e.printStackTrace();}
getIndexRequestParameterIndex方法
public int getIndexRequestParameterIndex(Method method, String name) {//1.得到method的所有形参参数Parameter[] parameters = method.getParameters();for (int i = 0; i < parameters.length; i++) {//取出当前的形参参数Parameter parameter = parameters[i];//判断parameter是不是有@RequestParam注解boolean annotationPresent = parameter.isAnnotationPresent(RequestParam.class);if (annotationPresent) {//说明有@RequestParam//取出当前这个参数的 @RequestParam(value = "xxx")RequestParam requestParamAnnotation =parameter.getAnnotation(RequestParam.class);String value = requestParamAnnotation.value();//这里就是匹配的比较if (name.equals(value)) {return i;//找到请求的参数,对应的目标方法的形参的位置}}}//如果没有匹配成功,就返回-1return -1;}
完成测试 (Redeploy Tomcat 即 可 )
浏览器输入http://localhost:8080/monster/find?name=牛魔王
💞完成: 在方法参数 没有指定 @RequestParam ,按照默认参数名获取值, 进行反射调用
修改MonsterController
@RequestMapping(value = "/monster/find")public void findMonsterByName(HttpServletRequest request,HttpServletResponse response,String name) {//设置编码和返回类型response.setContentType("text/html;charset=utf-8");System.out.println("--接收到的name---" + name);StringBuilder content = new StringBuilder("<h1>妖怪列表信息</h1>");//调用monsterServiceList<Monster> monsters = monsterService.findMonsterByName(name);content.append("<table border='1px' width='400px' style='border-collapse:collapse'>");for (Monster monster : monsters) {content.append("<tr><td>" + monster.getId()+ "</td><td>" + monster.getName() + "</td><td>"+ monster.getSkill() + "</td><td>"+ monster.getAge() + "</td></tr>");}content.append("</table>");//获取writer返回信息try {PrintWriter printWriter = response.getWriter();printWriter.write(content.toString());} catch (IOException e) {e.printStackTrace();}}
修改WyxDispatcherServlet.java类
private void executeDispatch(HttpServletRequest request,HttpServletResponse response) {WyxHandler wyxHandler = getWyxHandler(request);try {if (null == wyxHandler) {//说明用户请求的路径/资源不存在response.getWriter().print("<h1>404 NOT FOUND</h1>");} else {//匹配成功, 反射调用控制器的方法//目标将: HttpServletRequest 和 HttpServletResponse封装到参数数组//1. 得到目标方法的所有形参参数信息[对应的数组]Class<?>[] parameterTypes =wyxHandler.getMethod().getParameterTypes();//2. 创建一个参数数组[对应实参数组], 在后面反射调用目标方法时,会使用到Object[] params =new Object[parameterTypes.length];//3遍历parameterTypes形参数组,根据形参数组信息,将实参填充到实参数组for (int i = 0; i < parameterTypes.length; i++) {//取出每一个形参类型Class<?> parameterType = parameterTypes[i];//如果这个形参是HttpServletRequest, 将request填充到params//在原生SpringMVC中,是按照类型来进行匹配,这里简化使用名字来进行匹配if ("HttpServletRequest".equals(parameterType.getSimpleName())) {params[i] = request;} else if ("HttpServletResponse".equals(parameterType.getSimpleName())) {params[i] = response;}}//将http请求参数封装到params数组中, 提示,要注意填充实参的时候,顺序问题//1. 获取http请求的参数集合//解读//http://localhost:8080/monster/find?name=牛魔王&hobby=打篮球&hobby=喝酒//2. 返回的Map<String,String[]> String:表示http请求的参数名// String[]:表示http请求的参数值,为什么是数组////处理提交的数据中文乱码request.setCharacterEncoding("utf-8");Map<String, String[]> parameterMap =request.getParameterMap();//2. 遍历parameterMap 将请求参数,按照顺序填充到实参数组paramsfor (Map.Entry<String, String[]> entry : parameterMap.entrySet()) {//取出key,这name就是对应请求的参数名String name = entry.getKey();//说明:这里只考虑提交的参数是单值的情况,即不考虑类似checkbox提示的数据// 这里做了简化,如果小伙伴考虑多值情况,也不难..String value = entry.getValue()[0];//我们得到请求的参数对应目标方法的第几个形参,然后将其填充//这里专门编写一个方法,得到请求的参数对应的是第几个形参int indexRequestParameterIndex =getIndexRequestParameterIndex(wyxHandler.getMethod(), name);if (indexRequestParameterIndex != -1) {//找到对应的位置params[indexRequestParameterIndex] = value;} else {//说明并没有找到@RequestParam注解对应的参数,就会使用默认的机制进行配置[待..]//思路//1. 得到目标方法的所有形参的名称-专门编写方法获取形参名//2. 对得到目标方法的所有形参名进行遍历,如果匹配就把当前请求的参数值,填充到paramsList<String> parameterNames =getParameterNames(wyxHandler.getMethod());for (int i = 0; i < parameterNames.size(); i++) {//如果请求参数名和目标方法的形参名一样,说明匹配成功if (name.equals(parameterNames.get(i))) {params[i] = value;//填充到实参数组break;}}}}/*** 解读* 1. 下面这样写法,其实是针对目标方法是 m(HttpServletRequest request , HttpServletResponse response)* 2. 这里准备将需要传递给目标方法的 实参=>封装到参数数组=》然后以反射调用的方式传递给目标方法* 3. public Object invoke(Object obj, Object... args)..*///wyxHandler.getMethod()// .invoke(wyxHandler.getController(),request,response);//反射调用目标方法Object result = wyxHandler.getMethod().invoke(wyxHandler.getController(), params);
} catch (Exception e) {
e.printStackTrace();
}
public List<String> getParameterNames(Method method) {List<String> parametersList = new ArrayList<>();//获取到所以的参数名->这里有一个小细节//在默认情况下 parameter.getName() 得到的名字不是形参真正名字//而是 [arg0, arg1, arg2...], 这里我们要引入一个插件,使用java8特性,这样才能解决Parameter[] parameters = method.getParameters();//遍历parameters 取出名称,放入parametersListfor (Parameter parameter : parameters) {parametersList.add(parameter.getName());}System.out.println("目标方法的形参列表=" + parametersList);return parametersList;}
得 到 控 制 器 方 法 的 参 数 名 ,
比 如 public void findMonstersByName(HttpServletRequest request, HttpServletResponse response, @RequestParam(value = “name”) String name) request, response, name
💖注意:
- 在默认情况下,返回的并不是 request, response ,name 而是 arg0, arg1,arg2
- 需要使用到 jdk8 的新特性,并需要在 pom.xml 配置 maven 编译插件(可以百度搜索到),才能得到 request, response, name
<plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><version>3.7.0</version><configuration><source>1.8</source><target>1.8</target><compilerArgs><arg>-parameters</arg></compilerArgs><encoding>utf-8</encoding></configuration></plugin>
点击 maven 管理,clean 项目,在重启一下 tomcat ,完成测试
后台:
如果请求参数和方法参数不一致:
😄总结
本文完成了任务阶段 6-完成控制器方法获取参数-@RequestParam下一阶段完成
实现任务阶段 7- 完成简单视图解析
😉自己实现 SpringMVC 底层机制 核心分发 控制器+ Controller 和 Service 注入容器 + 对象自动装配 + 控制器 方法获取参数 + 视图解析 + 返回 JSON 格式数系列
第一篇->自己实现 SpringMVC 底层机制 系列之搭建 SpringMVC 底层机制开发环境和开发 WyxDispatcherServlet_springmvc分发器
第二篇->自己实现 SpringMVC 底层机制 系列之–实现任务阶段 2- 完成客户端浏览器可以请求控制层
第三篇->自己实现 SpringMVC 底层机制 系列之–实现任务阶段 3- 从 web.xml动态获取 wyxspringmvc.xml
第四篇-> 自己实现 SpringMVC 底层机制 系列之-实现任务阶段 4- 完成自定义@Service 注解功能
第五篇-> 自己实现 SpringMVC 底层机制 系列之-实现任务阶段 5- 完成 Spring 容器对象的自动装配 -@Autowried
😁热门专栏推荐
想学习vue的可以看看这个
java基础合集
数据库合集
redis合集
nginx合集
linux合集
等等等还有许多优秀的合集在主页等着大家的光顾感谢大家的支持
🤔欢迎大家加入我的社区 尘觉社区
文章到这里就结束了,如果有什么疑问的地方请指出,诸佬们一起来评论区一起讨论😁
希望能和诸佬们一起努力,今后我们一起观看感谢您的阅读🍻
如果帮助到您不妨3连支持一下,创造不易您们的支持是我的动力🤞
相关文章:

自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-@RequestParam
😀前言 自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-RequestParam 🏠个人主页:尘觉主页 🧑个人简介:大家好,我是尘觉,希望我的文章可以帮助到大家,…...
数据可视化:图表绘制详解
数据可视化是一种将抽象的数字和数据转化为直观图形的技术,使数据的模式、趋势和关系一目了然。本文将详细介绍如何绘制各种类型的图表,包括柱状图、折线图、饼图、散点图和热力图等。 第一部分:图表类型和选择 1. 柱状图 柱状图是用于比较类…...

【中危】Apache Ivy<2.5.2 存在XXE漏洞 (CVE-2022-46751)
漏洞描述 Apache Ivy 是一个管理基于 ANT 项目依赖关系的开源工具,文档类型定义(DTD)是一种文档类型定义语言,它用于定义XML文档中所包含的元素以及元素之间的关系。 Apache Ivy 2.5.2之前版本中,当解析自身配置、Ivy 文件或 Apache Maven 的 POM 文件…...

C#使用自定义的比较器对版本号(编码)字符串进行排序
给定一些数据,如下所示: “1.10.1.1.1.2”, “1.1”, “2.2”, “1.1.1.1”, “1.1.3.1”, “1.1.1”, “2.10.1.1.1”, “1.1.2.1”, “1.2.1.1”, “2.5.1.1”, “1.10.1.1”, “1.10.2.1”, “1.11.3.1”, “1.11.12.1”, “1.11.11.1”, “1.11.3.1”, “1”, “…...

AI在日常生活中的应用:从语音助手到自动驾驶
文章目录 AI的定义和发展AI在日常生活中的应用1. **智能语音助手**2. **智能家居**3. **智能医疗**4. **自动驾驶** 代码示例:使用Python实现基于机器学习的图片分类AI的未来前景结论 🎉欢迎来到AIGC人工智能专栏~探索AI在日常生活中的应用 ☆* o(≧▽≦…...

Windows10查看图片的分辨率
文章目录 查看方法 查看方法 鼠标悬停在想查看分辨率大小的图片上,稍等那么零点几秒,就会弹出图片的分辨率信息,如图所示:...

Spring事务和事务传播机制(2)
前言🍭 ❤️❤️❤️SSM专栏更新中,各位大佬觉得写得不错,支持一下,感谢了!❤️❤️❤️ Spring Spring MVC MyBatis_冷兮雪的博客-CSDN博客 在Spring框架中,事务管理是一种用于维护数据库操作的一致性和…...

计算机视觉 -- 图像分割
文章目录 1. 图像分割2. FCN2.1 语义分割– FCN (Fully Convolutional Networks)2.2 FCN--deconv2.3 Unpool2.4 拓展–DeconvNet 3. 实例分割3.1 实例分割--Mask R-CNN3.2 Mask R-CNN3.3 Faster R-CNN与 Mask R-CNN3.4 Mask R-CNN:Resnet1013…...

ubuntu18.04复现yolo v8之CUDA与pytorch版本问题以及多CUDA版本安装及切换
最近在复现yolo v8的程序,特记录一下过程 环境:ubuntu18.04ros melodic 小知识:GPU并行计算能力高于CPU—B站UP主说的 Ubuntu可以安装多个版本的CUDA。如果某个程序的Pyorch需要不同版本的CUDA,不必删除之前的CUDA,…...

Redis三种模式——主从复制,哨兵模式,集群
目录 一、主从复制 1.1主从复制的概念 1.2Redis主从复制作用 1.2.1数据冗余 1.2.2故障恢复 1.2.3负载均衡 1.2.4高可用基石 1.3Redis主从复制流程 1.4部署Redis 主从复制 1.4.1.环境部署 1.4.2.所有服务器都先关闭防火墙 1.4.3.所有服务器都安装Redis 1.4.4修改Master主节点R…...
mysql8.0.31新增只读远程普通用户
在 MySQL 8.0.31 中,可以通过以下步骤新增只读远程普通用户: 1、使用 root 用户登录 MySQL 数据库。 mysql -u root -p 2、创建用户: CREATE USER username% IDENTIFIED WITH mysql_native_password BY password ; 其中,username…...

揭开路由协议隐藏的风险
路由协议在互联网和基于其的服务的运行中发挥着至关重要的作用。然而,许多这些协议的开发都没有考虑到安全问题。 例如,边界网关协议 (BGP) 最初并未考虑对等点之间发生攻击的可能性。过去几十年来,BGP 中的起源和路径验证已投入了大量工作。…...

图片因固定宽高被拉伸了?object-fit:一个神奇的属性
一、问题产生的场景 近期在完成项目开发时,测试人员针对漫画长图上传后的展示提出了一个界面优化的点,因为其特点是长,但是我们展示图片的区域是固定的,如果我们按照正常思路将图片的宽高写死,确实占位大小的问题解决了…...

客户案例:中圣科技—CAC2.0防范盗号威胁,加固安全防线
客户背景 中圣科技(江苏)股份有限公司(以下简称“中圣科技”),是一家以技术研发为驱动,以清洁能源核心成套装备和节能环保工程服务为支撑的科技创新型企业。其以南京为核心运营基地,与当地政府…...

pandas数据分析40——读取 excel 合并单元格的表头
案例背景 真的很容易疯....上班的单位的表格都是不同的人做的,所以就会出现各种合并单元格的情况,要知道我们用pandas读取数据最怕合并单元格了,因为没规律...可能前几列没合并,后面几列又合并了....而且pandas对于索引很严格&am…...

Java后端开发面试题——微服务篇总结
Spring Cloud 5大组件有哪些? 随着SpringCloudAlibba在国内兴起 , 我们项目中使用了一些阿里巴巴的组件 注册中心/配置中心 Nacos 负载均衡 Ribbon 服务调用 Feign 服务保护 sentinel 服务网关 Gateway Ribbon负载均衡策略有哪些 ? RoundRobinRule&…...
第十一章MyBatis查询专题
返回单个Car 返回单个可以直接用Car接收返回参数 Car carCarMapper.getOne(100);返回多个Car 返回多个可以直接用List接收返回参数 List<Car> carCarMapper.getAll();用一个对象无法接受返回多个参数,用list可以接收返回一个参数 返回Map 如果没有合适的…...

测试驱动开发(TDD)
测试驱动开发(TDD) 本篇文章简单叙述一下什么是测试驱动开发,以及怎么进行测试驱动开发! TDD (Test Driven Development):(源于极限编程(XP))在不…...

深度学习|CNN卷积神经网络
CNN卷积神经网络 解决的问题人类的视觉原理原理卷积层——提取特征池化层——数据降维全连接层——输出结果 应用图像处理自然语言处理 解决的问题 在CNN没有出现前,图像对人工智能来说非常难处理。 主要原因: 图像要处理的数据量太大了。图像由像素组…...

【洁洁送书第五期】为什么我们要了解可观测性工程
导读 可观测性已成为一个热门话题,并广受关注。随着它的普及,“可观测性”不幸被误作“监控”或“系统遥测”的同义词。可观测性是软件系统的一个特征。而且,只有当团队采用新的实践进行持续开发时,才能在生产软件系统中有效利用这…...
在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能
下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能,包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...
macOS多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用
文章目录 问题现象问题原因解决办法 问题现象 macOS启动台(Launchpad)多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用。 问题原因 很明显,都是Google家的办公全家桶。这些应用并不是通过独立安装的…...

Cloudflare 从 Nginx 到 Pingora:性能、效率与安全的全面升级
在互联网的快速发展中,高性能、高效率和高安全性的网络服务成为了各大互联网基础设施提供商的核心追求。Cloudflare 作为全球领先的互联网安全和基础设施公司,近期做出了一个重大技术决策:弃用长期使用的 Nginx,转而采用其内部开发…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
Spring AI与Spring Modulith核心技术解析
Spring AI核心架构解析 Spring AI(https://spring.io/projects/spring-ai)作为Spring生态中的AI集成框架,其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似,但特别为多语…...

html-<abbr> 缩写或首字母缩略词
定义与作用 <abbr> 标签用于表示缩写或首字母缩略词,它可以帮助用户更好地理解缩写的含义,尤其是对于那些不熟悉该缩写的用户。 title 属性的内容提供了缩写的详细说明。当用户将鼠标悬停在缩写上时,会显示一个提示框。 示例&#x…...

Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...
Java求职者面试指南:计算机基础与源码原理深度解析
Java求职者面试指南:计算机基础与源码原理深度解析 第一轮提问:基础概念问题 1. 请解释什么是进程和线程的区别? 面试官:进程是程序的一次执行过程,是系统进行资源分配和调度的基本单位;而线程是进程中的…...

在Mathematica中实现Newton-Raphson迭代的收敛时间算法(一般三次多项式)
考察一般的三次多项式,以r为参数: p[z_, r_] : z^3 (r - 1) z - r; roots[r_] : z /. Solve[p[z, r] 0, z]; 此多项式的根为: 尽管看起来这个多项式是特殊的,其实一般的三次多项式都是可以通过线性变换化为这个形式…...