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

自己实现 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 和 方法参数名获取参数

完成任务说明

img

- 后端 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 , 仍然可以看 到正确的返回

img

❤️‍🔥完成: 在方法参数 指定 @RequestParam 的参数封装到参数数组,进行反射调用

- 测试页面

img

- 后端 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=牛魔王

img

💞完成: 在方法参数 没有指定 @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

💖注意:

  1. 在默认情况下,返回的并不是 request, response ,name 而是 arg0, arg1,arg2
  2. 需要使用到 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 ,完成测试

img

img

后台:

img

如果请求参数和方法参数不一致:

img

😄总结

本文完成了任务阶段 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

&#x1f600;前言 自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-RequestParam &#x1f3e0;个人主页&#xff1a;尘觉主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是尘觉&#xff0c;希望我的文章可以帮助到大家&#xff0c…...

数据可视化:图表绘制详解

数据可视化是一种将抽象的数字和数据转化为直观图形的技术&#xff0c;使数据的模式、趋势和关系一目了然。本文将详细介绍如何绘制各种类型的图表&#xff0c;包括柱状图、折线图、饼图、散点图和热力图等。 第一部分&#xff1a;图表类型和选择 1. 柱状图 柱状图是用于比较类…...

【中危】Apache Ivy<2.5.2 存在XXE漏洞 (CVE-2022-46751)

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

C#使用自定义的比较器对版本号(编码)字符串进行排序

给定一些数据&#xff0c;如下所示: “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. **自动驾驶** 代码示例&#xff1a;使用Python实现基于机器学习的图片分类AI的未来前景结论 &#x1f389;欢迎来到AIGC人工智能专栏~探索AI在日常生活中的应用 ☆* o(≧▽≦…...

Windows10查看图片的分辨率

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

Spring事务和事务传播机制(2)

前言&#x1f36d; ❤️❤️❤️SSM专栏更新中&#xff0c;各位大佬觉得写得不错&#xff0c;支持一下&#xff0c;感谢了&#xff01;❤️❤️❤️ Spring Spring MVC MyBatis_冷兮雪的博客-CSDN博客 在Spring框架中&#xff0c;事务管理是一种用于维护数据库操作的一致性和…...

计算机视觉 -- 图像分割

文章目录 1. 图像分割2. FCN2.1 语义分割– FCN &#xff08;Fully Convolutional Networks&#xff09;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&#xff1a;Resnet1013…...

ubuntu18.04复现yolo v8之CUDA与pytorch版本问题以及多CUDA版本安装及切换

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

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 中&#xff0c;可以通过以下步骤新增只读远程普通用户&#xff1a; 1、使用 root 用户登录 MySQL 数据库。 mysql -u root -p 2、创建用户&#xff1a; CREATE USER username% IDENTIFIED WITH mysql_native_password BY password ; 其中&#xff0c;username…...

揭开路由协议隐藏的风险

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

图片因固定宽高被拉伸了?object-fit:一个神奇的属性

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

客户案例:中圣科技—CAC2.0防范盗号威胁,加固安全防线

客户背景 中圣科技&#xff08;江苏&#xff09;股份有限公司&#xff08;以下简称“中圣科技”&#xff09;&#xff0c;是一家以技术研发为驱动&#xff0c;以清洁能源核心成套装备和节能环保工程服务为支撑的科技创新型企业。其以南京为核心运营基地&#xff0c;与当地政府…...

pandas数据分析40——读取 excel 合并单元格的表头

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

Java后端开发面试题——微服务篇总结

Spring Cloud 5大组件有哪些&#xff1f; 随着SpringCloudAlibba在国内兴起 , 我们项目中使用了一些阿里巴巴的组件 注册中心/配置中心 Nacos 负载均衡 Ribbon 服务调用 Feign 服务保护 sentinel 服务网关 Gateway Ribbon负载均衡策略有哪些 ? RoundRobinRule&…...

第十一章MyBatis查询专题

返回单个Car 返回单个可以直接用Car接收返回参数 Car carCarMapper.getOne(100);返回多个Car 返回多个可以直接用List接收返回参数 List<Car> carCarMapper.getAll();用一个对象无法接受返回多个参数&#xff0c;用list可以接收返回一个参数 返回Map 如果没有合适的…...

测试驱动开发(TDD)

测试驱动开发&#xff08;TDD&#xff09; 本篇文章简单叙述一下什么是测试驱动开发&#xff0c;以及怎么进行测试驱动开发&#xff01; TDD &#xff08;Test Driven Development&#xff09;&#xff1a;&#xff08;源于极限编程&#xff08;XP&#xff09;&#xff09;在不…...

深度学习|CNN卷积神经网络

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

【洁洁送书第五期】为什么我们要了解可观测性工程

导读 可观测性已成为一个热门话题&#xff0c;并广受关注。随着它的普及&#xff0c;“可观测性”不幸被误作“监控”或“系统遥测”的同义词。可观测性是软件系统的一个特征。而且&#xff0c;只有当团队采用新的实践进行持续开发时&#xff0c;才能在生产软件系统中有效利用这…...

将vue项目通过electron打包成windows可执行程序

将vue项目打包成windows可执行程序 1、准备好dist将整个项目打包 npm run build2、安装electron依赖 npm install electron --save-dev npm install electron-packager --save-dev"electron": "^13.1.4", "electron-packager": "^15.2.0…...

【0基础入门Python Web笔记】三、python 之函数以及常用内置函数

三、python 之函数以及常用内置函数 函数函数定义函数调用函数参数返回值 常用内置函数input()函数range()函数其它 更多实战项目可进入下方官网 函数 函数是一种用于封装可重复使用代码块的工具&#xff0c;能够将一系列操作组织成一个逻辑单元。 函数定义 在Python中&…...

相交链表00

题目链接 相交链表 题目描述 注意点 保证 整个链式结构中不存在环函数返回结果后&#xff0c;链表必须 保持其原始结构如果 listA 和 listB 没有交点&#xff0c;intersectVal 为 0 解答思路 两个链表从头开始遍历&#xff0c;如果其是在同一个位置处相交&#xff0c;则在…...

怎样压缩mp4视频大小?

怎样压缩mp4视频大小&#xff1f;由于视频文件的体积通常比其他类型的文件更大&#xff0c;因此它们需要更多的存储空间来保存。但是&#xff0c;如果我们的设备、应用程序或平台不支持某些视频格式或分辨率&#xff0c;或者我们没有足够的存储空间来容纳这些大型视频文件&…...

ubuntu20.04 安装使用 Indemind 双目相机

1、先按照官方wiki搭建环境 Ubuntu 安装 — IMSEE SDK 1.4.2 文档&#xff08;ubuntu20使用官网会报错&#xff0c;可以参考我下面的步骤&#xff09; 1.1、获取代码 sudo apt-get install git git clone https://github.com/indemind/IMSEE-SDK.git 1.2、准备依赖 cd <…...

一文读懂设备管理系统:是什么、谁需要、怎样选

工业的迅猛发展让人类向前迈出了史无前例的步伐&#xff0c;工业4.0将我们又带入了一个信息化技术促进工业变革的新时代——智能化时代。一台台机器设备是工业发展史上必不可少的参与者&#xff0c;但企业对设备的管理存在种种痛点&#xff0c;比如生产设备多&#xff0c;但备件…...

删除链表的中间节点

题目&#xff1a; 示例&#xff1a; 思路&#xff1a; 这个题类似于寻找链表中间的数字&#xff0c;slow和fast都指向head&#xff0c;slow走一步&#xff0c;fast走两步&#xff0c;也许你会有疑问&#xff0c;节点数的奇偶不考虑吗&#xff1f;while执行条件写成fast&&…...

Q/GDW 1597-2015《国家电网公司应用软件系统通用安全要求》

电力安全测试报告 电力行业检测标准 随着信息技术的快速发展和广泛应用&#xff0c;应用软件系统已成为企业信息化建设中不可或缺的重要组成部分。然而&#xff0c;应用软件系统的安全问题也随之而来&#xff0c;给企业和用户带来了潜在的风险和威胁。为了提高应用软件系统的…...

【前端从0开始】CSS——12、光标属性

光标属性 cursor 属性规定要显示的光标的类型&#xff08;形状&#xff09;。 该属性定义了鼠标指针放在一个元素边界范围内时所用的光标形状&#xff08;不过 CSS2.1 没有定义由哪个边界确定这个范围&#xff09;。 属性名效果crosshair精确定位“十”字形pointer“小手”形…...

文件四剑客

目录 前言 一、正则表达式 二、grep 三、find 四、sed 五、awk 前言 文件四剑客是指在计算机领域中常用的四个命令行工具&#xff0c;包括awk、find、grep和sed。它们在处理文本文件和搜索文件时非常强大和实用。 1. awk是一种强大的文本处理工具&#xff0c;它允许用户根据指…...

高平企业网站/社交媒体营销

1.如果我要继承的基类是动态的&#xff08;有时候是 A&#xff0c;有时候是 B&#xff09;&#xff0c;我应该如何部署我的代码&#xff0c;以便基类可以随意改变。 BaseAlias BaseClass # 为基类取别名class Derived(BaseAlias):def meth(self):BaseAlias.meth(self) # 通…...

网站关键词优化是什么/快速排名工具免费

本篇文章给大家带来的内容是关于Vue.js路由器的使用方法总结(附代码)&#xff0c;有一定的参考价值&#xff0c;有需要的朋友可以参考一下&#xff0c;希望对你有所帮助。是用于 路由器的无刷新跳转改变 标签默认显示标签标签默认显示 Dom 为 主页通过 tag 属性可以改变 如&…...

企业网站的基本内容有哪些/搜索引擎推广试题

这两天在看Java的时候看到了方法中的值传递与引用传递&#xff0c;对于值传递我们都可以理解&#xff0c;非常的简单&#xff0c;但是对于引用传递&#xff0c;对于我们学过c的人来说&#xff0c;那可不就来劲了吗&#xff0c;直接写一个函数来用“&”操作符直接改变参数的…...

专业做幼儿园网站/小红书推广怎么收费

2016-10-15更新 添加了3.3–为微信电脑版增加桌面启动器(快捷方式) CSDNGitHubLinux和Mac下的微信电脑版electronic-wechat(非官方)AderXCoding/system/tools/electronic_wechat 本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可 1 electronic-…...

手机端网站开发源码/百度提升排名

我们知道&#xff0c;pyecharts是个非常好的python画图工具包&#xff0c;但是在notebook使用的时候&#xff0c;会遇到图形没法显示的情况&#xff0c;这时候需要做如下设置&#xff1a;#在程序中添加一下代码&#xff0c;解决pecharts在notebook的依赖from pyecharts.globals…...

建网站公司公司/网络搜索优化

ARM V7版架构&#xff1a; 从V7版本后开始变成了Cortex架构。 Cortex-A系列: 应用处理器&#xff0c;主要用于移动计算、智能手机、车载娱乐、自动驾驶、服务器、高端处理器等领域。时钟频率超过1GHZ,支持Linux、Android、Windows等完整操作系统需要的内存管理单元MMU。 Cor…...