Mybatis 初识

目录

1. MyBatis入门

1.1 MyBatis的定义

1.2 MyBatis的核心

MyBatis的核心

JDBC 的操作回顾

1.3 MyBatis的执行流程

MyBatis基本工作原理

2. MyBatis的使用

2.1 MyBatis环境搭建

2.1.1 创建数据库和表

2.1.2 添加MyBatis框架支持

老项目添加MyBatis

新项目添加MyBatis

2.1.3 设置MyBatis配置信息

设置数据库连接的相关信息

MyBatis xml保存路径和xml命名规范

2.2 MyBatis 业务代码 (以查询为例)

2.2.1 MyBatis的开发模式

2.2.2 添加实体类

2.2.3 数据持久层实现

Interface

.xml

2.2.4 服务层Service

2.2.5 控制器层 Controller

2.2.6 程序测试

浏览器测试

Postman 测试


1. MyBatis入门

1.1 MyBatis的定义

MyBatis 是一款优秀的 持久层框架,它支持自定义 SQL、存储过程以及高级映射。MyBatis 去除了几乎所有的JDBC 代码以及设置参数和获取结果集的工作。MyBatis 可以通过简单的XML 或注解来配置和映射原始类型、接口和 Java POJO (Plain Old Java Obiects,普通老式 Java 对象)为数据库中的记录。

简单来说 MyBatis 是更简单完成程序和数据库交互的工具,也就是更简单的操作和读取数据库工具

MyBatis官网

持久化: 将内存中的数据长久的保存到磁盘中.
MyBatis是半自动化的框架: 要实现一个业务, 所有SQL需要自己实现.
存储过程: 将多个SQL语句连成一个方法.
高级映射: 将数据库中的多个表映射到一个对象中.

1.2 MyBatis的核心

对于后端开发来说,程序是由以下两个重要的部分组成的:

  1. 后端程序
  2. 数据库

而这两个重要的组成部分要通讯,就要依靠数据库连接工具MyBatis(之前则使用JDBC) 那么MyBatis是为了简化JDBC操作而诞生的.

MyBatis的核心

MyBatis的核心分为两部分

  • 配置MyBatis 开发环境;
  • 使用MyBatis 模式和语法操作数据库.

JDBC 的操作回顾

  1. 创建数据库连接池 DataSource
  2. 通过 DataSource 获取数据库连接 Connection
  3. 编写要执行带?占位符的 SQL 语句
  4. 通过 Connection 及 SQL 创建操作命令对象 Statement
  5. 替换占位符: 指定要替换的数据库字段类型,占位符索引及要替换的值
  6. 使用 Statement 执行 SQL 语句
  7. 查询操作: 返回结果集 ResultSet,更新操作:返回更新的数量
  8. 处理结果集
  9. 释放资源

下面的一个完整案例,展示了通过 JDBC 的 API 向数据库中添加一条记录,修改一条记录,查询一条记录的操作。

-- 创建数据库
create database if not exists `library` default character set utf8mb4;
-- 使用数据库
use library;
-- 创建表
create table if not exists `soft_bookrack` (
    `book_name` varchar(32) NOT NULL,
    `book_author` varchar(32) NOT NULL,
    `book_isbn` varchar(32) NOT NULL primary key
);

以下是 JDBC 操作的具体实现代码:

package com.example.demo;

import lombok.Getter;
import lombok.Setter;

import javax.sql.DataSource;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

public class SimpleJdbcOperation {
    private final DataSource dataSource;

    public SimpleJdbcOperation(DataSource dataSource) {
        this.dataSource = dataSource;
    }

    /**
     * 添加一本书
     */
    public void addBook() {
        Connection connection = null;
        PreparedStatement stmt = null;
        try {
            //获取数据库连接
            connection = dataSource.getConnection();
            //创建语句
            stmt = connection.prepareStatement(
                    "insert into soft_bookrack (book_name, book_author, book_isbn) values( ?,?,?);");
            //参数绑定
            stmt.setString(1, "Spring in Action");
            stmt.setString(2, "Craig Walls");
            stmt.setString(3, "9787115417305");
            //执行语句
            stmt.execute();
        } catch (SQLException e) {
            //处理异常信息
        } finally {
            //清理资源
            try {
                if (stmt != null) {
                    stmt.close();
                }
                if (connection != null) {
                    connection.close();
                }
            } catch (SQLException e) {
                //
            }
        }
    }

    /**
     * 更新一 本书
     */
    public void updateBook() {
        Connection connection = null;
        PreparedStatement stmt = null;
        try {
            //获取数据库连接
            connection = dataSource.getConnection();
            //创建语句
            stmt = connection.prepareStatement("update soft_bookrack set book_author=? where book_isbn=?;");
            //参数绑定
            stmt.setString(1, "张卫滨");
            stmt.setString(2, "9787115417305");
            //执行语句
            stmt.execute();
        } catch (SQLException e) {
            //处理异常信息
        } finally {
            //清理资源
            try {
                if (stmt != null) {
                    stmt.close();
                }
                if (connection != null) {
                    connection.close();
                }
            } catch (SQLException e) {
                //
            }
        }
    }

    /**
     * 查询一本书
     */
    public void queryBook() {
        Connection connection = null;
        PreparedStatement stmt = null;
        ResultSet rs = null;
        Book book = null;
        try {
            //获取数据库连接
            connection = dataSource.getConnection();
            //创建语句
            stmt = connection.prepareStatement("select book_name, book_author, book_isbn from soft_bookrack where book_isbn =?");
            //参数绑定
            stmt.setString(1, "9787115417305");
            //执行 语句
            rs = stmt.executeQuery();
            if (rs.next()) {
                book = new Book();
                book.setName(rs.getString("book_name"));
                book.setAuthor(rs.getString("book_author"));
                book.setIsbn(rs.getString("book_isbn"));
            }
            System.out.println(book);
        } catch (SQLException e) {
            //处理异常信息
        } finally {
            //清理资源
            try {
                if (rs != null) {
                    rs.close();
                }
                if (stmt != null) {
                    stmt.close();
                }
                if (connection != null) {
                    connection.close();
                }
            } catch (SQLException e) {
                //
            }
        }
    }

    @Setter
    @Getter
    public static class Book {
        private String name;
        private String author;
        private String isbn;
    }
}

从上述代码和操作流程可以看出,对于JDBC 来说,整个操作非常的繁琐,我们不但要拼接每一个参数,而且还要按照模板代码的方式,一步步的操作数据库,并且在每次操作完,还要手动关闭连接等而所有的这些操作步骤都需要在每个方法中重复书写。所以为了简化数据库的操作, 就有了MyBatis.


1.3 MyBatis的执行流程

上图描述了一个程序执行的标准流程, 那么其中的Mapper就是MyBatis的所在.
MyBatis包括Interface和.xml(实现接口), 那么Interface是方法的声明(接口), 但是没有方法的实现; 而方法的实现就在.xml中实现(SQL写在这里面). 但是在调用的时候调用的是Interface.
那么程序员操作的就是这两个组件, 这两个组件的操作也是基于JDBC的, 也就是说MyBatis是基于JDBC, 即 JDBC是给数据库上面封装了一层操作, 而MyBatis则是在JDBC基础上再加了一层, 能够更加的方便易用.

MyBatis基本工作原理

MyBatis 也是一个 ORM 框架,ORM(Object RelationalMapping),即对象关系映射。在面向对象编程语言中,将关系型数据库中的数据与对象建立起映射关系,进而自动的完成数据与对象的互相转换:

  1. 将输入数据 (即传入对象) +SQL 映射成原生 SQL
  2. 将结果集映射为返回对象,即输出对象

ORM 把数据库映射为对象:

  • 数据库表 (table) --> 类 (class)
  • 记录 (record,行数据)--> 对象 (object)
  • 字段 (field) --> 对象的属性 (attribute)

一般的 ORM 框架,会将数据库模型的每张表都映射为一个 Java 类。也就是说使用 MyBatis可以像操作对象一样来操作数据库中的表,可以实现对象和数据库表之间的转换。

2. MyBatis的使用

2.1 MyBatis环境搭建

2.1.1 创建数据库和表

首先我们要实现的功能是: 使用 MyBatis 的方式来读取用户表中的所有用户,我们使用个人博客的数据库和数据包,具体 SQL 如下。

-- 创建数据库
drop database if exists mycnblog;
create database mycnblog DEFAULT CHARACTER SET utf8mb4;

-- 使用数据数据
use mycnblog;

-- 创建表[用户表]
drop table if exists  userinfo;
create table userinfo(
    id int primary key auto_increment,
    username varchar(100) not null,
    password varchar(32) not null,
    photo varchar(500) default '',
    createtime timestamp default current_timestamp,
    updatetime timestamp default current_timestamp,
    `state` int default 1
) default charset 'utf8mb4';

-- 创建文章表
drop table if exists  articleinfo;
create table articleinfo(
    id int primary key auto_increment,
    title varchar(100) not null,
    content text not null,
    createtime timestamp default current_timestamp,
    updatetime timestamp default current_timestamp,
    uid int not null,
    rcount int not null default 1,
    `state` int default 1
)default charset 'utf8mb4';

-- 创建视频表
drop table if exists videoinfo;
create table videoinfo(
      vid int primary key,
      `title` varchar(250),
      `url` varchar(1000),
        createtime timestamp default current_timestamp,
        updatetime timestamp default current_timestamp,
      uid int
)default charset 'utf8mb4';

-- 添加一个用户信息
INSERT INTO `mycnblog`.`userinfo` (`id`, `username`, `password`, `photo`, `createtime`, `updatetime`, `state`) VALUES 
(1, 'admin', 'admin', '', '2021-12-06 17:10:48', '2021-12-06 17:10:48', 1);

-- 文章添加测试数据
insert into articleinfo(title,content,uid)
    values('Java','Java正文',1);
    
-- 添加视频
insert into videoinfo(vid,title,url,uid) values(1,'java title','http://www.baidu.com',1);

2.1.2 添加MyBatis框架支持

添加 MyBatis 框架支持分为两种情况:一种情况是对自己之前的 Spring 项目进行升级,另一种情况是创建一个全新的 MyBatis 和 Spring Boot 的项目,下面我们分别来演示这两种情况的具体实现。

老项目添加MyBatis

如果是在老项目中新增功能,添加框架支持:

在pom.xml下的<dependencies>标签内部Alt+Insert, 并点击Edit Starters(安装好这个插件),

点击OK, 即可在窗口中添加相关框架.

新项目添加MyBatis

如果是新项目创建Spring Boot 项目的时候添加引用就可以了,如下图所示:

注意: 添加MyBatis框架支持之后直接运行是会报错的, 所以要再设置MyBatis配置信息.

2.1.3 设置MyBatis配置信息

在application.properties配置文件中配置MyBatis相关信息. (也可以用yml, 这里省略)

设置数据库连接的相关信息

# 数据库连接配置
spring.datasource.url=jdbc:mysql://localhost:3306/mycnblog?characterEncoding=utf8&useSSL=false
spring.datasource.username=root
# 注意! password 要写自己本机 mysql 的密码, 不是此处固定的111111
spring.datasource.password=111111
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver

MyBatis xml保存路径和xml命名规范

# 设置 MyBatis
mybatis.mapper-locations=classpath:/mybatis/*Mapper.xml

下图为用户表的Mapper.xml, 即UserMapper.xml

2.2 MyBatis 业务代码 (以查询为例)

2.2.1 MyBatis的开发模式

配置完MyBatis的开发环境之后, 就要按照MyBatis的模式来开发我们的业务代码.

上图描述了MyBatis的开发模式, 针对于MyBatis来说, 它只有Interface和XXX.xml两个文件, Interface是给其他人调用的, 这个就是Java中普通的Interface接口, 然后通过注入的方式将这个接口注入上去, 那么问题是这个接口没有方法实现的, 前文提到, MyBatis在实现业务的时候是需要我们自己来写SQL代码的, 那么SQL就会写在XXX.xml中.
也就是说, 每一个功能都会对应两个组件, 一个是Interface给我们的服务层调用, 然后是针对Interface的代码实现.
那么在Java中只有普通的类可以实现接口, 重写接口的方法, 那xml怎么能够实现Interface中的方法?
这就是MyBatis的厉害之处, MyBatis通过代理的方式实现Interface中的方法, 也就是说, 对于我们用户来说, 我们感知的是Interface和XXX.xml, 到了MyBatis中运行时就会生成一个代理对象, 然后这个代理对象会将Interface里面的方法声明和XXX.xml里面的方法实现给它们组合成一个代理对象的方法进行填充.
也就是说在其他层去调用MyBatis, 其实就是调用那个代理对象了, 那么代理对象就是一个普通类了. 那么普通类中就自然而然有方法, 有方法的实现.
所以说, 我们看到的虽然是这两个组件, 但是最终实现的时候, MyBatis框架会帮我们封装好进行操作.

MyBatis模式开发由两部分组成:

  1. Interface: 让其他层可以注入使用的接口
  2. MyBatis: xml -> 具体实现SQL【它是上面Interface的"实现"】

明确了MyBatis中的构成之后我们才能够去合理的写它的代码, 由UserMapper.xml的图可知, xml的命名格式必须得是*Mapper.xml, 并且必须得放在/mybatis文件夹下. (注意: 配置了什么路径就得将xml放在什么文件夹下, 文件夹名要和路径相同)

2.2.2 添加实体类

先添加用户的实体类, 注意属性名和字段名尽量和SQL中的表一致, 这样MyBatis就会自动实现映射和关联(如果名字不一样就不会关联了):

package com.example.demo.entity;

import lombok.Data;

import java.time.LocalDateTime;

@Data
public class UserEntity {
    private Integer id;
    private String username;
    private String password;
    private String photo;
    private LocalDateTime createtime;
    private LocalDateTime updatetime;
    private Integer state;
}

2.2.3 数据持久层实现

Interface

以用户的查询功能为例, 在mapper下建立UserMapper, 注意这个类不需要有实现, 因为SQL是写在xml中, 它只需要有方法的声明让别人去调用即可.

package com.example.demo.mapper;

import com.example.demo.entity.UserEntity;
import org.apache.ibatis.annotations.Mapper;

import java.util.List;

@Mapper // 注意添加此注解
public interface UserMapper {
    List<UserEntity> getAll();
}

注意, 创建这个Interface之后的第一件事情就是加@Mapper注解, 因为这不是一个普通的类, 这个类是MyBatis的一个接口, 这个接口是需要和xml去对应起来的, 所以要把这个对象托管给MyBatis, 即加@Mapper注解.

以上就是第一个Interface的实现, 接下来就要实现*Mapper.xml, 它需要在resources.mybatis中创建

.xml


mybatis 的固定 xml 格式:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.demo.mapper.UserMapper">
   
</mapper>

添加 UserMapper.xmlUserMapper.xml 查询所有用户的具体实现 SQL:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.demo.mapper.UserMapper">
    <select id="getAll" resultType="com.example.demo.entity.UserEntity">
        select * from userinfo
    </select>
</mapper>

以下是对以上标签的说明

  • <mapper>标签: namespace属性是 指定当前xml实现的类是哪个Interface接口. 值为"包名.类名"
  • <select>查询标签: 是用来执行数据库的查询操作的:
    • id: 是和Interface(接口) 中定义的方法名称一样的,表示对接口的具体实现方法。
    • resultType: 是返回的数据类型,也就是开头我们定义的实体类。

2.2.4 服务层Service

添加 Service服务层, 实现代码如下:

package com.example.demo.service;

import com.example.demo.entity.UserEntity;
import com.example.demo.mapper.UserMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
public class UserService {
    @Autowired
    private UserMapper userMapper;

    public List<UserEntity> getAll(){
        return userMapper.getAll();
    }
}

2.2.5 控制器层 Controller

控制器层的实现代码如下:

package com.example.demo.controller;

import com.example.demo.entity.UserEntity;
import com.example.demo.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

@RequestMapping("/user")
@RestController
public class UserController {

    @Autowired
    private UserService userService;

    @RequestMapping("/getall")
    public List<UserEntity> getAll(){
        return userService.getAll();
    }
}

2.2.6 程序测试

浏览器测试

以上代码写完,整个 MyBatis 的查询功能就实现完了,接下来我们启动项目, 先使用浏览器输入路径的方式测试一下.

可以看到已经成功实现了查询功能, 查到了admin. 那么我们可以往数据库中再添加一条数据, 就可以查到第二条数据.

INSERT INTO `mycnblog`.`userinfo` (`id`, `username`, `password`, `photo`, `createtime`, `updatetime`, `state`) VALUES 
(2, 'zhangsan', 'zhangsan', '', '2021-12-06 17:10:48', '2021-12-06 17:10:48', 1);

插入第二条数据之后刷新页面, 可以看到数据成功的插入到数据库中了.

Postman 测试

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

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

相关文章

iphone拷贝照片中间带E自动去重软件,以及java程序如何打包成jar和exe

文章目录 一、前提二、问题描述三、原始处理方式四、程序处理4.1 java程序如何打包exe4.1.1 首先打包jar4.1.2 开始生成exe4.1.3 软件使用方式 4.2 更换图标4.2.1 更换swing的打包jar图标4.2.2 更换exe图标 4.3 如何使生成的exe在没有java环境的电脑上运行4.3.1 Inno Setup打包…

uniapp的uview-plus组件库的导入

uniapp的vue3中使用uview-plus组件库。在插件市场中找到该组件并点击如下所示绿色按钮&#xff0c;弹出弹窗选择要导入的项目后&#xff0c;就会在uni_modules文件中生成如下文件内容 关于插件的下载区别&#xff0c;可参考&#xff1a;https://uniapp.dcloud.net.cn/compone…

VSCode如何设置高亮

一、概述 本文主要介绍在 VSCode 看代码时&#xff0c;怎样使某个单词高亮显示&#xff0c;主要通过以下三步实现&#xff1a; 安装 highlight-words 插件 配置 highlight-words 插件 设置高亮快捷键F8 工作是嵌入式开发的&#xff0c;代码主要是C/C的&#xff0c;之前一直用…

uniapp 扩展组件 uni-forms 的表单验证之 validateFunction 只响应一次

uniapp 扩展组件 uni-forms 的表单验证之 validateFunction 只响应一次 问题代码官方说明参考资料 问题代码 直接从官方示例中复制过来改的。为了演示 <template><view><uni-forms ref"form" :modelValue"formData" :rules"rules&qu…

day 0815

计算文件有多少行&#xff1f; 2.文件的拷贝

Java中有哪些可以用于日期和时间的API?

从Java 8开始&#xff0c;java.time包提供了新的日期和时间API&#xff0c;新增的API严格区分了时刻、本地日期、本地时间&#xff0c;并且&#xff0c;对日期和时间进行运算更加方便。主要涉及的类型有以下几类&#xff1a; LocalDate&#xff1a;不包含具体时间的日期。 Lo…

ArcGIS入门操作手册

一.ArcGIS安装过程 参考本人博客&#xff1a;保姆级Arcgis安装图文安装教程_追忆苔上雪的博客-CSDN博客 二.ArcGIS植被指数计算 (1)使用工具&#xff1a;栅格计算器 打开软件&#xff0c;右侧搜索栅格计算器打开&#xff0c;要是搜索栏不小心叉掉找不到了&#xff0c;可以通…

初识结构体

文章目录 目录1. 结构体类型的声明1.1 结构的基础知识1.2 结构的声明1.3 结构成员的类型1.4 结构体变量的定义和初始化 2. 结构体成员的访问3. 结构体传参 目录 结构体类型的声明结构体初始化结构体成员访问结构体传参 1. 结构体类型的声明 1.1 结构的基础知识 结构是一些值的…

可白嫖的4家免费CDN,并测试其网络加速情况(2023版)

网站加载速度优化过程中&#xff0c;不可避免的会用上CDN来加速资源的请求速度。但是市面上的CDN资源几乎都是要收费的&#xff0c;而且价格还不便宜&#xff0c;对于小公司站长来讲&#xff0c;这将是一笔不小的开销。不过还是有一些良心公司给我们提供了免费的资源&#xff0…

私密相册管家-加密码保护私人相册照片安全

App Store史上最安全、最强大、最卓越的私密相册App&#xff01;再也不用担心私密照片视频被别人看见了&#xff01;
私密相册为你提供多重密码保护机制、简单便捷的照片存储空间&#xff0c;完美地将你的私密照片远离一切恶意偷窥者的窥探&#xff01; 【产品功能】
 √ 支…

【Pytorch:nn.Embedding】简介以及使用方法:用于生成固定数量的具有指定维度的嵌入向量embedding vector

文章目录 1、nn.Embedding2、使用场景 1、nn.Embedding 首先我们讲解一下关于嵌入向量embedding vector的概念 1&#xff09;在自然语言处理NLP领域&#xff0c;是将单词、短语或其他文本单位映射到一个固定长度的实数向量空间中。嵌入向量具有较低的维度&#xff0c;通常在几…

两天入门Linux、搭建Spring环境 第一天

一、Linux简介 1.什么是Linux 一个操作系统&#xff0c;未来公司里面会用到、接触的新操作系统。 2.为什么学Linux (1)个人职务需要&#xff0c;肯定会接触到Linux (2)职业发展&#xff0c;以后的发展肯定需要掌握Linux的许多使用方法 3.学哪些内容 (1)Linux基本介绍 (2)…

Prometheus入门

Prometheus(普罗米修斯) 是一种 新型监控告警工具,Kubernetes 的流行带动了 Prometheus 的应用。 全文参考自 prometheus 学习笔记(1)-mac 单机版环境搭建[1] Mac 上安装 Prometheus brew install prometheus 安装路径在 /usr/local/Cellar/prometheus/2.20.1, 配置文件在 /usr…

Qt 之 QPushButton,信号与槽机制

文章目录 前言一、QPushButton二、信号与槽机制总结 前言 一、QPushButton 当我们开发基于Qt框架的图形用户界面&#xff08;GUI&#xff09;应用程序时&#xff0c;经常需要在界面上添加按钮来实现用户交互。Qt提供了一个名为 QPushButton 的类作为按钮控件的实现。QPushButt…

题解 | #1002.Shortest path# 2023杭电暑期多校9

1002.Shortest path 签到题 记忆化搜索 题目大意 给定一个正整数 n n n &#xff0c;可以对其进行以下操作&#xff1a; 如果 n n n 能被 3 3 3 整除&#xff0c;则可以使 n n / 3 nn/3 nn/3 ;如果 n n n 能被 2 2 2 整除&#xff0c;则可以使 n n / 2 nn/2 nn/2 …

探索Java中的静态变量与实例变量:存储区域、生命周期以及内存分配方式的区别

文章目录 静态变量实例变量不可变对象静态变量和实例变量有什么区别&#xff1f;静态变量实例变量 Object 类都有哪些公共方法&#xff1f;Java 创建对象有哪几种方式&#xff1f;ab 与 a.equals(b) 有什么区别&#xff1f;总结 &#x1f389;欢迎来到Java面试技巧专栏~探索Jav…

贝锐蒲公英:快速搭建连锁门店监控体系,赋能企业高效管理

随着国民生活水平的提高和零售场景的变革&#xff0c;消费者对于餐饮类目的消费支出不断增加&#xff0c;线下社区生鲜商超作为下沉市场最主要的消费场景之一&#xff0c;蕴藏着巨大价值机会。 对于线下连锁生鲜超市而言&#xff0c;连锁门店多、员工多&#xff0c;门店管理时会…

如何使用CSS实现一个响应式网格布局?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 使用CSS实现响应式网格布局⭐ 设置基本的HTML结构⭐ 创建基本的CSS样式⭐ 添加媒体查询以实现响应式效果⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启航前端…

【Vue-Router】路由传参

1. query 传参 list.json {"data": [{"name": "面","price":300,"id": 1},{"name": "水","price":400,"id": 2},{"name": "菜","price":500,"…

C#质检工具(StyleCop、SonarLint)

1、StyleCop StyleCop工具主要类似java中的checkStyle,是检查代码样式规范的工具。 1.1、StyleCop安装流程: 图1.1 图1.2 图1.3 安装StyleCop插件时可能会遇到下载特慢或卡住不动的情况,需注意: 1)网上说的关闭IPV6功能不管用 2)网上说的自动指定dns不管用 3)网上…