数据库编程 MySQL 技巧与经验

CuterCorley 等级 832 0 0

1.MySQL创建数据表时设定引擎并添加外键约束

创建两个数据表,在它们之间添加外键约束,然后在被添加外键的表中添加数据,发现并没有提示报错,很正常地插入了数据,说明外键没有添加成功,在SQL可视化工具里查看表的属性,并点击外部键会出现弹窗,提示是因为引擎的问题,导致不能添加外键。MySQL安装时默认用的表引擎是MyISAM,而MyISAM是不支持外键的,如图, 数据库编程 MySQL 技巧与经验 要想解决这个问题,可以在当前的表设置引擎为InnoDB、PBXT或SolidDB,但这只是修改了这一个数据库,下次建新的数据库默认引擎还是MyISAM,我们可以在MySQL的安装目录下的配置文件my.ini中的 [mysqld] 下面加入default-storage-engine=INNODB(其他支持外键的引擎也可)保存,再重启Mysql服务器即可,小编在这块也遇到了问题,很久都没能解决,最后请教老师成功解决了。 可参考https://blog.csdn.net/renwudao24/article/details/52136955

以更改 Mysql 默认引擎为 InnoDB为例步骤如下:

  • 查看Mysql存储引擎情况: mysql>show engines,结果是: InnoDB | YES,说明此Mysql数据库服务器支持InnoDB引擎;
  • 设置InnoDB为默认引擎:在配置文件my.ini中的 [mysqld] 下面加入default-storage-engine=INNODB;
  • 重启Mysql服务器;
  • 登录Mysql数据库,mysql>show engines 如果出现 InnoDB |DEFAULT,则表示设置InnoDB为默认引擎成功。

    2.delimiter 详解

    作用是告诉mysql解释器,该段命令是否已经结束了,mysql是否可以执行了。默认情况下,delimiter是分号 ;
DELIMITER $$ 
DROP TRIGGER IF EXISTS `updateegopriceondelete`$$ 
CREATE 
    TRIGGER `updateegopriceondelete` AFTER  DELETE ON  `customerinfo` 
    FOR EACH ROW BEGIN 
DELETE FROM egoprice  WHERE customerId=OLD.customerId; 
    END$$ 
DELIMITER ; 

其中DELIMITER 定好结束符为"$$", 然后最后又定义为";", MYSQL的默认结束符为";". 在命令行客户端中,如果有一行命令以分号结束,那么回车后,mysql将会执行该命令。如输入下面的语句 select * from test_table; 然后回车,那么MySQL将立即执行该语句。 但有时候,不希望MySQL这么做。在为可能输入较多的语句,且语句中包含有分号。如试图在命令行客户端中创建函数或存储过程时

mysql> CREATE FUNCTION `SHORTEN`(S VARCHAR(255), N INT) 
mysql>     RETURNS varchar(255) 
mysql> BEGIN 
mysql> IF ISNULL(S) THEN 
mysql>     RETURN ''; 
mysql> ELSEIF N<15 THEN 
mysql>     RETURN LEFT(S, N); 
mysql> ELSE 
mysql>     IF CHAR_LENGTH(S) <=N THEN 
mysql>    RETURN S; 
mysql>     ELSE 
mysql>    RETURN CONCAT(LEFT(S, N-10), '...', RIGHT(S, 5)); 
mysql>     END IF; 
mysql> END IF; 
mysql> END; 

默认情况下,不可能等到用户把这些语句全部输入完之后,再执行整段语句。 因为mysql一遇到分号,它就要自动执行。 即,在语句RETURN '';时,mysql解释器就要执行了。 这种情况下,就需要事先把delimiter换成其它符号,如//或$$。

mysql> delimiter // 
mysql> CREATE FUNCTION `SHORTEN`(S VARCHAR(255), N INT) 
mysql>     RETURNS varchar(255) 
mysql> BEGIN 
mysql> IF ISNULL(S) THEN 
mysql>     RETURN ''; 
mysql> ELSEIF N<15 THEN 
mysql>     RETURN LEFT(S, N); 
mysql> ELSE 
mysql>     IF CHAR_LENGTH(S) <=N THEN 
mysql>    RETURN S; 
mysql>     ELSE 
mysql>    RETURN CONCAT(LEFT(S, N-10), '...', RIGHT(S, 5)); 
mysql>     END IF; 
mysql> END IF; 
mysql> END;// 

这样只有当//出现之后,mysql解释器才会执行这段语句。 默认情况下,delimiter “;” 用于向 MySQL 提交查询语句。在存储过程中每个 SQL 语句的结尾都有个 “;”,如果这时候,每逢 “;” 就向 MySQL 提交的话,当然会出问题了。于是更改 MySQL 的 delimiter:

delimiter //;     -- 改变 MySQL delimiter 为:“//” 

drop procedure if exists pr_stat_agent // 

-- call pr_stat_agent ('2008-07-17', '2008-07-18') 

create procedure pr_stat_agent 
( 
   pi_date_from  date 
  ,pi_date_to    date 
) 
begin 
   -- check input 
   if (pi_date_from is null) then 
      set pi_date_from = current_date(); 
   end if; 

   if (pi_date_to is null) then 
      set pi_date_to = pi_date_from; 
   end if; 

   set pi_date_to = date_add(pi_date_from, interval 1 day); 

   -- stat 
   select agent, count(*) as cnt 
     from apache_log 
    where request_time >= pi_date_from 
      and request_time <  pi_date_to 
    group by agent 
    order by cnt desc; 
end; // 

delimiter ; //   -- 改回默认的 MySQL delimiter:“;” 

mysql> delimiter //     -- 末尾不要符号 “;” 
mysql> 
mysql> drop procedure if exists pr_stat_agent // 
Query OK, 0 rows affected (0.00 sec) 

mysql> 
mysql> -- call pr_stat_agent ('2008-07-17', '2008-07-18') 
mysql> 
mysql> create procedure pr_stat_agent 
    -> ( 
    ->    pi_date_from  date 
    ->   ,pi_date_to    date 
    -> ) 
    -> begin 
    ->    -- check input 
    ->    if (pi_date_from is null) then 
    ->       set pi_date_from = current_date(); 
    ->    end if; 
    -> 
    ->    if (pi_date_to is null) then 
    ->       set pi_date_to = pi_date_from; 
    ->    end if; 
    -> 
    ->    set pi_date_to = date_add(pi_date_from, interval 1 day); 
    -> 
    ->    -- stat 
    ->    select agent, count(*) as cnt 
    ->      from apache_log 
    ->     where request_time >= pi_date_from 
    ->       and request_time <  pi_date_to 
    ->     group by agent 
    ->     order by cnt desc; 
    -> end; // 
Query OK, 0 rows affected (0.00 sec) 

mysql> 
mysql> delimiter ;  -- 末尾不要符号 “//” 
mysql> 

3.MySQL清空表之后设置id从1开始自增

我们都知道,在手动清空一个表中的数据后,再添加数据id会从之前数据的id开始自增而不会从1开始自增,这在有时候会显得不友好,因此可以通过执行命令使之从1开始自增,有两种情况:

  • 数据已清空 在数据已清空的情况下可以执行alter table table_name auto_increment=1;或者truncate table_name;来使再次添加数据时id从1开始自增。
  • 数据为清空 数据为清空,而要使清空后数据可以从1开始自增,可以直接执行命令truncate table_name;即可,因为truncate语句本身就是用来删除、截断表里的所有数据,并且相比于delete语句性能更好,还能使id从1开始自增。

    4.MySQL删除存在外键约束的数据

    在删除具有外键约束的数据时,很容易报错:
ERROR 1217 (23000): Cannot delete or update a parent row: a foreign key constraint fails

这是因为存在外键关系时,不同的表之间或表内部存在字段之间的约束,导致不能删除,要想强制删除数据,可以通过执行命令SET FOREIGN_KEY_CHECKS = 0;临时解除数据间的约束,然后再删除要删除的表数据。 在删除数据之后,可以通过执行命令SET FOREIGN_KEY_CHECKS = 1;来还原约束(当然,也可以不执行这一步,因为这个设置是基于Session的,当你关闭了你所使用的客户端之后,下次再重新建立连接的时候,变量会恢复默认值)。

本文原文首发来自博客专栏数据库编程,由本人转发至https://www.helloworld.net/p/2QLT76hvgFAD,其他平台均属侵权,可点击https://blog.csdn.net/CUFEECR/article/details/103430520查看原文,也可点击https://blog.csdn.net/CUFEECR浏览更多优质原创内容。

收藏
评论区

相关推荐

OMG!Java高级开发岗必问知识点
目录 1.Mysql 2.CHAR 与 VARCHAR 的区别? 3.能说下myisam 和 innodb的区别吗? 4.你能说下事务的基本特性和隔离级别吗? 5.并发问题 脏读、不可重复读、幻读? 6.事务的隔离级别? 7.说说自增主键、UUID? 8.mysql 的约束分类? 9.drop、delete 与 tru
A2
 转载自:[http://www.mysqltutorial.org/understand-mysql-table-types-innodb-myisam.aspx](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fwww.mysqltutorial.org%2Funderstand-mysq
MySQL InnoDB集群
作者:Miguel Araújo 译:徐轶韬 MySQL开发团队发布InnoDB集群的8.0维护版本 - 8.0.17! 除了重要的错误修复和改进之外,8.0.17还带来了改变游戏规则的功能! 这篇博文将涵盖MySQL Shell和AdminAPI,如需详细了解MySQL路由器的新功能,请关注即将发布的博客文章! MySQL Shell Admi
MySQL —— 存储引擎的 InnoDB 与 MyISAM 之争
作为 MySQL 数据库的两种主要的存储引擎,InnoDB 与 MyISAM 各有长处。 在 MySQL 5.1 及之前的版本中,MyISAM 是默认的存储引擎,而在 MySQL 5.5 版本以后,默认使用 InnoDB 存储引擎。 MyISAM 不支持行级锁,换句话说,MyISAM 会对整张表加锁,而不是针对行。同时,MyISAM 不支持事务和外键。M
MySQL在INSERT IGNORE未新增记录时避免AUTO_INCREMENT自增
在MySQL5.7中做INSERT IGNORE时发现, 即使INSERT未成功执行, 表的自增主键却自动加1了, 在某些情况下需要避免这种行为. 需要修改的变量是 innodb\_autoinc\_lock\_mode, 将其设为0后, 在INSERT未成功执行时不会自增主键. **innodb\_autoinc\_lock\_mode在MySQL各版本
MySQL字段约束及多表查询
**前言:mysql的字段约束是以后必不可免的,下面主要写了四个:主键约束用于唯一且不能为空;非空约束即不能为空可以重复;唯一约束即可以为空但必须唯一;外键约束是让表与表之间有一定的关联;当然如何使用还看下文,多表就不在这总结了。如果你对前面的知识有所遗忘或感兴趣----[MySQL数据库表的模糊/多行/分组/排序/分页查询以及字mysql数据类型的讲解--
MySQL数据库的卸载与安装
MySQL数据库的卸载与安装 ============== MySQL的完全卸载 ---------- 因为不知道什么原因,电脑里同时存在两个版本的mysql,所以决定卸载重新安装,但是大家都说MySQL很难清除干净,所以特地查找完全卸载MySQL的方法。 首先,快捷键win+r输入regedit进入注册表,找到HKEY\_LOCAL\_MACHINE
MySQL数据表类型
MySQL一共向用户提供了包括DBD、HEAP、ISAM、MERGE、MyIASM、InnoDB以及Gemeni这7种Mysql表类型。其中DBD、InnoDB属于事务安全类表,而其他属于事务非安全类表。 ①   Berkeley DB(DBD)表是支持事务处理的表,由Sleepycat软件公司开发。它提供MySQL用户期待已久的功能--事务控制。事务控制
MySQL系列(一)
**一、简介** -------- MySQL是最流行的开放源码SQL数据库管理系统,它是由MySQL AB公司开发、发布并支持的。有以下特点: MySQL是一种数据库管理系统。 MySQL是一种关联数据库管理系统。 MySQL软件是一种开放源码软件。 MySQL数据库服务器具有快速、可靠和易于使用的特点。 MySQL服务器工作在客户端/服务器模
Mysql技术总结总结
**一:mysql数据库引擎MyISAM和InnoDB的区别:** ================================= 1、MyISAM 具有检查和修复表格的大多数工具。表格可以被压缩,而且支持全文收索 。不支持事物,而且不支持外键。 2、innodb 这种表是事务安全的。提供了commit(提交) rollback(实务回滚)支持外键,
mysql更改表引擎INNODB为MyISAM的方法总结
常见的mysql表引擎有INNODB和MyISAM,主要的区别是INNODB适合频繁写[数据库](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fwww.111cn.net%2Fdatabase%2Fdatabase.html)操作,MyISAM适合读取数据库的情况多一点,如何把表引擎INNO
mysql经典面试题
1、MySQL的复制原理以及流程 基本原理流程,3个线程以及之间的关联; 2、MySQL中myisam与innodb的区别,至少5点 (1)、问5点不同; (2)、innodb引擎的4大特性 (3)、2者selectcount(\*)哪个更快,为什么 3、MySQL中varchar与char的区别以及varchar(50)中的50代表的涵义 (
mysql面试问题
来源:http://bbs.51cto.com/thread-1470880-1.html 1、MySQL的复制原理以及流程 基本原理流程,3个线程以及之间的关联; 2、MySQL中myisam与innodb的区别,至少5点 (1)、问5点不同; (2)、innodb引擎的4大特性 (3)、2者selectcount(\*)哪个更快,为什么 3
mysql面试题及答案
01\. 列举常见的关系型数据库和非关系型都有那些? 1.关系型数据库通过外键关联来建立表与表之间的关系,---------常见的有:SQLite、Oracle、mysql 2.非关系型数据库通常指数据以对象的形式存储在数据库中,而对象之间的关系通过每个对象自身的属性来决定 ---常见的有:MongoDb、redis 02\. MySQL常见数据库引擎
52.更改root密码 连接mysql mysql命令
13.1 设置更改root密码 13.2 连接mysql 13.3 mysql常用命令 扩展  mysql5.7 root密码更改 http://www.apelearn.com/bbs/thread-7289-1-1.html myisam 和innodb引擎对比 http://www.pureweber.com/article/myisam-v