后端之路第三站(Mybatis)——结合案例讲Mybatis怎么操作sql

先讲一下准备工作整体流程要做什么

我们要基于一个员工管理系统作为案例,进行员工信息的【增、删、改、查】

原理就是用Mybatis通过java语言来执行sql语句,来达到增、删、改、查】

一、准备工作

1、引入数据库数据

首先我们把一个员工、部门表的数据在数据库里建好先

老弟们我直接把黑马的资源代码放这里了,各位不用再去找、网盘下载,直接拿我下面这个代码放数据库查询控制台执行一下就行

-- 数据准备:-- 部门管理
create table dept(id int unsigned primary key auto_increment comment '主键ID',name varchar(10) not null unique comment '部门名称',create_time datetime not null comment '创建时间',update_time datetime not null comment '修改时间'
) comment '部门表';
insert into dept (id, name, create_time, update_time) values(1,'学工部',now(),now()),(2,'教研部',now(),now()),(3,'咨询部',now(),now()), (4,'就业部',now(),now()),(5,'人事部',now(),now());-- 员工管理
create table emp (id int unsigned primary key auto_increment comment 'ID',username varchar(20) not null unique comment '用户名',password varchar(32) default '123456' comment '密码',name varchar(10) not null comment '姓名',gender tinyint unsigned not null comment '性别, 说明: 1 男, 2 女',image varchar(300) comment '图像',job tinyint unsigned comment '职位, 说明: 1 班主任,2 讲师, 3 学工主管, 4 教研主管',entrydate date comment '入职时间',dept_id int unsigned comment '部门id, 说明: 1 学工部,2 教研部, 3 咨询部, 4 就业部, 5 人事部',create_time datetime not null comment '创建时间',update_time datetime not null comment '修改时间'
) comment '员工表';INSERT INTO emp(id, username, password, name, gender, image, job, entrydate, dept_id, create_time, update_time) VALUES(1,'jinyong','123456','金庸',1,'1.jpg',4,'2000-01-01',2,now(),now()),(2,'zhangwuji','123456','张无忌',1,'2.jpg',2,'2015-01-01',2,now(),now()),(3,'yangxiao','123456','杨逍',1,'3.jpg',2,'2008-05-01',2,now(),now()),(4,'weiyixiao','123456','韦一笑',1,'4.jpg',2,'2007-01-01',2,now(),now()),(5,'changyuchun','123456','常遇春',1,'5.jpg',2,'2012-12-05',2,now(),now()),(6,'xiaozhao','123456','小昭',2,'6.jpg',3,'2013-09-05',1,now(),now()),(7,'jixiaofu','123456','纪晓芙',2,'7.jpg',1,'2005-08-01',1,now(),now()),(8,'zhouzhiruo','123456','周芷若',2,'8.jpg',1,'2014-11-09',1,now(),now()),(9,'dingminjun','123456','丁敏君',2,'9.jpg',1,'2011-03-11',1,now(),now()),(10,'zhaomin','123456','赵敏',2,'10.jpg',1,'2013-09-05',1,now(),now()),(11,'luzhangke','123456','鹿杖客',1,'11.jpg',5,'2007-02-01',3,now(),now()),(12,'hebiweng','123456','鹤笔翁',1,'12.jpg',5,'2008-08-18',3,now(),now()),(13,'fangdongbai','123456','方东白',1,'13.jpg',5,'2012-11-01',3,now(),now()),(14,'zhangsanfeng','123456','张三丰',1,'14.jpg',2,'2002-08-01',2,now(),now()),(15,'yulianzhou','123456','俞莲舟',1,'15.jpg',2,'2011-05-01',2,now(),now()),(16,'songyuanqiao','123456','宋远桥',1,'16.jpg',2,'2007-01-01',2,now(),now()),(17,'chenyouliang','123456','陈友谅',1,'17.jpg',NULL,'2015-03-21',NULL,now(),now());

2、然后就是之前那些创建springboot工程、引入mybatis等等依赖的配置操作

因为我之前第一篇讲过了就不再演示了,跟着我配置好了的可以不用再创建多余的工程,直接在原先的数据库多加两个表而已,反正还是用原来的spring boot工程项目来连接同一个数据库

配置那一篇的链接:后端之路第三站(Mybatis)——入门配置-CSDN博客

当然如果你非要另开一个数据库来连接的话,那你就重新配置一遍并连接这个数据库吧,不过注意,可以在引入依赖的时候,除了【Mybatis Framework】和【MySQL Driver】还可以直接再勾选【Developer Tools】的【Lombok】依赖,因为这样就不用我们再去【pom.xml文件】那里在手动写代码的形式引入lombok的依赖了

3、然后连接你的数据库

这里我懒得换数据库,还是在原来的数据库建两个表而已,所以我就不演示了,想看的还是看我这一篇:后端之路第三站(Mybatis)——入门配置-CSDN博客

4、然后对应你建的表,在java这创建对应的实体类

别忘了

整型数:int   在java对应   Integer

字符串:varchar( )   在java对应   String

               char( )   在java对应   String

短整型数:tinyint   在java对应   Short

日期:date 在java对应   LocalDate

时间:datetime  在java对应   LocalDateTime

员工表的实体类

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;import java.time.LocalDate;
import java.time.LocalDateTime;@Data
@NoArgsConstructor
@AllArgsConstructor
public class Emp {private Integer id;private String username;private String password;private String name;private Short gender;private String image;private Short job;private LocalDate entrydate;private Integer dept_id;private LocalDateTime create_time;private LocalDateTime update_time;
}

部门表的实体类

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;import java.time.LocalDateTime;@Data
@NoArgsConstructor
@AllArgsConstructor
public class Dept {private Integer id;private String name;private LocalDateTime create_time;private LocalDateTime update_time;
}

这里我建议各位自己不要直接拿我代码复制粘贴,最好自己收敲一遍,加深代码流程记忆

6、到mapper目录创建【对应xxx表的SQL语句执行】的接口

二、开始sql语句操作

噢忘了跟各位提一嘴,idea的右边侧边栏是可以连接数据库的

点开这里,点击“+”加号添加数据源,就跟在数据库软件里连接数据库一样的,写个用户名密码啥的,我已经链接了我就不展示了,有手就行

然后每当我们在数据库创建好新的表之后,都要记得去idea这里刷新一下,不然的话idea不知道你多了两表,写代码的时候就没有相关的提示了

1、删除

删除的sql语句是:【 delete from 表 where 条件

删除的Mybatis注解是:【 @Delete( "delete from 表 where 条件" 

那么在MySQL里我们是这么写

在Mybatis就是

哪有的人就会问:你这where id = 17不就写死了吗?那如果我想动态的删除员工,而不是固定死只删除id为17的员工,怎么办?还有你下面那个public void delete();又是什么玩意?

比如我要实现管理系统里动态删除,点那个就删哪个

@XXX是注解,对于执行sql语句的注解,你要是想执行的话不能单单靠一个注解就执行啊,你必须得带一个方法,让外部调用你这个方法,才能执行你这个@注解来操作数据库啊 

那么好,这里我们就应该在下面写一个方法,首先考虑我们这个sql语句不是查询语句,不需要返回!!!所以直接类型为void无返回

然后加参数!在外面调用这个接口的这个方法时会动态地传入参数,这里我们考虑根据id来删,那么比如在管理系统里,我点击第2个人的“删除”,就传入一个【id=2】的参数

现在我们有了参数还剩最后一步,改@Delete这个注解里的sql代码了,很简单【#{ 变量 }】这样就可以动态绑定变量了!!是不是很像前端的【`${ 变量 }`】?

完整代码

import org.apache.ibatis.annotations.Delete;
import org.apache.ibatis.annotations.Mapper;
import org.springframework.beans.factory.annotation.Autowired;@Mapper
public interface EmpMapper {//@Sql的注解( "要执行的sql语句" )@Delete("delete from emp where id = #{id}")//对应这个sql语句执行的一些相关方法public void delete(Integer id);
}

ok,然后现在测试就行了,老步骤:

1、test的...ApplicationTests.java文件

2、在类里面@Autowired注解,然后创建EmpMapper的对象变量使用

3、然后@Test注解直接跟上要运行的方法,在方法里调用EmpMapper的delete方法并传入实参

运行搞定

总结

#{ 变量 } ${ 变量 }

在application.properties文件配置好这个下面这个代码,以后就可以在控制台看到具体的日志(也就是直观的mybatis执行sql语句的过程)

不用记,直接复制

#配置mybatis的日志,指定输出到控制台
mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

那么为什么我们where id = #{ 变量 } 在控制台日志里会变成 where id =

原理是:

#{ 变量 } 的语法会生成【预编译语句】

${ 变量 } 的语法不会生成【预编译语句】,而是像前端的${ 变量 } 一样直接把变量直接拼接到字符串

至于有什么用?预编译的 #{ } ${ } 有什么优劣?

简单的例子,不用深究了解

如果用 ${ } 直接字符串拼接的形式,就会

根据一整个语句,字符串里的【or、and、=】这些字符会被当成【判断逻辑符号】用,如果条件永远为真,那就算这个密码错的也可以登录

然而 #{ } 的是预编译,字符串以一整个参数形式传入,【or、and、=】这些字符还是当成普通字符串来判断

2、新增

新增的sql语句是:

(表里所有字段都传值时) —>【 insert into 表 values (对应的所有字段值1),(对应的所有字段值2)...... 】

(表里指定部分字段传值时) —>【 insert into 表(字段1,字段2...) values (对应的字段的值1),(对应的字段的值2)...... 】

新增的Mybatis注解是:

(表里所有字段都传值时) —>【 @Insert( "insert into 表 values (对应的所有字段值1),(对应的所有字段值2)...... "  】

(表里指定部分字段传值时) —>【 @Insert( "insert into 表(字段1,字段2...) values (对应的字段的值1),(对应的字段的值2)......

现在我们想要完成一个新增员工的这么一个功能

在数据库里sql语句是

(不会的先去学MySQL的知识,这里就是根据“用户名、姓名、性别、图像、职位、入职日期、归属部门、这个信息创建时间、这个信息的更新时间”来执行sql的新增信息,【这个信息创建时间、这个信息的更新时间】这两是一般更新新增信息固定带有的信息)

那么在Mybatis就这么写

还是那个问题,这是死数据,我需要动态传值,那就把value里写成 #{ } 参数

然后public void insert( )里定义参数,好让外部调用时传入实参

注意:

@注解里的sql语句】的参数名、表名要跟数据库的一样;因为它接收对应数据库的值

@注解里的values】的的参数是要跟java里的实体类的参数名一样!!!因为它接收java的值

​​​​​​​

方法里】的参数是要跟java里的实体类的参数名、类名一样!!!因为它接收java的值

然后因为新增操作涉及大量参数,所以在java里传多个值可以直接用【对象】!!所以【方法里】直接传【对象】即可,然后@注解里的values】的 #{ 变量 } 会自动解析出对象里的对应的变量的属性

那么就到最后一步,在外部ApplicationTest方法里调用方法

但是下面注意:这样就错了!!我们在接口定义的参数是一整个对象,不是这么零散的参数

那就要先构建一个实体类的对象,把这些参数作为对象的成员变量装进去,然后整个对象给回Mapper接口

@Test
public void testInsert(){//这样就错了!!我们在接口定义的参数是一整个对象,不是这么零散的参数//empMapper.insert("CZM","岑梓铭",(short)1,"1.jpg",(short)3, LocalDate.of(2024,1,1),1, LocalDateTime.now(),LocalDateTime.now());//得先构建一个对象,然后把对应的值调用setter方法设置//对象里没用的成员变量就不用设置,不设置的变量就是默认值,也不会报错也不会有影响Emp emp1 = new Emp();emp1.setJob((short)3);emp1.setGender((short)1);emp1.setUsername("CZM");emp1.setName("岑梓铭");emp1.setImage("1.jpg");emp1.setEntrydate(LocalDate.of(2024,1,1));emp1.setDept_id(1);emp1.setCreate_time(LocalDateTime.now());emp1.setUpdate_time(LocalDateTime.now());empMapper.insert(emp1);
}

能看到又报错了,很明显我们#{ }的地方有写错java里对应的变量

改一下就成功了

3、主键返回

上一个新增操作,我们因为在数据库设置了id字段是自增的主键,那么即使我们不给id传值,它也会自动在数据库增加id的值,这时数据库的基础知识

那么既然有值,我们能不能在java拿到数据库返回的id这个主键的值呢?不行

因为默认普通的插入是不会有返回值的,那要有返回值咋办?

加一个注解

@Option(useGeneratedKeys = true, keyProperty = "主键那个字段")

现在再执行一下

成功

4、更新

更新的sql语句是:

update set  表  字段1=新值1,字段2=值2......  where  条件 】

新增的Mybatis注解是:

@Update( "update set  表  字段1=新值1,字段2=值2......  where  条件 "  】

现在要实现这么个功能,点击编辑可以更改员工信息

那么根据图片的需求信息总结,需要更新的字段值是:username 、name、gender、image、job 、entrydate 、dept_id、还有一个【update_time】(新增的时候要有“新建时间” 和 “更新时间”,那更新就要有 “更新时间”)

sql里就应该这么写

那么现在Mybatis就应该这么写:(我就直接写成动态传参的了,不示范死数据格式了)

成功更新

5、查询

查询的sql语句是:

select  *  from  表  where  条件 】

查询的Mybatis注解是:

@Select( "select  *  from  表  where  条件"  】

sql语句是这样

Mybatis就应该这么写

成功

查询中的一些数据封装问题:

用我这篇文章里的代码的朋友们在刚刚的查询操作中应该不会有什么问题,因为我的代码里的【Java的实体类Emp里的成员变量的名】和【数据库的emp表的字段名】是一样的,所以查询时会对应映射、封装数据

但是有些跟着黑马程序员视频的朋友或者自己有自己的代码书写风格的人,可能在【数据库表的字段名】用的是xxx_xxx,然后在【Java的实体类的成员变量的名】用的是xxxXxx,然后因为二者名字不一样而导致数据没有对应封装

解决办法:

1、(个人首选推荐):在application.properties配置里加下面的代码

它会自动把【xxx_xxx】转换映射成【xxXxx】,很方便(前提必须严格按要求:数据库起名是xx_xxx、java这边起名是驼峰形xxXxx)

#开启mybatis的驼峰命名自动映射开关
mybatis.configuration.map-underscore-to-camel-case = true

2、(个人次推荐):咱就老老实实跟数据库表里的字段名都一样不就完事了嘛.......

3、(略麻烦):通过Results、Result注解手动映射封装

4、(巨麻烦):sql里的起别名的方法,在sql语句里把 * 换成写出所有字段名,然后名字不同的那几个在后面空格、再跟上别名,别名就是【Java的实体类的成员变量的名】

6、略复杂的查询

现在实现这么一个功能,根据姓名、性别、入职时间范围来组合条件查询

要求是:

        名字里含有什么字啥?

        性别是啥?

        入职时间在(时间范围)之间

        并要求按 “员工信息的更新时间” 来倒序排序显示

例子:

        名字里含有“张”字

        性别是男(1)

        入职时间在(2010年1月1日 — 2020年1月1日)之间

        并要求按 “员工信息的更新时间” 来倒序排序显示

那么在数据库sql是这样写

然后注意一下,这里sql里模糊查询条件like后面必须跟“字符串”,然后要想动态传值就得#{变量}

但是!!!#{变量}产生的预编译语句的“?”不能被包在字符串里!!!

那么有的教程就会教你用另一个“${变量}”,因为“${变量}”是直接拼接字符串嘛

但是要注意,单纯像下图这么写的话,老子试了半个小时越试越特么烦,浑身刺挠,

因为黑马程序员这个老毕登瞎钩八讲,在视频最后才提到说如果要传递多个参数,说什么springBoot 1.x版本要用【@Param(“变量”)】这个注解来指定#{ 变量 }的变量是下面函数里的参数

放他娘狗屁!早又不说非要老子查他妈半年bug,然后说的还是错的

记住了!!!不管你是哪个版本,哪怕是昨天spring公司开更新的最新版,只要Mapper接口的方法要传递多个参数时,都给我加上【@Param(“变量”)】在每一个参数前面!!!!!

ok,那现在还有一个问题,不是说${ }容易被攻击吗?想用#{ }?也可以

直接用这个方法:sql里用 【concat()方法】 可以拼接字符串!!!!

那么Mybatis这边这样写

成功

案例中完整的Mybatis代码如下

Mapper目录下EmpMapper的所有执行sql的操作接口代码

//这块别复制,你们自己有自己的路径
//package com.czm.mybatis01.mapper;//这块别复制,你们自己有自己定义的类的位置
//import com.czm.mybatis01.table.Emp;import org.apache.ibatis.annotations.*;
import org.springframework.beans.factory.annotation.Autowired;import java.time.LocalDate;
import java.util.List;@Mapper
public interface EmpMapper {//@Sql的注解( "要执行的sql语句" )@Delete("delete from emp where id = ${id}")//对应这个sql语句执行的一些相关方法public void delete(Integer id);@Options( useGeneratedKeys = true, keyProperty = "id" )@Insert( "insert into emp(username,name,gender,image,job,entrydate,dept_id,create_time,update_time)"+" values (#{username},#{name},#{gender},#{image},#{job},#{entrydate},#{dept_id},#{create_time},#{update_time})" )public void insert(Emp emp);@Options( useGeneratedKeys = true, keyProperty = "id")@Update(" update emp set username = #{username}, name = #{name}, gender = #{gender}, image = #{image}," +"job = #{job}, entrydate = #{entrydate}, dept_id = #{dept_id}, update_time = #{update_time} where id = #{id}; ")public void Update(Emp emp);//批量查询所有信息@Select( "select * from emp" )public List<Emp> selectAll();//根据id查询员工信息@Select( "select * from emp where id = #{id}" )public Emp select(Integer id);//复杂查询员工
//    @Select( "select * from emp where" +
//            " name like '%${name}%' " +
//            " and gender = #{gender}" +
//            " and entrydate between #{begin} and #{end}" +
//            " order by update_time desc" )@Select( "select * from emp where" +" name like concat('%',#{name},'%') " +" and gender = #{gender}" +" and entrydate between #{begin} and #{end}" +" order by update_time desc" )public List<Emp> DifficultSelect(@Param("name") String name, @Param("gender") Short gender, @Param("begin") LocalDate begin, @Param("end") LocalDate end);}

test目录下的ApplicationTest测试文件的代码:

//这块别复制,你们自己有自己的路径
//package com.czm.mybatis01;//这块别复制,你们自己有自己定义的接口的位置
//import com.czm.mybatis01.mapper.EmpMapper;
//import com.czm.mybatis01.mapper.UsrMapper;
//import com.czm.mybatis01.table.Emp;
//import com.czm.mybatis01.table.User;import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.List;@SpringBootTest //springboot整合单元测试的注解
class Mybatis01ApplicationTests {//使用@Autowired注解可以【依赖注入】,直接创建UserMapper接口的实例化对象//可以理解为跳过了【创建实现接口类】这一步,spring boot帮我们创建好了一个@Autowiredprivate EmpMapper empMapper;@Testpublic void testDelete(){//我这里为了简单测试,就当是前端已经把id传过来了,就是17,直接把实际参数传进去empMapper.delete(17);}@Testpublic void testInsert(){//这样就错了!!我们在接口定义的参数是一整个对象,不是这么零散的参数//empMapper.insert("CZM","岑梓铭",(short)1,"1.jpg",(short)3, LocalDate.of(2024,1,1),1, LocalDateTime.now(),LocalDateTime.now());//得先构建一个对象,然后把对应的值调用setter方法设置//对象里没用的成员变量就不用设置,不设置的变量就是默认值,也不会报错也不会有影响Emp emp1 = new Emp();emp1.setJob((short)3);emp1.setGender((short)2);emp1.setUsername("CYH");emp1.setName("蔡勇豪");emp1.setImage("19.jpg");emp1.setEntrydate(LocalDate.of(2024,1,1));emp1.setDept_id(1);emp1.setCreate_time(LocalDateTime.now());emp1.setUpdate_time(LocalDateTime.now());empMapper.insert(emp1);System.out.println(emp1.getId());}@Testpublic void testUpdate(){Emp emp2 = new Emp();emp2.setId(1);emp2.setName("王大陆");emp2.setUsername("WDL");emp2.setGender((short)1);emp2.setImage("19.jpg");emp2.setJob((short)2);emp2.setEntrydate(LocalDate.of(2024,1,3));emp2.setDept_id(2);emp2.setUpdate_time(LocalDateTime.now());empMapper.Update(emp2);}@Testpublic void testSelect(){empMapper.selectAll();empMapper.select(8);}@Testpublic void TestDifficultSelect(){List<Emp> list = empMapper.DifficultSelect("张", (short)1, LocalDate.of(2010,1,1), LocalDate.of(2020,1,1));System.out.println(list);}
}

基础操作就到此,下一篇:XML映射文件

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

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

相关文章

【51单片机入门】速通定时器

文章目录 前言定时器是什么初始化定时器初始化的大概步骤TMOD寄存器C/T寄存器 触发定时器中断是什么中断函数定时器点亮led 总结 前言 在嵌入式系统的开发中&#xff0c;定时器是一个非常重要的组成部分。它们可以用于产生精确的时间延迟&#xff0c;或者在特定的时间间隔内触…

Java--常用类APl(复习总结)

前言: Java是一种强大而灵活的编程语言&#xff0c;具有广泛的应用范围&#xff0c;从桌面应用程序到企业级应用程序都能够使用Java进行开发。在Java的编程过程中&#xff0c;使用标准类库是非常重要的&#xff0c;因为标准类库提供了丰富的类和API&#xff0c;可以简化开发过…

【接口自动化测试】第三节.实现项目核心业务接口自动化

文章目录 前言一、实现登录接口对象封装和调用 1.0 登录接口的接口测试文档 1.1 接口对象层&#xff08;封装&#xff09; 1.2 测试脚本层&#xff08;调用&#xff09;二、课程新增接口对象封装和调用 2.0 课程新增接口的接口测试文档 2.1 接口对象层…

AVL树模拟

1.概念 虽然二叉搜索树可以缩短查找的效率&#xff0c;但如果数据有序或者接近有序时二叉搜索树树将退化为单支树&#xff0c;查找元素相当于在顺序表中搜索元素&#xff0c;效率低下。AVL 树是具有一下性质的二叉搜索树&#xff1a; 1.它的左右子树都是AVL树 2.左右子…

Mac 如何安装 wget

1.安装 Homebrew2.安装 wget3.检测 wget 是否安装成功 1.安装 Homebrew 在安装 wget 之前需要安装一个适用于 mac 的包管理器 Homebrew&#xff0c;打开 mac 终端执行如下命令进行安装&#xff1a; /usr/bin/ruby -e "$(curl -fsSL https://cdn.jsdelivr.net/gh/ineo6/h…

新手第一个漏洞复现:MS17-010(永恒之蓝)

文章目录 漏洞原理漏洞影响范围复现环境复现步骤 漏洞原理 漏洞出现在Windows SMB v1中的内核态函数srv!SrvOs2FeaListToNt在处理FEA&#xff08;File Extended Attributes&#xff09;转换时。该函数在将FEA list转换成NTFEA&#xff08;Windows NT FEA&#xff09;list前&am…

【Golang - 90天从新手到大师】Day14 - 方法和接口

一&#xff0e; go方法 go方法&#xff1a;在函数的func和函数名间增加一个特殊的接收器类型&#xff0c;接收器可以是结构体类型或非结构体类型。接收器可以在方法内部访问。创建一个接收器类型为Type的methodName方法。 func (t Type) methodName(parameter list) {}go引入…

解决“Duplicate keys detected: ‘ ‘.This may cause an update error.”问题

问题原因 出现“Duplicate keys detected”的错误&#xff0c;通常表示在v-for指令中使的:key绑定值有重复。 如果前端是静态数据&#xff0c;一般能自我避免:key绑定值有重复。如果前端是绑定的动态数据&#xff0c;那么需要另外提供一个唯一的键。 在这个例子中&#xff0c…

【LeetCode】接雨水

目录 一、题目二、解法完整代码 一、题目 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例 1&#xff1a; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释&#xff…

【UML用户指南】-23-对高级行为建模-状态机

目录 1、概述 2、状态 2.1、状态的组成 3、转移 3.1、转移的组成 4、高级状态和转移 4.1、进入效应和退出效应 4.2、内部转移 4.3、do活动 4.4、延迟事件 4.5、子状态机 5、子状态 5.1、非正交子状态 5.2、历史状态 5.3、正交子状态 6、分叉与汇合 7、主动对象…

GOROOT GOPATH GOPROXY GO111MODULE

GOROOT GOROOT代表Go的安装目录。可执行程序go(或go.exe)和gofmt(或gofmt.exe)位于 GOROOT/bin目录中。 配置GOROOT环境变量&#xff0c;其值为Go的安装目录&#xff1b;然后在环境变量PATH中添加GOROOT/bin路径。 注意&#xff1a;GOROOT变量只是代表了安装目录&#xff0c;不…

【面试题】信息系统安全运维要做什么

信息系统安全运维是确保信息系统稳定、可靠、安全运行的一系列活动和措施。 其主要包括以下几个方面&#xff1a; 1.系统监控&#xff1a; 实时监测信息系统的运行状态&#xff0c;如服务器的性能指标、网络流量、应用程序的运行情况等。通过监控工具&#xff0c;及时发现系统…

企业数据治理的下一步是数据资产管理?

随着信息技术的飞速发展和数字化转型的深入推进&#xff0c;企业数据已成为驱动业务增长和创新的核心要素。当企业数据治理工作取得显著成效后&#xff0c;如何进一步发挥数据的价值&#xff0c;实现数据资产的有效管理&#xff0c;成为企业面临的重要课题。 数据治理的基石作用…

学习感悟丨在誉天学习数通HCIP怎么样

大家好&#xff0c;我是誉天学员的徐同学&#xff0c;学习的数通HCIP课程。 在学校的时候&#xff0c;听说下半年就要出去实习了&#xff0c;心中坎坷不安&#xff0c;现在我学到的知识远远不够的。然后就想着学点东西充实一下自己的知识面和专业能力&#xff0c;有一次和同学谈…

【漏洞复现】飞企互联——SQL注入

声明&#xff1a;本文档或演示材料仅供教育和教学目的使用&#xff0c;任何个人或组织使用本文档中的信息进行非法活动&#xff0c;均与本文档的作者或发布者无关。 文章目录 漏洞描述漏洞复现测试工具 漏洞描述 飞企互联-FE企业运营管理平台是一个基于云计算、智能化、大数据…

[图解] 向量数据库之何谓乘积量化器?

Product Quantization 在前面一节讲解了向量数据库索引相关的内容&#xff0c;那么本节将会讲解其中压缩方法的量化手段&#xff1a;乘积量化器。 简单来说将向量的所有维度划分为多个子空间&#xff0c;每个子空间一部分维度&#xff0c;然后每个子空间独立去找最近距离。例如…

haproxy实现代理和负载均衡

HaProxy介绍&#xff1a; haproxy是法国开发者威利塔罗在2000年使用C语言开发的一个开源软件&#xff0c;是一款具备高并发(一万以上)、高性能的TCP和HTTP负载均衡器&#xff0c;支持基于cookie的持久性&#xff0c;自动故障切换&#xff0c;支持正则表达式及web状态统计&…

Numpy array和Pytorch tensor的区别

1.Numpy array和Pytorch tensor的区别 笔记来源&#xff1a; 1.Comparison between Pytorch Tensor and Numpy Array 2.numpy.array 4.Tensors for Neural Networks, Clearly Explained!!! 5.What is a Tensor in Machine Learning? 1.1 Numpy Array Numpy array can only h…

信息学奥赛初赛天天练-39-CSP-J2021基础题-哈夫曼树、哈夫曼编码、贪心算法、满二叉树、完全二叉树、前中后缀表达式转换

PDF文档公众号回复关键字:20240629 2022 CSP-J 选择题 单项选择题&#xff08;共15题&#xff0c;每题2分&#xff0c;共计30分&#xff1a;每题有且仅有一个正确选项&#xff09; 5.对于入栈顺序为a,b,c,d,e的序列&#xff0c;下列( )不合法的出栈序列 A. a&#xff0c;b&a…

螺旋矩阵问题C代码

给定一个n行m列的二维数组&#xff0c;要求按顺时针螺旋顺序输出矩阵中的所有元素&#xff0c;n和m小于等于10 如下图是一个三行四列的螺旋矩阵 要求输出 1 2 3 4 8 12 11 10 9 5 6 7 全局变量定义 int a[11][11]; int vis[11][11]; // 访问标记数组关键代码如下 int dx[] …