当前位置: 首页 > news >正文

【MySQL】-10 MySQL 存储过程

MySQL 存储过程

    • 优点
    • 缺点
    • 一、存储过程的创建和调用
      • 创建存储过程
      • 实例
      • 1、in 输入参数
      • 2、out输出参数
      • 3、inout输入参数
    • 三、变量
      • 1. 变量定义
      • 2. 变量赋值
      • 3. 用户变量
    • 四、注释
      • MySQL存储过程的调用
      • MySQL存储过程的查询
      • MySQL存储过程的修改
      • MySQL存储过程的删除
      • MySQL存储过程的控制语句
    • 例子代码

MySQL 5.0 版本开始支持存储过程。

存储过程(Stored Procedure)是一种在数据库中存储复杂程序,以便外部程序调用的一种数据库对象。

存储过程是为了完成特定功能的SQL语句集,经编译创建并保存在数据库中,用户可通过指定存储过程的名字并给定参数(需要时)来调用执行。

存储过程思想上很简单,就是数据库 SQL 语言层面的代码封装与重用。

优点

  • 存储过程可封装,并隐藏复杂的商业逻辑。
  • 存储过程可以回传值,并可以接受参数。
  • 存储过程无法使用 SELECT 指令来运行,因为它是子程序,与查看表,数据表或用户定义函数不同。
  • 存储过程可以用在数据检验,强制实行商业逻辑等。

缺点

  • 存储过程,往往定制化于特定的数据库上,因为支持的编程语言不同。当切换到其他厂商的数据库系统时,需要重写原有的存储过程。
  • 存储过程的性能调校与撰写,受限于各种数据库系统。

一、存储过程的创建和调用

  • 存储过程就是具有名字的一段代码,用来完成一个特定的功能。
  • 创建的存储过程保存在数据库的数据字典中。

创建存储过程

CREATE[DEFINER = { user | CURRENT_USER }]PROCEDURE sp_name ([proc_parameter[,...]])[characteristic ...] routine_bodyproc_parameter:[ IN | OUT | INOUT ] param_name typecharacteristic:COMMENT 'string'| LANGUAGE SQL| [NOT] DETERMINISTIC| { CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }| SQL SECURITY { DEFINER | INVOKER }routine_body:Valid SQL routine statement[begin_label:] BEGIN[statement_list]……
END [end_label]

MYSQL 存储过程中的关键语法

声明语句结束符,可以自定义:

DELIMITER $$
或
DELIMITER //

声明存储过程:

CREATE PROCEDURE demo_in_parameter(IN p_in int)       

存储过程开始和结束符号:

BEGIN .... END    

变量赋值:

SET @p_in=1  

变量定义:

DECLARE l_int int unsigned default 4000000; 

创建mysql存储过程、存储函数:

create procedure 存储过程名(参数)

存储过程体:

create function 存储函数名(参数)

实例

创建数据库,备份数据表用于示例操作:

create database db1; 
use db1;     
create table PLAYERS as select * from TENNIS.PLAYERS; 
create table MATCHES  as select * from TENNIS.MATCHES;

下面是存储过程的例子,删除给定球员参加的所有比赛:

mysql> delimiter $$  #将语句的结束符号从分号;临时改为两个$$(可以是自定义)
mysql> CREATE PROCEDURE delete_matches(IN p_playerno INTEGER)-> BEGIN->   DELETE FROM MATCHES->    WHERE playerno = p_playerno;-> END$$
Query OK, 0 rows affected (0.01 sec)mysql> delimiter;  #将语句的结束符号恢复为分号

**解析:**默认情况下,存储过程和默认数据库相关联,如果想指定存储过程创建在某个特定的数据库下,那么在过程名前面加数据库名做前缀。 在定义过程时,使用 DELIMITER ∗ ∗ 命令将语句的结束符号从分号 ∗ ∗ ; ∗ ∗ 临时改为两个 ∗ ∗ ** 命令将语句的结束符号从分号 **;** 临时改为两个 ** 命令将语句的结束符号从分号;临时改为两个,使得过程体中使用的分号被直接传递到服务器,而不会被客户端(如mysql)解释。

调用存储过程:

call sp_name[(传参)];
mysql> select * from MATCHES;
+---------+--------+----------+-----+------+
| MATCHNO | TEAMNO | PLAYERNO | WON | LOST |
+---------+--------+----------+-----+------+
|       1 |      1 |        6 |   3 |    1 |
|       7 |      1 |       57 |   3 |    0 |
|       8 |      1 |        8 |   0 |    3 |
|       9 |      2 |       27 |   3 |    2 |
|      11 |      2 |      112 |   2 |    3 |
+---------+--------+----------+-----+------+
5 rows in set (0.00 sec)mysql> call delete_matches(57);
Query OK, 1 row affected (0.03 sec)mysql> select * from MATCHES;
+---------+--------+----------+-----+------+
| MATCHNO | TEAMNO | PLAYERNO | WON | LOST |
+---------+--------+----------+-----+------+
|       1 |      1 |        6 |   3 |    1 |
|       8 |      1 |        8 |   0 |    3 |
|       9 |      2 |       27 |   3 |    2 |
|      11 |      2 |      112 |   2 |    3 |
+---------+--------+----------+-----+------+
4 rows in set (0.00 sec)

**解析:**在存储过程中设置了需要传参的变量p_playerno,调用存储过程的时候,通过传参将57赋值给p_playerno,然后进行存储过程里的SQL操作。

存储过程体

  • 存储过程体包含了在过程调用时必须执行的语句,例如:dml、ddl语句,if-then-else和while-do语句、声明变量的declare语句等
  • 过程体格式:以begin开始,以end结束(可嵌套)
BEGINBEGINBEGINstatements; ENDEND
END

**注意:**每个嵌套块及其中的每条语句,必须以分号结束,表示过程体结束的begin-end块(又叫做复合语句compound statement),则不需要分号。

为语句块贴标签:

[begin_label:] BEGIN[statement_list]
END [end_label]

例如:

标签有两个作用:- 1、增强代码的可读性
- 2、在某些语句(例如:leave和iterate语句),需要用到标签## 二、存储过程的参数MySQL存储过程的参数用在存储过程的定义,共有三种参数类型,IN,OUT,INOUT,形式如:​```mysql
CREATEPROCEDURE 存储过程名([[IN |OUT |INOUT ] 参数名 数据类形...])
  • IN 输入参数:表示调用者向过程传入值(传入值可以是字面量或变量)
  • OUT 输出参数:表示过程向调用者传出值(可以返回多个值)(传出值只能是变量)
  • INOUT 输入输出参数:既表示调用者向过程传入值,又表示过程向调用者传出值(值只能是变量)

1、in 输入参数

mysql> delimiter $$
mysql> create procedure in_param(in p_in int)-> begin->   select p_in;->   set p_in=2;->    select P_in;-> end$$
mysql> delimiter ;mysql> set @p_in=1;mysql> call in_param(@p_in);
+------+
| p_in |
+------+
|    1 |
+------++------+
| P_in |
+------+
|    2 |
+------+mysql> select @p_in;
+-------+
| @p_in |
+-------+
|     1 |
+-------+

以上可以看出,p_in 在存储过程中被修改,但并不影响 @p_id 的值,因为前者为局部变量、后者为全局变量。

以上可以看出,p_in 在存储过程中被修改,但并不影响 @p_id 的值,因为前者为局部变量、后者为全局变量。

2、out输出参数

mysql> delimiter //
mysql> create procedure out_param(out p_out int)->   begin->     select p_out;->     set p_out=2;->     select p_out;->   end-> //
mysql> delimiter ;mysql> set @p_out=1;mysql> call out_param(@p_out);
+-------+
| p_out |
+-------+
|  NULL |
+-------+#因为out是向调用者输出参数,不接收输入的参数,所以存储过程里的p_out为null
+-------+
| p_out |
+-------+
|     2 |
+-------+mysql> select @p_out;
+--------+
| @p_out |
+--------+
|      2 |
+--------+#调用了out_param存储过程,输出参数,改变了p_out变量的值

3、inout输入参数

mysql> delimiter $$
mysql> create procedure inout_param(inout p_inout int)->   begin->     select p_inout;->     set p_inout=2;->     select p_inout;->   end-> $$
mysql> delimiter ;mysql> set @p_inout=1;mysql> call inout_param(@p_inout);
+---------+
| p_inout |
+---------+
|       1 |
+---------++---------+
| p_inout |
+---------+
|       2 |
+---------+mysql> select @p_inout;
+----------+
| @p_inout |
+----------+
|        2 |
+----------+
#调用了inout_param存储过程,接受了输入的参数,也输出参数,改变了变量

注意:

1、如果过程没有参数,也必须在过程名后面写上小括号例:

CREATE PROCEDURE sp_name ([proc_parameter[,...]]) ……

2、确保参数的名字不等于列的名字,否则在过程体中,参数名被当做列名来处理

建议:

  • 输入值使用in参数。
  • 返回值使用out参数。
  • inout参数就尽量的少用。

三、变量

1. 变量定义

局部变量声明一定要放在存储过程体的开始:

DECLAREvariable_name [,variable_name...] datatype [DEFAULT value];

其中,datatype 为 MySQL 的数据类型,如: int, float, date,varchar(length)

例如:

DECLARE l_int int unsigned default 4000000;  
DECLARE l_numeric number(8,2) DEFAULT 9.95;  
DECLARE l_date date DEFAULT '1999-12-31';  
DECLARE l_datetime datetime DEFAULT '1999-12-31 23:59:59';  
DECLARE l_varchar varchar(255) DEFAULT 'This will not be padded';

2. 变量赋值

SET 变量名 = 表达式值 [,variable_name = expression ...]

3. 用户变量

在MySQL客户端使用用户变量:

mysql > SELECT 'Hello World' into @x;  
mysql > SELECT @x;  
+-------------+  
|   @x        |  
+-------------+  
| Hello World |  
+-------------+  
mysql > SET @y='Goodbye Cruel World';  
mysql > SELECT @y;  
+---------------------+  
|     @y              |  
+---------------------+  
| Goodbye Cruel World |  
+---------------------+  mysql > SET @z=1+2+3;  
mysql > SELECT @z;  
+------+  
| @z   |  
+------+  
|  6   |  
+------+

在存储过程中使用用户变量

mysql > CREATE PROCEDURE GreetWorld( ) SELECT CONCAT(@greeting,' World');  
mysql > SET @greeting='Hello';  
mysql > CALL GreetWorld( );  
+----------------------------+  
| CONCAT(@greeting,' World') |  
+----------------------------+  
|  Hello World               |  
+----------------------------+

在存储过程间传递全局范围的用户变量

mysql> CREATE PROCEDURE p1()   SET @last_procedure='p1';  
mysql> CREATE PROCEDURE p2() SELECT CONCAT('Last procedure was ',@last_procedure);  
mysql> CALL p1( );  
mysql> CALL p2( );  
+-----------------------------------------------+  
| CONCAT('Last procedure was ',@last_proc       |  
+-----------------------------------------------+  
| Last procedure was p1                         |  +-----------------------------------------------+

注意:

  • 1、用户变量名一般以@开头
  • 2、滥用用户变量会导致程序难以理解及管理

四、注释

MySQL 存储过程可使用两种风格的注释

两个横杆**–**:该风格一般用于单行注释。

c 风格: 一般用于多行注释。

例如:

mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc1 --name存储过程名  -> (IN parameter1 INTEGER)   -> BEGIN   -> DECLARE variable1 CHAR(10);   -> IF parameter1 = 17 THEN   -> SET variable1 = 'birds';   -> ELSE -> SET variable1 = 'beasts';   -> END IF;   -> INSERT INTO table1 VALUES (variable1);  -> END   -> //  
mysql > DELIMITER ;

MySQL存储过程的调用

用call和你过程名以及一个括号,括号里面根据需要,加入参数,参数包括输入参数、输出参数、输入输出参数。具体的调用方法可以参看上面的例子。

MySQL存储过程的查询

我们像知道一个数据库下面有那些表,我们一般采用 showtables; 进行查看。那么我们要查看某个数据库下面的存储过程,是否也可以采用呢?答案是,我们可以查看某个数据库下面的存储过程,但是是另一钟方式。

我们可以用以下语句进行查询:

selectname from mysql.proc where db='数据库名';或者selectroutine_name from information_schema.routines where routine_schema='数据库名';或者showprocedure status where db='数据库名';

如果我们想知道,某个存储过程的详细,那我们又该怎么做呢?是不是也可以像操作表一样用describe 表名进行查看呢?

**答案是:**我们可以查看存储过程的详细,但是需要用另一种方法:

SHOWCREATE PROCEDURE 数据库.存储过程名;

就可以查看当前存储过程的详细。

MySQL存储过程的修改

ALTER PROCEDURE

更改用 CREATE PROCEDURE 建立的预先指定的存储过程,其不会影响相关存储过程或存储功能。

MySQL存储过程的删除

删除一个存储过程比较简单,和删除表一样:

DROP PROCEDURE

从 MySQL 的表格中删除一个或多个存储过程。

MySQL存储过程的控制语句

(1). 变量作用域

内部的变量在其作用域范围内享有更高的优先权,当执行到 end。变量时,内部变量消失,此时已经在其作用域外,变量不再可见了,应为在存储过程外再也不能找到这个申明的变量,但是你可以通过 out 参数或者将其值指派给会话变量来保存其值。

mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc3()  -> begin -> declare x1 varchar(5) default 'outer';  -> begin -> declare x1 varchar(5) default 'inner';  -> select x1;  -> end;  -> select x1;  -> end;  -> //  
mysql > DELIMITER ;

(2). 条件语句

  1. if-then-else 语句
mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc2(IN parameter int)  -> begin -> declare var int;  -> set var=parameter+1;  -> if var=0 then -> insert into t values(17);  -> end if;  -> if parameter=0 then -> update t set s1=s1+1;  -> else -> update t set s1=s1+2;  -> end if;  -> end;  -> //  
mysql > DELIMITER ;
  1. case语句:
mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc3 (in parameter int)  -> begin -> declare var int;  -> set var=parameter+1;  -> case var  -> when 0 then   -> insert into t values(17);  -> when 1 then   -> insert into t values(18);  -> else   -> insert into t values(19);  -> end case;  -> end;  -> //  
mysql > DELIMITER ; 
casewhen var=0 theninsert into t values(30);when var>0 thenwhen var<0 thenelse
end case

(3). 循环语句

  1. while ···· end while
mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc4()  -> begin -> declare var int;  -> set var=0;  -> while var<6 do  -> insert into t values(var);  -> set var=var+1;  -> end while;  -> end;  -> //  
mysql > DELIMITER ;
while 条件 do--循环体
endwhile
  1. repeat···· end repea

它在执行操作后检查结果,而 while 则是执行前进行检查。

mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc5 ()  -> begin   -> declare v int;  -> set v=0;  -> repeat  -> insert into t values(v);  -> set v=v+1;  -> until v>=5  -> end repeat;  -> end;  -> //  
mysql > DELIMITER ;
repeat--循环体
until 循环条件  
end repeat;
  1. loop ·····endloop

loop 循环不需要初始条件,这点和 while 循环相似,同时和 repeat 循环一样不需要结束条件, leave 语句的意义是离开循环。

mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc6 ()  -> begin -> declare v int;  -> set v=0;  -> LOOP_LABLE:loop  -> insert into t values(v);  -> set v=v+1;  -> if v >=5 then -> leave LOOP_LABLE;  -> end if;  -> end loop;  -> end;  -> //  
mysql > DELIMITER ;
  1. LABLES 标号:

标号可以用在 begin repeat while 或者 loop 语句前,语句标号只能在合法的语句前面使用。可以跳出循环,使运行指令达到复合语句的最后一步。

(4). ITERATE迭代

ITERATE 通过引用复合语句的标号,来从新开始复合语句:

mysql > DELIMITER //  
mysql > CREATE PROCEDURE proc10 ()  -> begin -> declare v int;  -> set v=0;  -> LOOP_LABLE:loop  -> if v=3 then   -> set v=v+1;  -> ITERATE LOOP_LABLE;  -> end if;  -> insert into t values(v);  -> set v=v+1;  -> if v>=5 then -> leave LOOP_LABLE;  -> end if;  -> end loop;  -> end;  -> //  
mysql > DELIMITER ;

例子代码

CREATE TABLE player (`id` int(11) NOT NULL AUTO_INCREMENT,`playerName` varchar(20) DEFAULT NULL,`playerPosition` varchar(20) DEFAULT NULL,`playerNumber` int(11) DEFAULT NULL,`teamId` int(11) DEFAULT NULL,`championNumber` int(11) DEFAULT NULL,PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4SELECT * FROM tbl_userCREATE VIEW v_user AS SELECT * FROM tbl_user WHERE id =4CREATE VIEW v_user AS   SELECT count(*) FROM tbl_userDROP VIEW v_userCREATE VIEW v_user (id1,`name111`) AS SELECT id,`NAME` FROM tbl_userSELECT * FROM v_user WHERE id1 =4
UPDATE v_user SET `name`='majian' WHERE id1 =4DROP VIEW v_userSELECT `name` FROM tbl_user WHERE id=4-- 创建自定义函数
CREATE FUNCTION getuser()
RETURNS VARCHAR(20) -- 定义返回值的类型
RETURN -- 指定返回值 为后面的 select 语句结果
(SELECT `name` FROM tbl_user WHERE id=4);SELECT getuser(); -- 调用函数-- 创建带参数的函数  参数格式  参数名字 类型DELIMITER ;; -- 指定函数的结束标记为;; 默认是; ,但是我们的函数中可能会有多次;  为了防止误结束,所以先修改结束标记
CREATE FUNCTION   addPlayer(playerName VARCHAR(20) ,playerPosition
VARCHAR(20) ,playerNumber INT UNSIGNED ,teamId INT UNSIGNED
,championNumber  INT)RETURNS INT   UNSIGNED
BEGININSERT INTO player
VALUES(NULL,playerName,playerPosition,playerNumber,teamId,championNumber)
;RETURN LAST_INSERT_ID();
END  ;;
DELIMITER ;-- 调用带参数的函数
SELECT addPlayer('世纪佳缘','程序员的最终归宿',334,2423,235234)
-- 钱多,话少,死的早
-- 删除函数
DROP FUNCTION  IF EXISTS getuser-- 创建存储过程,无参数, 其实就是一个函数
DELIMITER ;;
CREATE PROCEDURE getuser()
BEGIN -- 下面是要执行的内容
SELECT `name` FROM tbl_user WHERE id=4;  
END ;;
-- 调用过程
CALL getuser();-- 定义带参数的过程,参数类型是输入参数
DELIMITER ;;
CREATE PROCEDURE   addPlayer(in playerName VARCHAR(20) ,in playerPosition
VARCHAR(20) ,in playerNumber INT UNSIGNED ,in teamId INT UNSIGNED
,in championNumber  INT)
BEGININSERT INTO player
VALUES(NULL,playerName,playerPosition,playerNumber,teamId,championNumber)
;SELECT LAST_INSERT_ID();
END  ;;
DELIMITER ;CALL addPlayer('content','几分伤心几分痴',321,312,3123)-- 创建带输出参数的过程,注意输出参数不能直接在函数内调用
CREATE PROCEDURE getuser1(out out_a int)
BEGIN
-- SELECT `name` FROM tbl_user WHERE id=4;
INSERT INTO player
VALUES(NULL,'dasdas','打扫打扫大厦',213,31231,3131);
SELECT LAST_INSERT_ID() INTO out_a; -- 赋值给输出参数
SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数
END ;;-- 声明变量
SET @adasdas=1;-- SELECT LAST_INSERT_ID() INTO @adasdas;
-- SELECT @adasdas;-- 盗用参数, 将声明的变量作为参数传递
CALL getuser1(@adasdas);-- 创建带输入/输出参数的过程,这个参数可以作为输入参数也可以作为输出参数
CREATE PROCEDURE getuser2(inout out_a int)
BEGIN
-- SELECT `name` FROM tbl_user WHERE id=4;
INSERT INTO player
VALUES(NULL,'dasdas','打扫打扫大厦',out_a,31231,3131); -- 取值
SELECT LAST_INSERT_ID() INTO out_a; -- 赋值
SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数
END ;;
-- 调用过程
SET @adasdas=999;
CALL getuser2(@adasdas);-- 创建带 if 条件的过程
CREATE PROCEDURE getuser3(inout out_a int)
BEGIN
-- SELECT `name` FROM tbl_user WHERE id=4;
IF out_a=0 THEN -- if 条件INSERT INTO playerVALUES(NULL,'dasdas','打扫打扫大厦',out_a,000000,000000000);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数
END IF;
IF out_a =1 THENINSERT INTO playerVALUES(NULL,'ewaqe','每次来上海',out_a,11111111,111111);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数
ELSEINSERT INTO playerVALUES(NULL,'dasdas','就要来台风',out_a,666666,666666);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数
END IF;
END ;;
-- 调用过程
SET @bianliang1=0;
CALL getuser3(@bianliang1)-- 创建带 switch case 的过程
CREATE PROCEDURE getuser4(inout out_a int)
BEGINCASE out_aWHEN 0 THENINSERT INTO playerVALUES(NULL,'xiangyao','汤臣一品',out_a,312310000,310031);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数WHEN 1 THENINSERT INTO playerVALUES(NULL,'zhuanche','地铁2号线',out_a,111,312111);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数WHEN 2 THENINSERT INTO playerVALUES(NULL,'duanzi',' 踩刹车,踩刹车,刹车是哪个',out_a,222,222);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数ELSE -- 默认操作INSERT INTO playerVALUES(NULL,'majian','马健有一个特殊爱好',out_a,333,333);SELECT LAST_INSERT_ID() INTO out_a;SELECT out_a; -- 因为默认是将 select 的结果当做返回值,所以我们必须手动 select 一下参数
END CASE;
END ;;SET @bianliang1=3;
CALL getuser4(@bianliang1);-- 常见循环操作的过程
CREATE PROCEDURE getuser5(inout out_a int)
BEGINDECLARE var int DEFAULT 0;WHILE var <out_a DOINSERT INTO playerVALUES(NULL,'majian','马健有一个特殊爱好',var,333,333);SET var =var+1;END WHILE;
END ;;
-- 执行操作
SET @cishu =10;
CALL getuser5(@cishu)

相关文章:

【MySQL】-10 MySQL 存储过程

MySQL 存储过程 优点缺点一、存储过程的创建和调用创建存储过程实例1、in 输入参数2、out输出参数3、inout输入参数 三、变量1. 变量定义2. 变量赋值3. 用户变量 四、注释MySQL存储过程的调用MySQL存储过程的查询MySQL存储过程的修改MySQL存储过程的删除MySQL存储过程的控制语句…...

3.闭包 - JS

作用域 一般认为 JS 中作用域有三种&#xff1a; 全局作用域&#xff1a;一个脚本运行代码的默认作用域&#xff1b;模块作用域&#xff1a;一个模块运行代码的默认作用域&#xff1b;函数作用域&#xff1a;一个函数运行代码的默认作用域。 而由于 let/const 声明变量的作用…...

Java实现批量视频抽帧2.0

继上个版本 对其进行略微升级 &#x1f913; 上个版本仅对一个视频进行抽帧处理 此版本可对一个文件夹内的全部视频进行抽帧并对应的文件夹进行帧图片的保存 1️⃣配置pom.xml &#xff08;保持上次不变&#xff09; <dependencies><dependency><grou…...

MFC 原生LsitCtrl单元格嵌入图标

// ListItemInsertIconDlg.h: 头文件 //#pragma once// CListItemInsertIconDlg 对话框 class CListItemInsertIconDlg : public CDialogEx { // 构造 public:CListItemInsertIconDlg(CWnd* pParent nullptr); // 标准构造函数// 对话框数据 #ifdef AFX_DESIGN_TIMEenum { IDD…...

黑马头条 - minio

我是南城余&#xff01;阿里云开发者平台专家博士证书获得者&#xff01; 欢迎关注我的博客&#xff01;一同成长&#xff01; 一名从事运维开发的worker&#xff0c;记录分享学习。 专注于AI&#xff0c;运维开发&#xff0c;windows Linux 系统领域的分享&#xff01; 知…...

认识Tomcat (一)

认识Tomcat &#xff08;一&#xff09; 一、服务器 1.1 服务器简介 ​ 硬件服务器的构成与一般的PC比较相似&#xff0c;但是服务器在稳定性、安全性、性能等方面都要求更高&#xff0c;因为CPU、芯片组、内存、磁盘系统、网络等硬件和普通PC有所不同。 ​ 软件服务器&…...

SSH免密切换服务器案例-ssh协议(公钥和私钥)

公钥和私钥理解 公钥提供加密&#xff0c;私钥解密&#xff0c;公钥可以共享&#xff0c;私钥不可以。举例公钥相当于锁头&#xff0c;可以给别人用&#xff0c;钥匙相当于私钥&#xff0c;只能开自己发出去的锁头&#xff0c;也就是私钥和公钥成对&#xff0c;私钥只能解密对…...

AMH面板如何安装与公网远程访问本地面板界面

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...

【教3妹学编程-算法题】1696. 跳跃游戏 VI

3妹&#xff1a;好冷啊&#xff0c; 冻得瑟瑟发抖啦 2哥 : 没想到都立春了还这么冷啊~ 3妹&#xff1a;暴雪、冻雨、大雨&#xff0c;这天气还让不让人活啦&#xff01;&#xff01;&#xff01; 2哥 :哎&#xff0c;好多人都滞留的高铁站了&#xff0c;没法回家了 3妹&#xf…...

解决C#中无限递归导致的System.StackOverflowException异常

目录 背景&#xff1a; 错误示例分析: 为什么是错误的&#xff1f; 正确的使用递归&#xff1a; 修改后的代码&#xff1a; 原理和原因&#xff1a; 结论&#xff1a; 背景&#xff1a; 在软件开发中&#xff0c;递归是一种常见的编程技术&#xff0c;它允许方法调用自…...

ASP.NET Core 预防开放式重定向攻击

写在前面 为预防钓鱼网站的常用套路&#xff0c;在进行 Web 应用程序的开发时&#xff0c;原则上应该将所有由用户提交的数据视为不可信。如果应用程序中包含了基于 URL 内容重定向的功能&#xff0c;需要确保这种类型的重定向操作只能在应用本地完成&#xff0c;或者明确判断…...

HashCat 恢复Excel、Word、PPT密码保姆教程

HashCat 恢复Excel、Word、PPT密码 一、流程 整体需要两个步骤 先用office2john.py获取下文件的hash值 python office2john.py 1.xlsx > hash这个命令需要你电脑有python环境&#xff0c;然后在cmd命令窗口中执行此命令就行 文件链接&#xff1a;https://github.com/magnu…...

flink实战--flink的job_listener使用解析

背景 生产环境可能有如下的需求:当一个flink作业提交完成或者是运行中不定时给我们触发某个接口或发送一个消息,然后我们在做其他的操作,尤其是batch作业。 flink的job_listener就可以满足我们监听flink任务提交和运行状态的需求,具体如何使用本文将全面介绍一下。 注册入…...

ASR 概述

前言 随着企业加强了与客户的线上沟通&#xff0c;企业越发依赖于虚拟助手、聊天机器人以及其他的语音技术&#xff0c;以实现与客户的高效互动。这几类人工智能&#xff0c;都是依赖于自动语音识别技术&#xff0c;简称为 ASR。ASR 涉及到将语音转换为文本&#xff0c;促使计…...

聊聊比特币----比特币地址

⽐特币地址是⼀个标识符&#xff08;帐号&#xff09;&#xff0c;包含27-34个字母数字拉丁字符&#xff08;0&#xff0c;O&#xff0c;I除外&#xff09;。地址可以以QR码形式表⽰&#xff0c;是匿名的&#xff0c;不包含关于所有者的信息。 地址⽰例&#xff1a;14qViLJfdG…...

(4)【Python数据分析进阶】Machine-Learning模型与算法应用-回归、分类模型汇总

线性回归、逻辑回归算法应用请参考: https://codeknight.blog.csdn.net/article/details/135693621https://codeknight.blog.csdn.net/article/details/135693621本篇主要介绍决策树、随机森林、KNN、SVM、Bayes等有监督算法以及无监督的聚类算法和应用PCA对数据进行降维的算法…...

Python 调用 OpenAI ChatGPT API

一、安装环境1. 安装python环境 $ pip install openai 2. 验证是否安装成功 方法1&#xff0c;bash命令验证 $ pip show openai 方法2&#xff0c;python脚本验证 import openai print(openai.__version__) 3. 找到你的 OpenAI API Key&#xff1a;进入OpenAI官网&#xff0…...

springboot155基于JAVA语言的在线考试与学习交流网页平台

简介 【毕设源码推荐 javaweb 项目】基于springbootvue 的 适用于计算机类毕业设计&#xff0c;课程设计参考与学习用途。仅供学习参考&#xff0c; 不得用于商业或者非法用途&#xff0c;否则&#xff0c;一切后果请用户自负。 看运行截图看 第五章 第四章 获取资料方式 **项…...

echarts使用之地图(五)

1 基本使用 百度地图 API : 使用百度地图的 api , 它能够在线联网展示地图 , 百度地图需要申请 ak 矢量地图 : 可以离线展示地图 , 需要开发者准备矢量地图数据。本文使用该方式。 json格式的数据如下&#xff1a; 格式参照&#xff1a;GeoJSON <!DOCTYPE html&…...

【已解决】青龙面板依赖安装失败原因

青龙面板必须安装依赖&#xff0c;才可以执行脚本&#xff0c;这是不争的事实。 如果脚本跑不起来&#xff0c;就去看看依赖吧。 NodeJs 依赖如下 axios request canvas cheerio js-base64 dotenv magic tough-cookie ws7.4.3 require requests date-fns ts-md5 typescript j…...

Java 8 Stream API 入门到实践详解

一、告别 for 循环&#xff01; 传统痛点&#xff1a; Java 8 之前&#xff0c;集合操作离不开冗长的 for 循环和匿名类。例如&#xff0c;过滤列表中的偶数&#xff1a; List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...

DAY 47

三、通道注意力 3.1 通道注意力的定义 # 新增&#xff1a;通道注意力模块&#xff08;SE模块&#xff09; class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...

【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)

可以使用Sqliteviz这个网站免费编写sql语句&#xff0c;它能够让用户直接在浏览器内练习SQL的语法&#xff0c;不需要安装任何软件。 链接如下&#xff1a; sqliteviz 注意&#xff1a; 在转写SQL语法时&#xff0c;关键字之间有一个特定的顺序&#xff0c;这个顺序会影响到…...

Nginx server_name 配置说明

Nginx 是一个高性能的反向代理和负载均衡服务器&#xff0c;其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机&#xff08;Virtual Host&#xff09;。 1. 简介 Nginx 使用 server_name 指令来确定…...

令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍

文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结&#xff1a; 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析&#xff1a; 实际业务去理解体会统一注…...

【Zephyr 系列 10】实战项目:打造一个蓝牙传感器终端 + 网关系统(完整架构与全栈实现)

🧠关键词:Zephyr、BLE、终端、网关、广播、连接、传感器、数据采集、低功耗、系统集成 📌目标读者:希望基于 Zephyr 构建 BLE 系统架构、实现终端与网关协作、具备产品交付能力的开发者 📊篇幅字数:约 5200 字 ✨ 项目总览 在物联网实际项目中,**“终端 + 网关”**是…...

C# SqlSugar:依赖注入与仓储模式实践

C# SqlSugar&#xff1a;依赖注入与仓储模式实践 在 C# 的应用开发中&#xff0c;数据库操作是必不可少的环节。为了让数据访问层更加简洁、高效且易于维护&#xff0c;许多开发者会选择成熟的 ORM&#xff08;对象关系映射&#xff09;框架&#xff0c;SqlSugar 就是其中备受…...

今日科技热点速览

&#x1f525; 今日科技热点速览 &#x1f3ae; 任天堂Switch 2 正式发售 任天堂新一代游戏主机 Switch 2 今日正式上线发售&#xff0c;主打更强图形性能与沉浸式体验&#xff0c;支持多模态交互&#xff0c;受到全球玩家热捧 。 &#x1f916; 人工智能持续突破 DeepSeek-R1&…...

Java多线程实现之Thread类深度解析

Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...

Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决

Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决 问题背景 在一个基于 Spring Cloud Gateway WebFlux 构建的微服务项目中&#xff0c;新增了一个本地验证码接口 /code&#xff0c;使用函数式路由&#xff08;RouterFunction&#xff09;和 Hutool 的 Circle…...