mysql的锁-->一篇读懂所有锁机制

目录

mysql的锁

概述:根据mysql锁的大类型可以分为

我们先来讲一下范围最大的全局锁

使用

为什么要使用全局锁?

使用全局锁进行备份的缺点

表级锁

表锁

1.共享读表锁的语法

2.排斥写表锁

元数据锁

意向锁

什么是意向锁

 怎么产生意向锁

意向锁的作用

 AUTO-INC 锁

行级锁

 记录锁

记录共享锁

记录排斥锁 

间隙锁

next-key锁

​编辑

 插入意向锁

行级锁是怎么加锁的

主键索引等值查询

1.等值查询的值存在,next-key锁退化成这个值的记录锁。

2.等值查询的数据不存在,next-key锁会退化成间隙锁

​编辑 主键索引范围查询

大于的情况 

1.条件值不存在 如 id>5  5是条件值

2.条件值存在

 小于的情况

1.条件值不存在

2.条件值存在

​编辑

二级索引的普通索引的等值查询

1.等值查询的值存在

2.查询的值不存在

普通索引的范围查询

大于的情况

1.条件值不存在

2.条件值存在

 小于的情况

1.条件值存在

​编辑 2.条件值不存在

 没有使用的索引字段进行查询,修改删除的加锁情况

 死锁的产生

死锁就是这样产生的

间隙锁在不同的事务之间可以共存,那不是会频繁的导致死锁的产生吗?

死锁会大概率发生在最后一个数据之后的那一段区间

那如果发生了死锁要怎么去解决

 插入遇见唯一键冲突


mysql的锁

主要规则:读读锁兼容,写写锁不兼容,读写锁不兼容

概述:根据mysql锁的大类型可以分为

  • 全局锁
  • 表级锁:表锁 意向锁 元数据锁 AUTO-INC锁
  • 行级锁:记录锁,间隙锁,next-key锁(记录锁加间隙锁)

我们先来讲一下范围最大的全局锁

使用

只要全局锁一旦加上, 所有的增删改的操作都不能被执行,整个数据库只会处于只读状态

在一个会话里加上全局锁

flush tables with read lock

只能执行读操作,执行增删改操作会直接报错

在其他会话里

也可以执行查询操作,但是执行增删改操作会被阻塞,直到全局锁释放才会去执行当前操作

解锁

unlock tables

解锁之后 ,阻塞的操作就会被执行

为什么要使用全局锁?

我们需要对整个数据库的数据进行备份时 ,就需要使用全局锁。

如果在一个库中有一张订单表,商品余额表

当我们对这个数据库的数据进行备份时,先对订单表进行备份,这时一个其他的并发事务执行了购买操作,减少的订单余额,这时候再去备份的商品余额表就与之前备份的订单表数据不一致了。

使用全局锁进行备份的缺点

当我们使用全局锁时,只有查询的业务可以运行,而增删改的业务都被阻塞,这是我们十分不想看到的,那有什么方法可以解决吗,有的,如果数据库的引擎支持的事务支持可重复读的隔离级别,那么在备份数据库之前先开启事务,会先创建 Read View,,然后整个事务执行期间都在用这个 Read View,而且由于 MVCC 的支持,备份期间业务依然可以对数据进行更新操作。

这就是事务中的隔离性,就算其他事务对数据表进行了操作,数据库备份时只会对read view进行备份

备份数据库的工具是 mysqldump,在使用 mysqldump 时加上 –single-transaction 参数的时候,在备份数据库之前先开启事务。这种操作只适合于可重复读级别的隔离机制,而mysql默认的innodb存储引擎就是可重复读的隔离机制。

表级锁

表锁

我们现在来讲以下表级锁的表锁 ,这是一个用来锁着一整张表的锁。

可以分为共享读表锁,排斥写表锁。

1.共享读表锁的语法

共享读表锁就是让这张表只能读,增删改操作都不可以执行

lock tables 表名 read;

演示共享读表锁

在一个会话里给person表加上共享读表锁

可以发现对这张表只能读,增删改操作都会直接报错

 在其它会话里

也可以读,但是增删改操作都会被阻塞

 释放锁

其他会话的阻塞操作也会被执行

注意:如果当前会话对person表加了共享读表锁后,也不能操作其他表的数据了,连查询都不可以,其他会话则可以

2.排斥写表锁

如果当前会话对一张表加了排斥写表锁,那么只能让当前对话能对这张表进行CRUD操作,其他会话不能查询这个表,也不能增删改

lock  table 表名 write; 

 演示

当前会话对person加了排斥写表锁,可以进行CRUD操作

其他会话,所有对这个表的操作都会被阻塞

我们也可以发现对整个表直接进行加排斥写锁后,其他会话都无法操作 这个表,对并发事务十分不友好,性能十分低下,所以innodb有了行锁这种更小粒度的锁。

元数据锁

我们之前讲的全局锁和表锁都是需要我们进行显示调用的,而这个元数据锁,意向锁(增删改)和行锁(增删改)是自动调用的,所以我们需要开启事务来查询加锁的情况,因为事务一旦结束,这些锁都会自动释放。这也是我们为什么一定要开启事务来查看加锁的情况,因为mysql是自动提交事务的(执行一行sql后自动提交),然后这些自动加的锁我们就无法观察到了。

元数据锁的作用是什么-->用来解决DML语句和DDL语句的冲突,假设我们在事务a中进行了增删改操作,而事务b把表的结构给改了,这显然是不被允许的。 

元数据锁的类型-->MDL读锁(进行CRUD时的事务进行加的锁)

                        -->MDL写锁(进行修改表结构时的事务加的锁)

 这两个锁是不兼容的,在事务a中先对person表进行了CRUD操作,那么这个表会被加上MDL读锁,这时事务b想对person表进行DDL,操作表数据时,想要去给这个表加MDL写锁时,因为MDL读锁与MDL写锁不兼容,所以MDL写锁无法加上,DDL语句也无法执行,b事务就只能阻塞等待a事务提交释放MDL读锁。

但是经过我的操作,如果先在事务a修改的表结构,在事务b中可以进行CURD操作

演示

在事务a中进行CURD操作,这是这张表已经被加上的MDL读锁

 事务b想要去修改表结构时,会阻塞,因为MDL写锁无法加上,与MDL读锁无法兼容,会阻塞等待

事务a提交

表结构成功改变 

意向锁

意向锁进行增删改操作也是自动调用加锁的,事务结束自动解锁。

什么是意向锁

意向锁和行锁是息息相关的,我们在对行数据进行加锁时,会先产生一个表级别的意向锁,共享读行锁产生意向共享锁,排斥写行锁产生意向排斥锁。 

意向锁之间也不会发生冲突

  •  意向共享锁读可以兼容共享读表锁,与排斥写表锁不兼容(读读锁共享)
  • 意向排斥写锁与共享读表锁不兼容,与排斥写表锁不兼容(写读锁不共享,写写锁不共享)
 怎么产生意向锁

普通的select语句不会产生行锁,只会产生MDL读锁

这就要先讲一下怎么产生共享读行锁,排斥写行锁

共享读行锁:加上这个共享读行锁之前,先在表加上了意向共享读锁

select ... lock in share mode;

 排斥写行锁:加上这个排斥写行锁之前,先在表加上了意向排斥写锁

select ... for update;

当执行插入、更新、删除操作,需要先对表加上「意向独占锁」 ,然后对该记录加独占锁。

执行插入操作

产生了意向排斥锁 

 IX

 select * from performance_schema.data_locks\G;

意向锁的作用

 在对一张表加表锁时,需要判断这张表里有没有行锁,如果没有意向锁,那就需要去全表扫描判断有没有行锁,这显然效率十分低下,如果有了意向锁就可以直接判断这张表里有没有行锁。

 AUTO-INC 锁

现在数据表进行插入数据时,主键都是自增的,使用auto_increment;

AUTO-INC锁就是实现这个的基础,在插入数据时,先给这个表加上AUTO-INC锁,然后把赋值数据给自增字段,插入完成之后,锁会自动释放

所以这个AUTO-INC锁不是事务结束后释放,而是执行插入语句结束后释放。

 AUTO-INC 锁再对大量数据进行插入的时候,会影响插入性能,因为另一个事务中的插入会被阻塞(等着被赋值)。 

InnoDB 存储引擎提供了一种轻量级的锁来实现自增。

一样也是在插入数据的时候,会为被 AUTO_INCREMENT 修饰的字段加上轻量级锁,然后给该字段赋值一个自增的值,就把这个轻量级锁释放了,而不需要等待整个插入语句执行完后才释放锁

行级锁

innodb相比于myisam的三大特点:支持事务,外键约束,行级锁

行级锁的类型主要有三类:

  • Record Lock,记录锁,也就是仅仅把一条记录锁上(更新和删除数据时会对这个数据加排斥写锁,插入数据时也会,防止被其他事务修改);
  • Gap Lock,间隙锁,锁定一个范围,但是不包含记录本身;
  • Next-Key Lock:Record Lock + Gap Lock 的组合,锁定一个范围,并且锁定记录本身。(加入有1,3两个数据,next-lock锁的是3记录本身,和1,3之间的范围)

 记录锁

记录锁也分有记录共享读锁和记录排斥写锁,也是经典的读读锁兼容,读写锁不兼容,写写锁不兼容

记录共享锁

如何加上记录共享读锁

再使用select 语句后加上lock in share mode;

演示:

这里给id=1的数据加上了记录共享读锁

可以看到现在的锁有表级锁 意向共享锁,记录共享锁

S,REC_NOT_GAP就是 记录共享锁,然后lock_data加锁的行数据

 其他事务也可以继续加记录共享锁

现在就有四个锁了 ,也是可以发现意向锁之间可以兼容,而是与共享读表锁和排斥写表锁不兼容

但是不可以加记录排斥锁

 加了就阻塞

记录排斥锁 

使用更新,删除操作时,还有在select 语句后加for update时就加上记录排斥锁

给id=1的数据加上记录排斥锁

可以发现现有两个锁,意向排他锁, 

其他事务要更新这个id=1的数据时会阻塞,因为更新和删除操作都要去加排斥记录锁,但是现在事务a已经给id=1加了排斥记录锁,而排斥记录锁之间不兼容,所以会阻塞。

 

在事务a中给id=2的数据加上排斥记录锁

就会有两个意向排斥锁,也可以看出来意向锁之间互相兼容

 

间隙锁

间隙锁存在于可重复读的隔离机制中,主要用来解决幻读的问题,什么是幻读-->就是在一个事务中前后查询的记录数不同(其他事务进行了删除操作或者插入操作)

间隙锁虽然也分为S型和X型,但是不同的间隙锁之间是兼容的。

演示:

现在查询了id=3的数据,这是不存在的,所以会在(2,4)之间加了一个间隙锁。

 这就会有两个锁

一个锁是表级别的意向排斥锁,还有一个是行级的间隙锁

X,GAP 4 就是id=4数据前面那段的区间,也就是(2,4)

插入id=3的数据时就会阻塞,因为(2,4)这个区间已经加了间隙锁 ,想要插入数据时,就需要获取一个区间的插入意向锁(行级锁),这个锁与间隙锁不兼容,所以会阻塞,无法获取插入意向锁

next-key锁

 next-key是间隙锁和记录锁的总和,其实间隙锁和记录锁都是由next-key临键锁退化来的(怎么退化,下文会讲)

注意next-key也是满足读读锁兼容,读写锁不兼容,写写锁不兼容。

虽然间隙锁之间都互相兼容,但是记录锁不是互相兼容的。

演示

在事务a查询id>=12的数据加next-key锁

事务b有一个表级别的意向排斥锁

还有12数据的记录锁,还有(12,+∞】的next-key锁 因为这里∞的数据是不存在的,所以只是不能插入id大于12的数据

如果加的是(2,4]的next-key锁,就不能插入id=3的数据,也不能修改或者删除4的数据

 插入意向锁

插入意向锁是一个行级锁,是想要在一个区间内插入数据时要获取的锁,与间隙锁不兼容

演示:

无法插入id=15的数据,因为在(12,+无穷]的区间里有间隙锁,所以无法加上这个区间内的插入意向锁 

也就是说加入一个事务a有了一个区间的间隙锁,其他事务就无法获取这个区间的插入意向锁,无法插入数据,但是事务a是可以插入数据的,因为这个锁是事务a的 (锁就是防止并发事务的操作导致数据不一致)

行级锁是怎么加锁的

我们上文只是简单介绍了记录锁,间隙锁和next-key锁的简单概念

这里我们来介绍一下行级锁是怎么加的 

主键索引等值查询

1.等值查询的值存在,next-key锁退化成这个值的记录锁。

这个记录就不会被其他事务进行修改和删除

可以发现对id=1的数据加了记录锁

2.等值查询的数据不存在,next-key锁会退化成间隙锁

id=3的数据不存在,会在(3,4)范围加上范围锁

 主键索引范围查询

大于的情况 
1.条件值不存在 如 id>5  5是条件值

条件值不存在,>和>=的情况是一样的

这里我查询id>=11的数据

11的数据不存在,会在(10,12]加上next-key锁,还有(12,+无穷]的next-key锁

2.条件值存在

>11的情况

id=11存在,会在 (11,12]加上next-key锁和(12,+无穷]加上next-ley锁

>=11的情况

id=11存在,会在id=11的记录锁,(11,12]加上next-key锁和(12,+无穷]加上next-ley锁

 

 小于的情况
1.条件值不存在

<2的情况 id=2数据不存在

在(1,3)加上间隙锁,(-无穷,1]加上next-key锁

<=2的情况 

同上

2.条件值存在

<2的情况

(1,2)间隙锁,(-无穷,1]加上next-key锁

<=2的情况

(1,2]的next-key锁, (-无穷,1]加上next-key锁

二级索引的普通索引的等值查询

因为唯一索引的加锁情况跟主键索引的情况类似,只不过给唯一索引加锁时,还要给主键索引的记录加上记录锁,因为二级索引的数据就是主键id

条件值存在:因为普通索引的值不是唯一的,所以普通索引的等值查询也是一个扫描的过程,在扫描的过程中,会给符合条件的普通索引加上next-key锁,并给对应的主键id的索引加上记录锁。直到第一个不符合条件的索引key值,然后这个不符合条件的索引的next-key退化成间隙锁

条件值不存在:就是扫描到第一个比条件值大的索引时,这个索引的next-key直接退化成间隙锁

这里给age字段加上普通索引 

1.等值查询的值存在

如查询 age=17

会在(16,17]加上next-key锁,(17,18)就加间隙锁,这里假设16,18的值存在

还会给age=17的记录的主键索引加上记录锁

查询age等于25

可以发现加的锁有(24,25]id=1的next-key锁,(25,25]id=7的next-key锁,(25,27)的间隙锁,还有在主键索引的id=1和id=7的记录锁

这里注意,如果插入age=24的数据,且id<6(当前age=24的数据id=6)可以插入,因为,没有间隙锁,如果插入age=24,id>6就无法插入,阻塞等待,因为有24,25]id=1的next-key锁

注:二级索引的b加树上,字段值相同就按主键值排序

还有如果插入age=27的数据,且id<8则无法插入((25,27)的间隙锁),age=27,id>8可以插入

 

 

2.查询的值不存在

 会加一个间隙锁

普通索引的范围查询

普通索引进行范围查询时,不会出现next-key锁退化成记录锁和间隙锁的情况

大于的情况
1.条件值不存在

如age>=26

会在(25,27]的next-key锁,后面的数据都是next-key锁

2.条件值存在

如age>27

在(27,28]加上next-key锁,(28,+无穷]加上next-key锁,还有28的主键id也加上记录锁

如age>=27

会在(25,27]id=4加上next-key锁,后面也都是next-key锁

这里28虽然是最后一条数据,但是mysql会维护一个supremum pseudo-record的值,代表无穷大

这是一个(28,+∞]的next-key锁

还会在主键索引的id=4,8,9,10的索引上加记录锁,防止被其他事务进行修改和删除

为什么第一个27的索引数据不会跟主键索引一样退换成记录锁,因为一旦退化成记录锁,那么其他事务是可以在第一个27索引数据前 插入一个age=27,且id<4的新数据,那么这样就会导致数据前后不一致。

 小于的情况
1.条件值存在

可以发现这里加了(-无穷,13]id=2,(13,13]id=3的next-key锁,也给id=1,3主键索引加了记录锁,这都是正常的,但是(13,18]的next-key锁并没有退化成间隙锁,也没有给id=18的主键索引加上记录锁 

 

其他事务想要去修改age=18的数据也会被阻塞 

 

使用主键id索引就可以修改成功 

 

所以我还是不太清楚为什么(13,18]的next-key锁不直接退化成间隙锁,这样都无法直接修改age=18的数据 

就算是<= ,后面第一个不符合数据的索引的数据也不会退化成记录锁

(13,18]next-key锁不会退化成间隙锁

 2.条件值不存在

(13,18]这个next-key锁也不会退化成间隙锁 ,但是并没有对主键索引id=12加上记录锁

 

 没有使用的索引字段进行查询,修改删除的加锁情况

如果没有使用索引字段进行操作,就会去把全表扫描把所有主键索引id的记录都加上next-key的行级别锁,导致其他事务无法对这张表进行操作,这显然是不对。

name字段没有索引

给每一条主键索引都加上了next-key锁,就相当于加上了表级别的排斥写表锁 

 死锁的产生

这里我们首先得先知道

意向锁也可以共存,不同事务都可以对同一张表进行加意向锁,因为意向锁的存在是防止其他事务直接对这张表直接加上表级锁(意向共享锁可以于共享读表锁兼容,不可以于排斥写表锁兼容)

间隙锁的意义只在于阻止区间被插入,因此不同事务之间加的间隙锁是可以共存的 ,共享和排他的间隙锁是没有区别的,他们相互不冲突,且功能相同,即两个事务可以同时持有包含共同间隙的间隙锁。

这里的共同间隙包括两种场景:

  • 其一是两个间隙锁的间隙区间完全一样;
  • 其二是一个间隙锁包含的间隙区间是另一个间隙锁包含间隙区间的子集。

而接着我们要了解插入意向锁,这个锁是想要在某一个区间进行插入数据时要加的锁,而且事务a在一个区间有了间隙锁,事务b无法在这个区间获取插入意向锁,也就无法插入数据。可以发现同一个区间的插入意向锁和间隙锁在不同的事务之间不兼容。

死锁就是这样产生的

事务a和事务b都在某一个区间有着相同的间隙锁,那事务a想要在这个区间插入数据时,因为事务b有着这个区间间隙锁,所以事务a就无法操作,而事务b也想要插入数据时,也无法插入,所以这两个事务就一直相互阻塞着,导致死锁。

间隙锁在不同的事务之间可以共存,那不是会频繁的导致死锁的产生吗?

所以虽然间隙锁可以共存,但是记录锁不可以,而next-key是包括间隙锁和记录锁的所以只要某一个区间内包括记录锁,不同的事务之间就无法加上相同范围的锁,就可以大概率避免死锁的发生。

死锁会大概率发生在最后一个数据之后的那一段区间

最后一个数据之后加的next-key锁是 (最后一个数据,+无穷】,但是无穷这个值是不存在的,也可以近似的看作成是间隙锁。

那如果发生了死锁要怎么去解决

死锁的条件:

互斥、占有且等待、不可强占用、循环等待

1.设置事务等待锁的超时时间

当一个事务的等待时间超过该值后,就对这个事务进行回滚,于是锁就释放了,另一个事务就可以继续执行了,在 InnoDB 中, 参数 innodb_lock_wait_timeout 是用来设置超时时间的,默认值时 50 秒。

2.开启主动死锁检测

主动死锁检测在发现死锁后,主动回滚死锁链条中的某一个事务,让其他事务得以继续执行。将参数 innodb_deadlock_detect 设置为 on,表示开启这个逻辑,默认就开启。

所以当事务a,b都有最后一段的间隙锁时,又都想对这个区间进行插入数据,相互阻塞导致死锁时,这两个事务会有一个事务回滚退出事务,假设b事务退出,那么事务b加的锁也就会释放,那么事务a就可以在这个区间插入数据了。

 插入遇见唯一键冲突

插入意向锁是针对某一个区间的。

那如果我往已经存在的主键id又插入一个id相同的数据,会直接报错误,插入新记录的事务还会给已存在的主键值重复的聚簇索引记录添加 S 型记录锁

如果唯一二级索引重复,插入新记录的事务都会给已存在的二级索引列值重复的二级索引记录添加 S 型 next-key 锁

插入普通索引的值重复,因为普通索引的next-key不会直接退化成记录锁,如 age=18 for update;

会在(17,18]有next-key锁,(18,19)有间隙锁,所以不会导致插入相同数据的值,导致幻读。

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

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

相关文章

Vue 实现智能检测文字是否溢出,溢出显示省略号,鼠标悬浮显示全部【附封装组件完整代码+详细注释+粘贴即食】

一、场景需求 在项目中&#xff0c;经常会遇到文本内容超出容器的情况。为了提高用户体验&#xff0c;我希望在文字溢出时显示悬浮提示&#xff0c;未溢出时则不显示。 二、效果演示 三、实现原理 DOM宽度对比法&#xff1a;通过比较元素的scrollWidth&#xff08;实际内容宽…

用Deepseek写一个 HTML 和 JavaScript 实现一个简单的飞机游戏

大家好&#xff01;今天我将分享如何使用 HTML 和 JavaScript 编写一个简单的飞机游戏。这个游戏的核心功能包括&#xff1a;控制飞机移动、发射子弹、敌机生成、碰撞检测和得分统计。代码简洁易懂&#xff0c;适合初学者学习和实践。 游戏功能概述 玩家控制&#xff1a;使用键…

《Spring日志整合与注入技术:从入门到精通》

1.Spring与日志框架的整合 1.Spring与日志框架进行整合&#xff0c;日志框架就可以在控制台中&#xff0c;输出Spring框架运行过程中的一些重要的信息。 好处&#xff1a;方便了解Spring框架的运行过程&#xff0c;利于程序的调试。 Spring如何整合日志框架 Spring5.x整合log4j…

关于mybatis查询时,时间字段的映射问题

目录 1.mysql中&#xff0c;关于时间的两种类型 1.1 date 1.2 datetime 2.mybatis从mysql数据库查询出上述两种类型的字段后&#xff0c;映射到Java实体类时的问题 3.结语 1.mysql中&#xff0c;关于时间的两种类型 1.1 date 格式&#xff1a;2002-09-23 特点&#xff1a…

高效自动化测试:打造Python+Requests+Pytest+Allure+YAML的接口测试框架

一、背景 在快节奏的开发周期中&#xff0c;如何确保接口质量&#xff1f;自动化测试是关键。通过构建标准化、可复用的测试框架&#xff0c;能显著提升测试效率与准确性&#xff0c;为项目质量保驾护航[1][7]。 二、目标 ✅ 核心目标&#xff1a; ● 实现快速、高效的接口测试…

【鸿蒙开发】MongoDB入门

https://www.mongodb.com/try/download/community 下载MongoDB: var mongoose require("mongoose");// localhost 域名&#xff0c;代表本机 // 127.0.0.1 ip , 代码本机 mongoose.connect("mongodb://localhost:27017/jiaju").then(() > {console.l…

Linux中的TCP编程接口基本使用

TCP编程接口基本使用 本篇介绍 在UDP编程接口基本使用已经介绍过UDP编程相关的接口&#xff0c;本篇开始介绍TCP编程相关的接口。有了UDP编程的基础&#xff0c;理解TCP相关的接口会更加容易&#xff0c;下面将按照两个方向使用TCP编程接口&#xff1a; 基本使用TCP编程接口…

wireshark 如何关闭混杂模式 wireshark操作

Fiddler和Wireshark都是进行抓包的工具&#xff1a;所谓抓包就是将网络传输发送与接收的数据包进行截获、重发、编辑、转存等操作&#xff0c;也用来检查网络安全。抓包也经常被用来进行数据截取等。黑客常常会用抓包软件获取你非加密的上网数据&#xff0c;然后通过分析&#…

IDEA2024又一坑:连接Docker服务连不上,提示:Cannot run program “docker“: CreateProcess error=2

为新电脑安装了IDEA2024版&#xff0c;因为局域网中安装有Docker,所以这台电脑上没有安装&#xff0c;当运行时发现死活连不上Docker报&#xff1a;Cannot run program “docker“: CreateProcess error2 分析&#xff1a; Docker服务有问题 其它电脑都能连&#xff0c;排除 网…

文件包含漏洞第一关

一、什么是文件包含漏洞 1.文件包含漏洞概述 和SQL注入等攻击方式一样&#xff0c;文件包含漏洞也是一种注入型漏洞&#xff0c;其本质就是输入一段用户能够控制的脚本或者代码&#xff0c;并让服务端执行。 什么叫包含呢&#xff1f;以PHP为例&#xff0c;我们常常把可重复使…

网络安全事件响应--应急响应(windows)

应用系统日志 Windows主要有以下三类日志记录系统事件&#xff1a;应用程序日志、系统日志和安全日志。 系统和应用程序日志存储着故障排除信息&#xff0c;对于系统管理员更为有用。安全日志记录着事件审计信息&#xff0c;包括用户验证&#xff08;登录、远程访问等&#x…

C++蓝桥杯基础篇(九)

片头 嗨&#xff01;小伙伴们&#xff0c;大家好~ 今天我们将学习蓝桥杯基础篇&#xff08;十&#xff09;&#xff0c;学习函数相关知识&#xff0c;准备好了吗&#xff1f;咱们开始咯&#xff01; 一、函数基础 一个典型的函数定义包括以下部分&#xff1a;返回类型、函数名…

JVM内存结构笔记01-运行时数据区域

文章目录 前言运行时数据区域1.程序计数器定义特点总结 2.虚拟机栈2.1 定义局部变量表 ★操作数栈动态链接方法返回地址(方法出口) 2.2 栈内存溢出演示栈内存溢出 java.lang.StackOverflowError 2.3问题辨析1. 垃圾回收是否涉及栈内存&#xff1f;2. 栈内存分配越大越好吗&…

01-简单几步!在Windows上用llama.cpp运行DeepSeek-R1模型

1.llama.cpp介绍 Llama.cpp 是一个开源的、轻量级的项目&#xff0c;旨在实现 Meta 推出的开源大语言模型 Llama 的推理&#xff08;inference&#xff09;。Llama 是 Meta 在 2023 年开源的一个 70B 参数的高质量大语言模型&#xff0c;而 llama.cpp 是一个用 C 实现的轻量化…

对开源VLA sota π0的微调——如何基于各种开源数据集、以及你自己的私有数据集微调π0(含我司的微调实践)

前言 25年2.4日&#xff0c;几个月前推出π0的公司Physical Intelligence (π)宣布正式开源π0及π0-FAST&#xff0c;如之前所介绍的&#xff0c;他们对用超过 10,000 小时的机器人数据进行了预训练 该GitHub代码仓库「 π0及π0-FAST的GitHub地址&#xff1a;github.com/Ph…

Redis网络模型

redis为什么快 1.主要原因是因为redis是基于内存操作的&#xff0c;比起直接操作磁盘速度快好几倍 2.基于内存的数据库瓶颈主要是在网络io这一块&#xff0c;redis网络模型采用io多路复用技术能够高效的处理并发连接。 3.redis使用单线程执行命令&#xff0c;可以避免上下文…

PyTorch系列教程:Tensor.view() 方法详解

这篇简明扼要的文章是关于PyTorch中的tensor.view()方法的介绍与应用&#xff0c;与reshape()方法的区别&#xff0c;同时给出示例进行详细解释。 Tensor基础 Tensor(张量)的视图是一个新的Tensor&#xff0c;它与原始Tensor共享相同的底层数据&#xff0c;但具有不同的形状或…

Python数据分析之数据可视化

Python 数据分析重点知识点 本系列不同其他的知识点讲解&#xff0c;力求通过例子让新同学学习用法&#xff0c;帮助老同学快速回忆知识点 可视化系列&#xff1a; Python基础数据分析工具数据处理与分析数据可视化机器学习基础 四、数据可视化 图表类型与选择 根据数据特…

swift -(5) 汇编分析结构体、类的内存布局

一、结构体 在 Swift 标准库中&#xff0c;绝大多数的公开类型都是结构体&#xff0c;而枚举和类只占很小一部分 比如Bool、 Int、 Double、 String、 Array、 Dictionary等常见类型都是结构体 ① struct Date { ② var year: Int ③ var month: Int ④ …

推荐一个比较好的开源的工作流引擎

由于DeepSeek等AI大模型的出现&#xff0c;工作流模式再次流行起来&#xff0c;低代码甚至零代码就可以实现应用开发&#xff0c;而且有DeepSeek这样的超级AI作为大脑&#xff0c;人人都可以开发自动化工作流。 比如搭建邮件助手工作流&#xff0c;可以自动润色各种邮件内容。…