SpringMVC[从零开始]

SpringMVC

  1. SpringMVC简介

1.1什么是MVC

MVC是一种软件架构的思想,将软件按照模型、视图、控制器来划分

M:Model,模型层,指工程中的JavaBean,作用是处理数据

JavaBean分为两类:

一类称为实体类Bean:专门存储业务数据的,比如:Emp、Dept、User

一类称为业务处理Bean:指Service或Dao对象,专门用于处理业务逻辑和数据访问

V:View,视图层,指工程中的Html或jsp页面,作用是与用户进行交互,展示数据

C:Controller,控制层,指工程中的servlet,作用是接收请求和响应浏览器

MVC的工作流程:用户通过V(视图层)发送请求到服务器,在服务器中由C(控制器)接收,C(控制器)调用相应的M(模型层)层处理请求,处理完毕将结果返回到C,C再根据请求处理的结果,找到相应的V,渲染数据后最终响应给浏览器。

1.2、什么是SpringMVC

SpringMVC是Spring家族的一个后续产品,是Spring的一个子项目

SpringMVC是为表述层开发提供的一整套完备的解决方案。

注:三层架构分为表述层(表示层)、业务逻辑层、数据访问层。表述层表示前台页面+后台servlet

1.3SpringMVC的特点

  1. Spring 家族原生产品,与 IOC 容器等基础设施无缝对接
  2. 基于原生的Servlet,通过了功能强大的前端控制器DispatcherServlet,对请求和响应进行统一处理
  3. 表述层各细分领域需要解决的问题全方位覆盖,提供全面解决方案
  4. 代码清新简洁,大幅度提升开发效率
  5. 内部组件化程度高,可插拔式组件即插即用,想要什么功能配置相应组件即可
  6. 性能卓著,尤其适合现代大型、超大型互联网项目要求
  1. 入门案例

2.1、开发环境

IDE:idea 2022.2.3

构建工具:maven 3.6.0

服务器:tomcat8.5

Spring版本:5.3.1

2.2、创建maven工程

导pom依赖时,由于Maven具有传递性的,所以不需要将需要的所有包全部配置依赖,只需要配置最顶端的依赖即可,其它靠maven的传递性导入

2.3配置web.xml

注:

<url-pattern>标签中使用/和/*的区别:

/所匹配的请求可以是/login或.html或.js或.css方式的请求路径,但是/不能匹配.jsp请求路径的请求因此就可以避免在访问jsp页面时,该请求被DispatcherServlet处理,从而找不到相应的页面/*则能够匹配所有请求,例如在使用过滤器时,若需要对所有请求进行过滤,就需要使用/*的写法

3、@RequestMapping注解

3.1、功能

就是将用户的请求和处理请求的控制器的方法关联起来,建立映射关系。

SpringMVC接收到指定的请求以后,就会找到在映射关系中对应的控制器的方法来处理这个请求

3.2、注解的位置

@RequestMapping注解标识到类或方法上

3.3、value属性

通过value的属性请求的请求地址匹配请求映射

value属性是必须设置的

value属性是一个字符串类型的数组,表示该请求映射能够匹配多个请求地址所对应的请求

3.4、method属性

method属性通过请求的请求方式(getpost)匹配请求映射

它是一个RequestMethod[]类型的数组,表示该请求映射能够匹配多种请求方式的请求,当前浏览器所发送的请求方式匹配method属性中的任何一种请求方式,则当前请求就会被注解所标识的方法进行处理

如果当前请求的请求地址满足映射的value属性,但是请求方式不满足method属性,则浏览器报405错误:Request method 'xxx' not supported

在RequestMapping注解的基础上,结合请求方式派生出了一些注解

@PostMapping 就等同与@RequestMapping(value = "/hello",method =RequestMethod.POST)
@GetMapping
@PutMapping
@DeleteMapping

目前浏览器只支持getpost请求,如果在form表单提交时,为method设置了其他请求方式的字符串,比如putdelete,则按照默认的请求方式get处理

如果要发生putdelete的请求,则需要通过spring提供的过滤器HiddenHttpMethodFilter,在后续RESTful部分详细讲解

3.5、params属性 (了解)

通过请求的请求参数匹配请求,即浏览器发送的请求的请求参数必须满足params的属性设置

params可以使用四种表达式

“params”:表示当前所匹配的请求的请求参数中必须携带params参数

 Parameter conditions "xxx" not met for actual request parameters:

“!params”表示当前所匹配的请求的请求参数中一定不能携带params参数

Parameter conditions "!xxx" not met for actual request parameters:

“params=value”:表示当前所匹配的请求的请求参数中必须携带params参数并且值必须为value

“params!=value”:表示当前所匹配的请求的请求参数中可以不携带params参数,如果携带值一定不能是value

3.6headers

四种

"header":要求请求映射所匹配的请求必须携带header请求头信息

"!header":要求请求映射所匹配的请求必须不能携带header请求头信息

"header=value":要求请求映射所匹配的请求必须携带header请求头信息且header=value

"header!=value":要求请求映射所匹配的请求必须携带header请求头信息且header!=value

若当前请求满足@RequestMapping注解的value和method属性,但是不满足headers属性,此时页面显示404错误,即资源未找到

3.7、支持ant风格的路径

?:表示任意的单个字符

*:表示任意的0个或者多个字符  不包括/

**:表示任意层数的任意目录,注意:使用**时,只能使用/**/xxx的方式

3.8SpringMVC支持路径中的占位符

原始的方式:/deleteUserById?id=1

rest方式:/deleteUserById/1

占位符常用于restful风格中,当请求路径中将某些数据通过路径的方式传输到服务器中,就可以在相应的@RequestMapping注解的value属性中通过占位符{xxx}表示传输的数据,再通过@PathVariable注解,将占位符表示的数据赋值给控制器方法的形参

只需要在控制器方法的形参位置,设置一个形参,形参的名字和请求参数的名字一致即可,但是必须加@PathVariable注解

4、springmvc获取请求参数

4.1、通过的ServletAPI获取

4.2、通过控制器方法的形参获取请求参数

<form th:action="@{/param}" method="post">

    用户名:<input type="text" name="username"><br>

    密码:<input type="password" name="password"><br>

    <input type="submit" value="登录">

</form>

@RequestMapping("/param")

  public String getParam(String username,String password){

    System.out.println("username:"+username+",password:"+password);

    return "success";

}

* 只需要在控制器方法的形参位置,设置一个形参,形参的名字和请求参数的名字一致即可

4.3、@RequestParam

@RequestParam是将请求参数和控制器方法的形参创建映射关系

http://localhost:8080/springmvc_rest/param?password=123

@RequestMapping("/param")

  public String getParam(@RequestParam(value = "username",required = true,

        defaultValue = "hello")String username, String password){

    /**

     * @RequestParam注解的属性:

     * value:设置和形参绑定的请求参数的名字

     * required:设置是否必须传输value所对应的请求参数

     *          默认值为true,表示value所对应的请求参数必须传输,否则页面报错

     *          Required String parameter 'xxx' is not present

     *          若设置为false,则表示value所对应的请求参数不是必须传输,

     *          如果此时获取该参数的值,则为null

     *defaultValue:设置当没有传输value所对应的请求参数时,

     *              为形参设置的默认值,此时和required属性值无关

     */

    System.out.println("username:" + username + ",password:" + password);

    return "success";

}

4.4、@RequestHeaher

@RequestParam是将请求头信息和控制器方法的形参创建映射关系

@RequestMapping("/param")

  public String getParam(@RequestParam(value = "username",required = true,

        defaultValue = "hello")String username, String password,

                       @RequestHeader("referer") String referer){

4.5、@CookieValue

@CookieValue是将cookie数据和控制器方法的形参创建映射关系

@CookieValue("JSESSIONID")String jsessionId

如果没有cookie数据JSESSIONID会报错:

Missing cookie 'JSESSIONID' for method parameter of type String

通过该超链接访问/param/servletAPI地址

<a th:href="@{/param/servletAPI}">测试@RequestMapping注解的value属性的占位符</a>

跳转到该方法,通过 HttpSession session = request.getSession();生成JSESSIONID

@RequestMapping("/param/servletAPI")

  public String getParamByServletAPI(HttpServletRequest request){

    HttpSession session = request.getSession();

      String username = request.getParameter("username");

    String password = request.getParameter("password");

    System.out.println("username:"+username+",password:"+password);

    return "success";

}

再通过

<form th:action="@{/param}" method="post">

    用户名:<input type="text" name="username"><br>

    密码:<input type="password" name="password"><br>

    <input type="submit" value="登录">

</form>

访问控制器方法

@RequestMapping("/param")

  public String getParam(

        @RequestParam(value = "username",required = true,

        defaultValue = "hello")String username, String password,

                       @RequestHeader("referer") String referer,

                       @CookieValue("JSESSIONID")String jsessionId){

      System.out.println("username:" + username + ",password:" + password);

    System.out.println("referer:"+referer);

    System.out.println("jsessionId:"+jsessionId);

    return "success";

}

输出:

4.6、通过pojo获取请求参数

可以在控制器方法的形参位置设置一个实体类类型的形参,此时如果浏览器传输的请求参数的参数名和实体类中的属性名一致,那么请求参数就会为此属性赋值

pojo

package com.qcby.pojo;

/**
 * @Author che
 * @Date 2024/6/9 10:46
 * Description:
TODO
 
* Version 1.0
 */

public class User {
   
private  Integer id;
   
private String username;
   
private String password;

   
public User() {
    }

   
public User(Integer id, String username, String password) {
       
this.id = id;
       
this.username = username;
       
this.password = password;
    }

   
public Integer getId() {
       
return id;
    }

   
public void setId(Integer id) {
       
this.id = id;
    }

    
public String getUsername() {
       
return username;
    }

   
public void setUsername(String username) {
       
this.username = username;
    }

   
public String getPassword() {
       
return password;
    }

   
public void setPassword(String password) {
       
this.password = password;
    }

   
@Override
   
public String toString() {
       
return "User{" +
               
"id=" + id +
               
", username='" + username + '\'' +
               
", password='" + password + '\'' +
                
'}';
    }
}

Index.html

<form th:action="@{/param/pojo}" method="post">

    用户名:<input type="text" name="username"><br>

    密码:<input type="password" name="password"><br>

    <input type="submit" value="登录">

</form>

控制器

@RequestMapping("/param/pojo")

  public String getParamByPojo(User user){

    System.out.println(user);

    return "success";

}

4.7解决获取请求参数的乱码问题

Springmvc提供的编码过滤器CharacterEncodingFilter

encoding->utf-8
forceEncoding->true

SpringMVC中处理编码的过滤器一定要配置到其他过滤器之前,否则无效

<!--配置springmvc的编码过滤器-->

  <filter>

    <filter-name>CharacterEncodingFilter</filter-name>

    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>

  

    <init-param>

        <param-name>encoding</param-name>

        <param-value>utf-8</param-value>

    </init-param>

  

    <init-param>

        <param-name>forceEncoding</param-name>

        <param-value>true</param-value>

    </init-param>

</filter>

  

<filter-mapping>

    <filter-name>CharacterEncodingFilter</filter-name>

    <url-pattern>/*</url-pattern>

</filter-mapping>

5、域对象共享数据

5.1、使用ServletAPI向request域对象共享数据

<a th:href="@{testServletAPI}">测试通过ServletAPI向请求域共享数据</a>

@RequestMapping("/testServletAPI")

  public String testServletAPI(HttpServletRequest request){

    request.setAttribute("testScope","hello,servletAPI");

    return "success";

}

<p th:text="${testScope}"></p>

5.2、使用ModelAndView向request域对象共享数据

<a th:href="@{/testmav}">测试通过ModelAndView向请求域共享数据</a>

@RequestMapping("/testmav")

  public ModelAndView testmav(){

    /**

     * ModelAndView包含ModelView的功能

     * Model:向请求域中共享数据

     * View:设置逻辑视图实现页面跳转

     */

    ModelAndView modelAndView = new ModelAndView();

    //向请求域中共享数据

    modelAndView.addObject("testReqeustScope","hello,ModelAndView");

    //设置逻辑视图

    modelAndView.setViewName("success");

    //使用View设置逻辑视图,但是控制器方法一定要将ModelAndView的对象作为方法的返回值

    return modelAndView;

}

<p th:text="${testReqeustScope}"></p>

5.3、使用Model向request域对象共享数据

<a th:href="@{/testmodel}">测试通过Model向请求域共享数据</a><br>

@RequestMapping("/testmodel")

  public String testmodel(Model model){

    //org.springframework.validation.support.BindingAwareModelMap

    System.out.println(model.getClass().getName());

    model.addAttribute("testReqeustScope","hello,Model");

    return "success";

}

5.4、使用map向request域对象共享数据

@RequestMapping("/testMap")

  public String testMap(Map<String,Object> map){

    //org.springframework.validation.support.BindingAwareModelMap

    System.out.println(map.getClass().getName());

    map.put("testReqeustScope","hello,map");

    return "success";

}

5.5、使用ModelMap向request域对象共享数据

@RequestMapping("/testmodelmap")

  public String testModelMap(ModelMap modelMap){

    //org.springframework.validation.support.BindingAwareModelMap

    System.out.println(modelMap.getClass().getName());

    modelMap.addAttribute("testReqeustScope","hello,ModelMap");

    return "success";

}

5.6 Model、ModelMap、Map的关系

Model、ModelMap、Map类型的参数本质上其实都是BindingAwareModelMap类型的

org.springframework.validation.support.BindingAwareModelMap

所以在底层中,这些类型的形参最终都是通过BindingAwareModelMap创建的
5.7、向session域共享数据

<a th:href="@{/testsession}">测试通过session向回话域共享数据</a><br>

@RequestMapping("/testsession")

  public String testSession(HttpSession session){

    session.setAttribute("testSessionScope","hello,session");

    return "success";

}

<p th:text="${session.testSessionScope}"></p>

5.8、向application域共享数据

@RequestMapping("/testapplication")

  public String testApplication(HttpSession session){

    ServletContext servletContext = session.getServletContext();

    servletContext.setAttribute("testApplicationScope","hello,Application");

    return "success";

}

<!--通过${application.属性名}获取的是application域对象数据-->

  <p th:text="${application.testApplicationScope}"></p>

6、springmvc的视图

SpringMVC中的视图是View接口,视图的作用渲染数据,将模型Model中的数据展示给用户,SpringMVC视图的种类很多,默认有转发视图重定向视图

当工程引入jstl的依赖,转发视图会自动转换为JstlView ,若使用的视图技术为Thymeleaf,在SpringMVC的配置文件中配置了Thymeleaf的视图解析器,由此视图解析器解析之后所得到的是ThymeleafView

6.1、ThymeleafView

当控制器方法中所设置的视图名称没有任何前缀时,此时的视图名称会被SpringMVC配置文件中所配置的视图解析器解析,视图名称拼接视图前缀和视图后缀所得到的最终路径,会通过转发的方式实现跳转

6.2、转发视图

SpringMVC中默认的转发视图是InternalResourceView

SpringMVC中创建转发视图的情况:

当控制器方法中所设置的视图名称以"forward:"为前缀时,创建InternalResourceView视图,此时的视图名称不会被SpringMVC配置文件中所配置的视图解析器解析,而是会将前缀"forward:"去掉,剩余部分作为最终路径通过转发的方式实现跳转

例如"forward:/","forward:/employee"

<a th:href="@{/test/view/forward}">测试SpringMVC的视图InternalResourceView </a>

@RequestMapping("/testmodel")

  public String testmodel(Model model){

    //org.springframework.validation.support.BindingAwareModelMap

    System.out.println(model.getClass().getName());

    model.addAttribute("testReqeustScope","hello,Model");

    return "success";

}

@RequestMapping("/test/view/forward")

  public String testInternalResourceView(){

    return "forward:/testmodel";

}

<p th:text="${testReqeustScope}"></p>

6.3、重定向视图

SpringMVC中默认的重定向视图是RedirectView

当控制器方法中所设置的视图名称以"redirect:"为前缀时,创建RedirectView视图,此时的视图名称不会被SpringMVC配置文件中所配置的视图解析器解析,而是会将前缀"redirect:"去掉,剩余部分作为最终路径通过重定向的方式实现跳转

例如"redirect:/","redirect:/employee"

注:

重定向视图在解析时,会先将redirect:前缀去掉,然后会判断剩余部分是否以/开头,若是则会自动拼接上下文路径

6.4、视图控制器view-contronller

当控制器方法中,仅仅用来实现页面跳转,即只需要设置视图名称时,可以将处理器方法使用view-controller标签进行表示

<mvc:view-controller path="/" view-name="index"></mvc:view-controller>

注:

当springmvc中设置任何一个view-controller时,其它控制器中的请求映射将全部失效,

此时需要再SpringMVC的核心配置文件中开启mvc的注解驱动

<!--开启注解驱动-->

  <mvc:annotation-driven></mvc:annotation-driven>

  1. RESTFul

7.1、简介

REST:Representational State Transfer,表现层资源状态转移。

7.2、实现

具体说,就是 HTTP 协议里面,四个表示操作方式的动词:GET、POST、PUT、DELETE。

它们分别对应四种基本操作:GET 用来获取资源,POST 用来新建资源,PUT 用来更新资源,DELETE用来删除资源。

REST 风格提倡 URL 地址使用统一的风格设计,从前到后各个单词使用斜杠分开,不使用问号键值对方式携带请求参数,而是将要发送给服务器的数据作为 URL 地址的一部分,以保证整体风格的一致性。

操作

传统方式

REST风格

查询操作

getUserById?id=1

user/1-->get请求方式

保存操作

saveUser

user-->post请求方式

删除操作

deleteUser?id=1

user/1-->delete请求方式

更新操作

updateUser

user-->put请求方式

7.3、HiddenHttpMethodFilter

由于浏览器只支持发送get和post方式的请求,那么该如何发送put和delete请求呢?

SpringMVC 提供了HiddenHttpMethodFilter,将 POST 请求转换为 DELETE 或 PUT 请求HiddenHttpMethodFilter 处理put和delete请求的条件:

a>当前请求的请求方式必须为post

b>当前请求必须传输请求参数_method

满足以上条件,HiddenHttpMethodFilter 过滤器就会将当前请求的请求方式转换为请求参数_method的值,因此请求参数_method的值才是最终的请求方式

在web.xml中注册HiddenHttpMethodFilter

<!--处理请求方式的过滤器-->

  <filter>

    <filter-name>HiddenHttpMethodFilter</filter-name>

    <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>

</filter>

<filter-mapping>

    <filter-name>HiddenHttpMethodFilter</filter-name>

    <url-pattern>/*</url-pattern>

</filter-mapping>

8、RESTFul案例

8.1、准备工作

对员工表(模拟表数据)的员工信息进行增删改查

搭建环境

准备实体类

package com.qcby.pojo;

  

  /**

 * @Author che

 * @Date 2024/6/9 19:16

 * Description:TODO

 * Version 1.0

 */

  public class Employee {

    private  Integer id;//主键

    private String lastName;//姓名

    private String email;//邮箱

    private Integer sex;//性别,1male,0female

  

    public Employee() {

    }

  

    public Employee(Integer id, String lastName, String email, Integer sex) {

        this.id = id;

        this.lastName = lastName;

        this.email = email;

        this.sex = sex;

    }

  

    public Integer getId() {

        return id;

    }

  

    public void setId(Integer id) {

        this.id = id;

    }

  

    public String getLastName() {

        return lastName;

    }

  

    public void setLastName(String lastName) {

        this.lastName = lastName;

    }

  

    public String getEmail() {

        return email;

    }

  

    public void setEmail(String email) {

        this.email = email;

    }

  

    public Integer getSex() {

        return sex;

    }

  

    public void setSex(Integer sex) {

        this.sex = sex;

    }

  

    @Override

    public String toString() {

        return "Employee{" +

                "id=" + id +

                ", lastName='" + lastName + '\'' +

                ", email='" + email + '\'' +

                ", sex=" + sex +

                '}';

    }

}

准备dao模拟数据

package com.qcby.dao;

  

  import com.qcby.pojo.Employee;

  import org.springframework.stereotype.Repository;

  

  import java.util.Collection;

  import java.util.HashMap;

  import java.util.Map;

  

  /**

 * @Author che

 * @Date 2024/6/9 19:19

 * Description:TODO

 * Version 1.0

 */

  @Repository

  public class EmployeeDao {

    private static Map<Integer, Employee> employees = null;

    static {

        employees = new HashMap<Integer,Employee>();

        employees.put(1001,new Employee(1001,"E-AA","aa@qq.com",1));

        employees.put(1002,new Employee(1002,"E-AA","aa@qq.com",1));

        employees.put(1003,new Employee(1003,"E-AA","aa@qq.com",0));

        employees.put(1004,new Employee(1004,"E-AA","aa@qq.com",0));

        employees.put(1005,new Employee(1005,"E-AA","aa@qq.com",1));

    }

  

    private  static  Integer initId = 1006;

  

    /**

     * 模拟添加操作

     * @param employee

     */

    public void save(Employee employee){

        if(employee.getId() == null){

            employee.setId(initId++);

        }

        employees.put(employee.getId(),employee);

    }

  

    /**

     * 删除

     * @param id

     */

    public void delete(Integer id){

        employees.remove(id);

    }

  

    /**

     * 修改

     * @param id

     * @return

     */

    public Employee get(Integer id){

        return employees.get(id);

    }

  

    /**

     * 查找

     * @return

     */

    public Collection<Employee> getAll(){

        return employees.values();

    }

  

  

}

8.2功能清单

功能

URL 地址

请求方式

访问首页√

/

GET

查询全部数据√

/employee

GET

删除√

/employee/2

DELETE

跳转到添加数据页面√

/toAdd

GET

执行保存√

/employee

POST

跳转到更新数据页面√

/employee/2

GET

执行更新√

/employee

PUT

index.html

8.3、访问首页

SpringMVC.xml

<mvc:view-controller path="/" view-name="index"></mvc:view-controller>

创建index页面

<a th:href="@{/employee}">查询所有的员工信息</a><br>

控制器

@Autowired

  private EmployeeDao employeeDao;

  

  //@RequestMapping(value = "/employee",method = RequestMethod.GET)

  @GetMapping("/employee")

  public String getAllEmployee(Model model) {

    //获取所有的员工信息

    Collection<Employee> allEmployee = employeeDao.getAll();

    //将所有的员工信息在请求域中共享

    model.addAttribute("allEmployee",allEmployee);

    //返回到员工列表页面

    return "employee_list";

}

//@RequestMapping(value = "/employee/{id}",method = RequestMethod.DELETE)

  @DeleteMapping("/employee/{id}")

  public String deleteEmploee(@PathVariable Integer id) {

    //删除员工信息

    employeeDao.delete(id);

    return "redirect:/employee";

}

⑴创建处理删除delete请求方式的表单

<!-- 作用:通过超链接控制表单的提交,将post请求转换为delete请求 -->

  <form  method="post">

    <!-- HiddenHttpMethodFilter要求:必须传输_method请求参数,并且值为最终的请求方式 -->

    <input type="hidden" name="_method" value="delete">

</form>

⑵删除超链接绑定点击事件

<a @click="deleteEmployee()" th:href="@{'/employee/'+${employee.id}}">delete</a>

<script type="text/javascript" th:src="@{/static/js/vue.js}"></script>

<script type="text/javascript">

    var vue = new Vue({

        el:"#app",

        methods:{

            deleteEmployee(){

                //获取form表单

                var form = document.getElementsByTagName("form")[0];

                //将超链接的href属性赋值给form表单的action属性

                //event.target表示当前触发事件的标签

                form.action =event.target.href;

                //表单提交

                form.submit();

                //阻止超链接的默认行为

                event.preventDefault();

            }

        }

    });

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

对猫毛过敏?怎么有效的缓解过敏症状,宠物空气净化器有用吗?

猫过敏是一种常见的过敏反应&#xff0c;由猫的皮屑、唾液或尿液中的蛋白质引起。这些蛋白质被称为过敏原&#xff0c;它们可以通过空气传播&#xff0c;被人体吸入后&#xff0c;会触发免疫系统的过度反应。猫过敏是宠物过敏中最常见的类型之一&#xff0c;对许多人来说&#…

【Java】static 修饰变量

static 一种java内置关键字&#xff0c;静态关键字&#xff0c;可以修饰成员变量、成员方法。 static 成员变量 1.static 成员变量2.类变量图解3.类变量的访问4.类变量的内存原理5.类变量的应用 1.static 成员变量 成员变量按照有无static修饰&#xff0c;可以分为 类变量…

Python学习打卡:day02

day2 笔记来源于&#xff1a;黑马程序员python教程&#xff0c;8天python从入门到精通&#xff0c;学python看这套就够了 8、字符串的三种定义方式 字符串在Python中有多种定义形式 单引号定义法&#xff1a; name 黑马程序员双引号定义法&#xff1a; name "黑马程序…

如何为色盲适配图形用户界面

首发日期 2024-05-25, 以下为原文内容: 答案很简单: 把彩色去掉, 测试. 色盲, 正式名称 色觉异常. 众所周知, 色盲分不清颜色. 如果用户界面设计的不合理, 比如不同项目只使用颜色区分, 而没有形状区分, 那么色盲使用起来就会非常难受, 甚至无法使用. 色盲中最严重的情况称为…

2024PTA算法竞赛考试编程题代码

目录 前言 题目和代码 L1-006 连续因子 L1-009 N个数求和 L2-004 这是二叉搜索树吗&#xff1f; L2-006 树的遍历 L2-007 家庭房产 L4-118 均是素数 L4-203 三足鼎立 L2-002 链表去重 L2-003 月饼 L2-026 小字辈 L4-201 出栈序列的合法性 L4-205 浪漫侧影 前言 所…

【数据结构】AVL树(平衡二叉树)

目录 一、AVL树的概念二、AVL树的节点三、AVL树的插入四、AVL树的旋转1.插入在较高左子树的左侧&#xff0c;使用右单旋2.插入在较高右子树的右侧&#xff0c;使用左单旋3.插入较高左子树的右侧&#xff0c;先左单旋再右单旋4.插入较高右子树的左侧&#xff0c;先右单旋再左单旋…

unity基础(五)地形详解

目录 一 创建地形 二 调整地形大小 三 创建相邻地形 四 创建山峰 五 创建树木 七 添加风 八 添加水 简介: Unity 中的基础地形是构建虚拟场景的重要元素之一。 它提供了一种直观且灵活的方式来创建各种地形地貌&#xff0c;如山脉、平原、山谷等。 通过 Unity 的地形…

C51学习归纳9 --- I2C通讯学习(重点)

首先&#xff0c;我自己学习过以后的直观感觉&#xff0c;通信协议是单片机的灵魂之一&#xff0c;只有规定好了通信协议我们才能够正确的接收到信息&#xff0c;才能实现更加深入的研究。所以这一部分是需要好好学习的。 本节借助一个可存储的芯片AT24C02&#xff0c;进行在I2…

开源低代码平台技术为数字化转型赋能!

实现数字化转型升级是很多企业未来的发展趋势&#xff0c;也是企业获得更多发展商机的途径。如何进行数字化转型&#xff1f;如何实现流程化办公&#xff1f;这些都是摆在客户面前的实际问题&#xff0c;借助于开源低代码平台技术的优势特点&#xff0c;可以轻松助力企业降低开…

【设计模式】创建型设计模式之 建造者模式

文章目录 一、介绍定义UML 类图 二、用法1 简化复杂对象具体构建过程省略抽象的 Builder 类省略 Director 类 三、用法2 控制对象构造方法、限制参数关系Guava 中使用建造者模式构建 cache 来进行参数校验 一、介绍 定义 建造者模式&#xff0c;将一个复杂的对象的构建过程与…

互联网应用主流框架整合之SpringMVC初始化及各组件工作原理

Spring MVC的初始化和流程 MVC理念的发展 SpringMVC是Spring提供给Web应用领域的框架设计&#xff0c;MVC分别是Model-View-Controller的缩写&#xff0c;它是一个设计理念&#xff0c;不仅仅存在于Java中&#xff0c;各类语言及开发均可用&#xff0c;其运转流程和各组件的应…

探索OrangePi AIpro:单板计算机的深度体验之旅

准备阶段&#xff1a;环境与资料 在开始我们的探索之旅前&#xff0c;确保您已准备好以下装备&#xff1a; OrangePi AIpro&#xff1a;我们的主角&#xff0c;一台功能强大的单板计算机。Windows 10笔记本电脑&#xff1a;作为我们的辅助工具&#xff0c;用于管理和测试。路…

FastAPI:在大模型中使用fastapi对外提供接口

通过本文你可以了解到&#xff1a; 如何安装fastapi&#xff0c;快速接入如何让大模型对外提供API接口 往期文章回顾&#xff1a; 1.大模型学习资料整理&#xff1a;大模型学习资料整理&#xff1a;如何从0到1学习大模型&#xff0c;搭建个人或企业RAG系统&#xff0c;如何评估…

python ---使用python操作mysql ---> pymysql

本章内容: 1:能够完成从MySQL中读取出数据; [重点] 查询: execute()、fetchall() 2:能够将数据写入MySQL数据库。 [重点] 插入数据: execute() sql insert into xxx [掌握]pymysql模块的安装 目标&#xff1a;了解如何安装pymysql模块&#xff1f; 当要使用Python和M…

操作系统复习-存储管理之虚拟内存

虚拟内存概述 有些进程实际需要的内存很大&#xff0c;超过物理内存的容量。多道程序设计&#xff0c;使得每个进程可用物理内存更加稀缺。不可能无限增加物理内存&#xff0c;物理内存总有不够的时候。虚拟内存是操作系统内存管理的关键技术。使得多道程序运行和大程序运行称…

永久免费的iPhone,iPad,Mac,iWatch锁屏,桌面壁纸样机生成器NO.105

使用这个壁纸样机生成器&#xff0c;生成iPhone&#xff0c;iPad&#xff0c;Mac&#xff0c;iWatch锁屏&#xff0c;桌面壁纸&#xff0c;展示你的壁纸作品&#xff0c;一眼就看出壁纸好不好看&#xff0c;适不适合 资源来源于网络&#xff0c;免费分享仅供学习和测试使用&am…

【C语言初阶】分支语句

&#x1f31f;博主主页&#xff1a;我是一只海绵派大星 &#x1f4da;专栏分类&#xff1a;C语言 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 目录 一、什么是语句 二、if语句 悬空else 三、switch语句 default 四、switch语句与if-else语句性能对比如何&#xff1f…

【Python核心数据结构探秘】:元组与字典的完美协奏曲

文章目录 &#x1f680;一、元组⭐1. 元组查询的相关方法❤️2. 坑点&#x1f3ac;3. 修改元组 &#x1f308;二、集合⭐1. 集合踩坑❤️2. 集合特点&#x1f4a5;无序性&#x1f4a5;唯一性 ☔3. 集合&#xff08;交&#xff0c;并&#xff0c;补&#xff09;&#x1f3ac;4. …

手撕设计模式——克隆对象之原型模式

1.业务需求 ​ 大家好&#xff0c;我是菠菜啊&#xff0c;前俩天有点忙&#xff0c;今天继续更新了。今天给大家介绍克隆对象——原型模式。老规矩&#xff0c;在介绍这期之前&#xff0c;我们先来看看这样的需求&#xff1a;《西游记》中每次孙悟空拔出一撮猴毛吹一下&#x…

【电赛】STM32-PID直流减速电机小车【寻迹+避障+跟随】【更新ing】

一.需求分析 1.主控&#xff1a;STM32C8T6&#xff08;没什么好说的哈哈&#xff09; 2.电机&#xff1a;JAG25-370电机 【问】为什么要用直流减速电机&#xff1f;&#xff1f; PID控制器需要依靠精确的反馈信号来调整其输出&#xff0c;确保电机按照预定的速度和位置运行…