自己实现 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连支持一下,创造不易您们的支持是我的动力🤞

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/102653.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Feign:使用接口方式调用服务

创建一个新的消费者模块并导入依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://ma…

Linux学习之firewallD

systemctl status firewalld.service查看一下firewalld服务的状态&#xff0c;发现状态是inactive (dead)。 systemctl start firewalld.service启动firewalld&#xff0c;systemctl status firewalld.service查看一下firewalld服务的状态&#xff0c;发现状态是active (runni…

无涯教程-PHP - preg_grep()函数

preg_grep() - 语法 array preg_grep ( string $pattern, array $input [, int $flags] ); 返回由与给定模式匹配的输入数组元素组成的数组。 如果将flag设置为PREG_GREP_INVERT&#xff0c;则此函数返回输入数组中与给定模式不匹配的元素。 preg_grep() - 返回值 返回使用…

pdf转word最简单方法~

pdf转word最简单方法&#xff01;pdf转word最简单方法我们都知道&#xff0c;PDF文件是一种只读文件格式&#xff0c;无法按照需求对PDF文件进行更改与编辑&#xff0c;从而影响到了PDF文件的使用。所以&#xff0c;我们需要将PDF文件转换为word文档&#xff0c;以此来保证文件…

LLM 生成式配置的推理参数温度 top k tokens等 Generative configuration inference parameters

在这个视频中&#xff0c;你将了解一些方法和相关的配置参数&#xff0c;这些参数可以用来影响模型在下一个词生成时的最终决策方式。如果你在Hugging Face网站或AWS的游乐场中使用过LLMs&#xff0c;你可能已经看到了这些控制选项&#xff0c;用来调整LLM的行为。每个模型都暴…

【数据库】表操作 习题总结

目录 关系建表 数据库sql的执行顺序 内外连接的写法 1.设计一张商品表 2.设计一张老师表 3.设计一张图书表 4.查询练习 5.查询练习 6.设计一个考勤系统 7.设计一个学校宿舍管理系统 8.设计一个车辆违章系统 9.设计一个学校食堂管理系统 10.有一张员工表emp&#xf…

AWS SDK 3.x for .NET Framework 4.0 可行性测试

前言 为了应对日益增长的网络安全挑战, 越来越多的互联网厂商已经陆续开始或者已经彻底停止了对 SSL 3 / TLS 1.0 / TLS1.1 等上古加密算法的支持. 而对于一些同样拥有悠久历史的和 AWS 服务相关联的应用程序, 是否可以通过仅更新 SDK 版本的方式来适应新的环境. 本文将以 Win…

前端处理图片文件的方法

在项目开发过程中&#xff0c;有一个需求&#xff0c;需要前端对上传的图片进行处理&#xff0c;以字符串的形式传给后端&#xff0c;实现效果如下&#xff1a; 1.上传图片的组件 在该项目中&#xff0c;使用了element plus组件库 <el-uploadv-model:file-list"fileL…

2023年目标检测研究进展

综述 首先关于写这个笔记&#xff0c;我个人思考了很久关于以下几点。1&#xff1a;19年开始从做OCR用到图像和文本这种多模态联合处理的后&#xff0c;也就有意识的开始关注自然语言处理&#xff0c;这样的结果导致可能停留在前期图像上的学习和实践&#xff0c;停滞的研究如…

17万字数字化医院信息化建设大数据平台建设方案WORD

导读&#xff1a;原文《17万字数字化医院信息化建设大数据平台建设方案WORD》&#xff08;获取来源见文尾&#xff09;&#xff0c;本文精选其中精华及架构部分&#xff0c;逻辑清晰、内容完整&#xff0c;为快速形成售前方案提供参考。 目录 第1章 医院信息化概述 1.1 国内…

【开发】视频云存储EasyCVR视频汇聚平台AI智能算法定制

安防视频集中存储EasyCVR视频汇聚平台&#xff0c;可支持海量视频的轻量化接入与汇聚管理。平台能提供视频存储磁盘阵列、视频监控直播、视频轮播、视频录像、云存储、回放与检索、智能告警、服务器集群、语音对讲、云台控制、电子地图、平台级联、H.265自动转码等功能。为了便…

2023国赛数学建模思路 - 案例:最短时间生产计划安排

文章目录 0 赛题思路1 模型描述2 实例2.1 问题描述2.2 数学模型2.2.1 模型流程2.2.2 符号约定2.2.3 求解模型 2.3 相关代码2.4 模型求解结果 建模资料 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 最短时…

Log4j反序列化命令执行漏洞(CVE-2017-5645)Apache Log4j2 lookup JNDI 注入漏洞(CVE-2021-44228)

一.Log4j反序列化命令执行漏洞(CVE-2017-5645&#xff09; Apache Log4j是一个用于Java的日志记录库&#xff0c;其支持启动远程日志服务器。Apache Log4j 2.8.2之前的2.x版本中存在安全漏洞。攻击者可利用该漏洞执行任意代码 环境&#xff1a;vulhub 工具下载地址&#xff1…

Oracle Database12c数据库官网下载和安装教程

文章目录 下载安装Oracle自带的客户端工具使用 下载 进入oracle官网 点击下载连接之后右上角会有一个下载 我们只需要数据库本体就够了 运行这个下载器 等待下好之后即可 出现 Complete 之后代表下载成功&#xff0c;然后我们解压即可 安装 双击 双击setup.exe 根据…

opencv进阶09-视频处理cv2.VideoCapture示例(打开本机电脑摄像头)

视频信号&#xff08;以下简称为视频&#xff09;是非常重要的视觉信息来源&#xff0c;它是视觉处理过程中经常要处理的一类信号。实际上&#xff0c;视频是由一系列图像构成的&#xff0c;这一系列图像被称为帧&#xff0c;帧是以固定的时间间隔从视频中获取的。获取&#xf…

在C中使用Socket实现多线程异步TCP消息发送

目录 基础知识开始实现主要函数说明结束语 在本篇文章中&#xff0c;我们会探讨如何在C语言中使用socket来实现多线程&#xff0c;异步发送TCP消息的系统。虽然C标准库并没有原生支持异步和多线程编程&#xff0c;但是我们可以结合使用POSIX线程&#xff08;pthread&#xff09…

【⑬MySQL | 数据类型(一)】简介 | 整数 | 浮点 | 定点类型

前言 ✨欢迎来到小K的MySQL专栏&#xff0c;本节将为大家带来MySQL数据类型简介 | 整数 | 浮点 | 定点类型的分享✨ 目录 前言0.数据类型简介1 整数类型2 浮点类型3 定点类型4 日期/时间类型总结 0.数据类型简介 数据类型&#xff08;data_type&#xff09;是指系统中所允许的…

学会Mybatis框架:让你的代码更具灵活性、可维护性、安全性和高效性【二.动态SQL】

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Mybatis的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.Mybatis动态SQL如何应用 1.需求 2.…

Win解答 | 解决键盘中 字母+空格 导致的输入法弹窗导致的一系列问题

近三个月来&#xff0c;一直都有一个键盘组合键的问题影响我的电脑使用&#xff0c;不管是打字还是打游戏&#xff0c;都会出现按键盘的 字母空格 弹出一个特殊符号的候选框&#xff0c;如下图所示 图片中为 S空格 所出现的弹窗 一个看似方便&#xff0c;实则难受的功能 其实打…

针对论坛系统进行功能测试和性能测试

项目链接:飞鸽论坛 目录 一. 项目背景 二. 项目功能 三. 功能测试 注册: 登录: 更改用户信息: 发布帖子: 更新帖子信息: 点赞: 评论: 发送私信: 测试报告 四. 性能测试 Virtual User Generator Controller Analysis 测试报告: 一. 项目背景 该论坛系统采用前…