jjzjj

mysql - 将 MySQL 键列转换为 FOREIGN 列后,站点速度变慢

coder 2023-10-09 原文

我以前从未使用过 FOREIGN KEYS。我总是将 id 放在一个键列中,但它不一定是外键。所以它就像一个外键,但它不是。

所以如果我有以下两个表

CREATE TABLE `accounts` (
 `account_id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `name` varchar(60) NOT NULL,
 `owner_id` int(11) unsigned NOT NULL,
 PRIMARY KEY (`account_id`),
 KEY `owner_id` (`owner_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8


CREATE TABLE `users` (
 `user_id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `name` varchar(60) NOT NULL
 PRIMARY KEY (`user_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

account.owner_id 列链接到 users.user_id 但没有设置外键关系。

所以我可以做这样的事情

SELECT
a.name AS account_name,
u.name AS user_name FROM
accounts AS a INNER JOIN users AS u ON u.user_id = a.owner_id

所以在了解了外键是什么以及它们的作用之后,我创建了一个像这样的外键。

ALTER TABLE accounts 
ADD CONSTRAINT fk_owner_id FOREIGN KEY(owner_id)
REFERENCES users(user_id) ON DELETE NO ACTION ON UPDATE CASCADE;

但是,我注意到性能发生了变化,因为它们的系统速度变慢了很多。我不确定添加外键是否会重用性能?或者在向表中添加外键时性能会降低吗? 请注意,我的数据库中有很多表和列。所以我做了很多 INNER/LEFT/RIGHT JOINS 并且我将它们标识为 FOREIGN KEYS 的所有列都被索引了。请注意,我没有向该列添加任何索引,因为在将外键添加到我的数据库之前它们已经存在。

我的问题是,FOREIGN KEY 会降低 UPDATE/INSERT/DELETE/SELECT 的性能吗? 当我们指定 ON DELETE NO ACTION ON UPDATE NO ACTION 时,添加 FOREIGN KEY 约束是否也有好处?

谢谢

最佳答案

外键约束意味着对FK列的任何插入都必须检查该值是否存在于用户引用的列中。这可能会有一些开销,但根据定义它是索引查找(可能是 PK 查找),因此成本应该不会很高。

在对子表进行某些更新期间,外键还会在父表上创建共享锁。这会妨碍对该表的并发更新,并使系统看起来性能较慢。参见 http://www.mysqlperformanceblog.com/2006/12/12/innodb-locking-and-foreign-keys/

如果索引不存在,外键还会在 FK 列上隐式创建索引。每次insert、update、delete都要在改变的时候修改一张表的所有索引,所以会有一点开销。出于这个原因,有人说索引“损害”了插入、更新、删除的性能。但这并没有那么简单——支持 WHERE 子句中的条件的索引可以通过更有效地查找受影响的行来加快更新或删除的运行速度。


回复你的评论:

是的,对子表的任何更新都会在子表中的修改行上创建一个独占锁。你可能预料到了这一点。但是此操作还会在 clients 中的引用行上创建一个共享锁。任意数量的 session 都可以在同一行上独立创建共享锁(因此称为共享)。但是独占锁要求没有任何类型的锁。因此,如果 clients 中的某行经常存在未完成的共享锁,则直接更新 clients 中的该行将无法获得所需的独占锁。

这些共享锁的目的基本上是当有人更新依赖于它的行时,clients 中的行不会被删除或修改。换句话说,“不要删除我的 parent ”。但根据更新频率和事务持续时间,可能很难对父行执行更新。

缓解这种情况的一种方法是尝试使锁的存活时间更短,方法是立即完成事务的工作,然后提交。

关于mysql - 将 MySQL 键列转换为 FOREIGN 列后,站点速度变慢,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17983591/

有关mysql - 将 MySQL 键列转换为 FOREIGN 列后,站点速度变慢的更多相关文章

  1. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  2. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

  3. ruby - 将数组的内容转换为 int - 2

    我需要读入一个包含数字列表的文件。此代码读取文件并将其放入二维数组中。现在我需要获取数组中所有数字的平均值,但我需要将数组的内容更改为int。有什么想法可以将to_i方法放在哪里吗?ClassTerraindefinitializefile_name@input=IO.readlines(file_name)#readinfile@size=@input[0].to_i@land=[@size]x=1whilex 最佳答案 只需将数组映射为整数:@land边注如果你想得到一条线的平均值,你可以这样做:values=@input[x]

  4. ruby - 将散列转换为嵌套散列 - 2

    这道题是thisquestion的逆题.给定一个散列,每个键都有一个数组,例如{[:a,:b,:c]=>1,[:a,:b,:d]=>2,[:a,:e]=>3,[:f]=>4,}将其转换为嵌套哈希的最佳方法是什么{:a=>{:b=>{:c=>1,:d=>2},:e=>3,},:f=>4,} 最佳答案 这是一个迭代的解决方案,递归的解决方案留给读者作为练习:defconvert(h={})ret={}h.eachdo|k,v|node=retk[0..-2].each{|x|node[x]||={};node=node[x]}node[

  5. ruby-on-rails - Ruby url 到 html 链接转换 - 2

    我正在使用Rails构建一个简单的聊天应用程序。当用户输入url时,我希望将其输出为html链接(即“url”)。我想知道在Ruby中是否有任何库或众所周知的方法可以做到这一点。如果没有,我有一些不错的正则表达式示例代码可以使用... 最佳答案 查看auto_linkRails提供的辅助方法。这会将所有URL和电子邮件地址变成可点击的链接(htmlanchor标记)。这是文档中的代码示例。auto_link("Gotohttp://www.rubyonrails.organdsayhellotodavid@loudthinking.

  6. ruby-on-rails - 使用 ruby​​ 将多个实例变量转换为散列的更好方法? - 2

    我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作:response.instance_variables.eachdo|r|my_hash.merge!(r.to_s.delete("@").intern=>response.instance_eval(r.to_s.delete("@")))end这有效,它将生成{:first="charlie",:last=>"kelly"},但它似乎有点hacky和不稳定。有更好的方法吗?编辑:我刚刚意识到我可以使用instance_variable_get作为该等式的第二部分,但这仍然是主要问题。

  7. 使用canal同步MySQL数据到ES - 2

    文章目录一、概述简介原理模块二、配置Mysql使用版本环境要求1.操作系统2.mysql要求三、配置canal-server离线下载在线下载上传解压修改配置单机配置集群配置分库分表配置1.修改全局配置2.实例配置垂直分库水平分库3.修改group-instance.xml4.启动监听四、配置canal-adapter1修改启动配置2配置映射文件3启动ES数据同步查询所有订阅同步数据同步开关启动4.验证五、配置canal-admin一、概述简介canal是Alibaba旗下的一款开源项目,Java开发。基于数据库增量日志解析,提供增量数据订阅&消费。Git地址:https://github.co

  8. python ffmpeg 使用 pyav 转换 一组图像 到 视频 - 2

    2022/8/4更新支持加入水印水印必须包含透明图像,并且水印图像大小要等于原图像的大小pythonconvert_image_to_video.py-f30-mwatermark.pngim_dirout.mkv2022/6/21更新让命令行参数更加易用新的命令行使用方法pythonconvert_image_to_video.py-f30im_dirout.mkvFFMPEG命令行转换一组JPG图像到视频时,是将这组图像视为MJPG流。我需要转换一组PNG图像到视频,FFMPEG就不认了。pyav内置了ffmpeg库,不需要系统带有ffmpeg工具因此我使用ffmpeg的python包装p

  9. ruby-on-rails - 将字符串转换为 ruby​​-on-rails 中的函数 - 2

    我需要一个通过输入字符串进行计算的方法,像这样function="(a/b)*100"a=25b=50function.something>>50有什么方法吗? 最佳答案 您可以使用instance_eval:function="(a/b)*100"a=25.0b=50instance_evalfunction#=>50.0请注意,使用eval本质上是不安全的,尤其是当您使用外部输入时,因为它可能包含注入(inject)的恶意代码。另请注意,a设置为25.0而不是25,因为如果它是整数a/b将导致0(整数)。

  10. ruby-on-rails - 将 ruby​​ 数组转换为整齐的列字符串? - 2

    我是ruby​​的新手,我正在尝试制作一个程序来自动格式化给定的字符串和数组。我试图弄清楚的一种自动格式化功能是一种用于数组的功能。假设我有一个如下例所示的数组myArray=["a","b","c"]我想把它变成一个列化的字符串,这样putsmyString就会给出`1)a``2)b``3)c`我该怎么做呢?我能找到的最接近的东西是使用.each这不是我想要的,我不能让每一行都有一个单独的条目。这一切都必须是一个带有换行符的字符串。任何帮助将不胜感激,提前致谢 最佳答案 您可以使用.map与.with_index:myArray=

随机推荐