
今天我们来介绍另一种场景:MySQL 客户端给服务端发送一条 SQL 之后,服务端执行 SQL 的过程中,客户端没有任何通知,就直接断开了连接。
这种情况下,服务端会怎么办?
本文内容基于 MySQL 8.0.32 源码,涉及存储引擎为 InnoDB。
为了和前一篇文章介绍的场景区分开,我们用两个虚构小故事把两种场景放在一起作个对比。
张三(MySQL 客户端)和李四(服务端)是好朋友,它送给了李四一个礼物(发送了一条 DML/DDL SQL)。
有一天,张三和李四闹别扭,它后悔送礼物给李四了,于是它对李四说:把我送你的礼物还给我(Ctrl + C 要求服务端中断 SQL 执行)。
李四要先把张三送给它的礼物找出来,才能还给张三。
如果礼物还在(事务还没有提交),李四就能把礼物还给张三(中断执行,回滚事务);如果礼物不在了(事务已经提交了),也就没法还了。
张三(MySQL 客户端)和李四(服务端)是好朋友,它送给了李四一个礼物(发送了一条 DML/DDL SQL)。
有一天,李四因为一件事情把张三惹毛了。
张三心里很不爽,它要跟李四绝交(直接断开了连接),但对它而言,送出去的礼物就是泼出去的水,它不想收回。
李四的性格大大咧咧,它不知道自己把张三惹毛了,还在美滋滋的欣赏张三送给它的礼物(执行 SQL)。
等它回过头来想找张三的时候,发现找不着了,它才回想起来,可能自己把张三惹毛了,朋友没得做了。
此时,李四要怎么对待张三送给它的礼物呢?
接下来,我们跳出虚构,回归现实,来捋一下场景二的流程。
这种场景只会出现在通过程序连接 MySQL 服务端,程序没有关闭数据库连接就执行结束或者崩溃了
MySQL 客户端发送一条 DML/DDL SQL 给服务端,服务端收到之后,就开始吭哧吭哧地执行。
SQL 执行完成之前,客户端再没有给服务端发送任何消息,就直接断开连接了。
SQL 执行过程中,服务端并不能感知到连接已经断开了,它还会一直卖力地执行 SQL。
SQL 执行完成之后,问题来了。
如果服务端执行的是 DDL 语句,一条 SQL 执行完成之后,会自动提交事务。
如果服务端执行的是 DML 语句,并且系统变量 auto_commit = on,一条 SQL 执行完成之后,也会自动提交事务。
因为服务端不知道客户端已经断开连接了,事务提交之后,它会把 SQL 执行结果发送给客户端。
把结果发送给客户端,执行的是 send() 系统调用,send() 有两种行为:
行为一,如果 send() 把 SQL 执行结果写入 socket 缓冲区,会返回写入成功,此时,服务端还不会感知到客户端已经断开连接。
send() 执行成功之后,服务端认为这条 SQL 就告一段落了,会等待客户端发送下一条命令。
由于客户端已经断开连接,从当前连接读取下一条命令时会出错。
此时,服务端会感知到客户端已经断开连接了。
行为二,如果 send() 直接把 SQL 执行结果发送给客户端,服务端就能马上感知到客户端已经断开连接了。
不管 send() 发生哪种行为,服务端都会感知到客户端已经断开连接了,无非早一点晚一点而已。
感知到连接断开之后,服务端会回滚事务。
但是,由于执行 send() 之前,服务端已经把事务提交了,这里回滚事务并不会生效。
那么,最终结果就是:服务端对于数据的修改会被持久化,永久生效。
如果服务端执行的是 DML 语句,并且用 begin 或 start transaction 显式开启了事务,一条 SQL 执行完成之后,不会自动提交事务,而是会等待客户端发送下一条命令。
读取下一条命令之前,服务端会执行 send() 系统调用,把当前 SQL 的执行结果发送给客户端。
行为一,如果 send() 把 SQL 执行结果写入 socket 缓冲区,会返回写入成功,此时,服务端还不会感知到客户端已经断开连接。
send() 执行成功之后,服务端认为这条 SQL 就告一段落了,会等待客户端发送下一条命令。
由于客户端已经断开连接,从当前连接读取下一条命令时会出错。
此时,服务端会感知到客户端已经断开连接了。
行为二,如果 send() 直接把 SQL 执行结果发送给客户端,服务端就能马上感知到客户端已经断开连接了。
不管 send() 发生哪种行为,服务端都会感知到客户端已经断开连接了,无非早一点晚一点而已。
感知到连接断开之后,服务端会回滚事务,由于执行 send() 之前,并没有提交过事务,这里回滚事务会生效。
那么,最终结果就是:服务端对于数据的修改会被回滚,相当于没有执行过 DML 操作。
前面展开介绍了 MySQL 客户端不辞而别之后,服务端进行的一系列操作,总结起来就 3 条:
第 1 条:如果服务端执行的是 DDL 语句,DDL 会执行成功。
第 2 条:如果服务端执行的是 DML 语句,并且系统变量 auto_commit = on,DML 也会执行成功。
第 3 条:如果服务端执行的是 DML 语句,并且用 begin、start transaction 显式开启了事务或者系统变量 auto_commit = off,事务会被回滚,DML 相当于没有执行。
前面介绍到 send() 有两种行为:
这两种行为由操作系统内核决定,目前对于这个机制还没有完全研究清楚,留待后续。
本文转载自微信公众号「一树一溪」,可以通过以下二维码关注。转载本文请联系一树一溪公众号。

我在我的Rails3示例应用程序上使用CarrierWave。我想验证远程位置上传,因此当用户提交无效URL(空白或非图像)时,我不会收到标准错误异常:CarrierWave::DownloadErrorinImageController#createtryingtodownloadafilewhichisnotservedoverHTTP这是我的模型:classPaintingtrue,:length=>{:minimum=>5,:maximum=>100}validates:image,:presence=>trueend这是我的Controller:classPaintingsC
文章目录一、概述简介原理模块二、配置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
我看到其他人也遇到过类似的问题,但没有一个解决方案对我有用。0.3.14gem与其他gem文件一起存在。我已经完全按照此处指示完成了所有操作:https://github.com/brianmario/mysql2.我仍然得到以下信息。我不知道为什么安装程序指示它找不到include目录,因为我已经检查过它存在。thread.h文件存在,但不在ruby目录中。相反,它在这里:C:\RailsInstaller\DevKit\lib\perl5\5.8\msys\CORE\我正在运行Windows7并尝试在Aptana3中构建我的Rails项目。我的Ruby是1.9.3。$gemin
我已经开始使用mysql2gem。我试图弄清楚一些基本的事情——其中之一是如何明确地执行事务(对于批处理操作,比如多个INSERT/UPDATE查询)。在旧的ruby-mysql中,这是我的方法:client=Mysql.real_connect(...)inserts=["INSERTINTO...","UPDATE..WHEREid=..",#etc]client.autocommit(false)inserts.eachdo|ins|beginclient.query(ins)rescue#handleerrorsorabortentirelyendendclient.commi
我想在Ruby的TCPServer中获取客户端的IP地址。以及(如果可能的话)MAC地址。例如,Ruby中的时间服务器,请参阅评论。tcpserver=TCPServer.new("",80)iftcpserverputs"Listening"loopdosocket=tcpserver.acceptifsocketThread.newdoputs"Connectedfrom"+#HERE!HowcanigettheIPAddressfromtheclient?socket.write(Time.now.to_s)socket.closeendendendend非常感谢!
电脑启动出现显示器黑屏是一个相当常见的问题。如果您遇到了这个问题,不要惊慌,因为它有很多可能的原因,可以采取一些简单的措施来解决它。在本文中,小编将介绍下面4种常见的电脑启动后显示器黑屏的原因,排查这些原因,快速解决! 演示机型:联想Ideapad700-15ISK-ISE系统版本:Windows10一、显示器问题如果出现电脑启动后显示器黑屏的情况。那么首先您需要检查一下显示器是否正常工作。您可以通过更换另一个显示器或将当前显示器连接到另一台计算机来检查显示器是否存在问题。如果问题仍然存在,那么您可以排除显示器故障的可能性。 二、显卡问题如果您的电脑配备了独立显卡,那么显卡故障也可能是导致电脑
我正在尝试绕过rails配置这个极其复杂的迷宫。到目前为止,我设法在ubuntu上设置了rvm(出于某种原因,ruby在ubuntu存储库中已经过时了)。我设法建立了一个Rails项目。我希望我的测试项目使用mysql而不是mysqlite。当我尝试“rakedb:migrate”时,出现错误:“!!!缺少mysql2gem。将其添加到您的Gemfile:gem'mysql2'”当我尝试“geminstallmysql”时,出现错误,告诉我需要为安装命令提供参数。但是,参数列表很大,我不知道该选择哪些。如何通过在ubuntu上运行的rvm和mysql获取rails3?谢谢。
我有一个模型User,它在创建后的回调中创建了选项#Userhas_one:user_optionsafter_create:create_optionsprivatedefcreate_optionsUserOptions.create(user:self)end我对此有一些简单的Rspec覆盖:describe"newuser"doit"createsuser_optionsaftertheuseriscreated"douser=create(:user)user.user_options.shouldbe_kind_of(UserOptions)endend一切正常,直到我将自
我正在尝试使用RubyEventMachine访问使用SSL证书身份验证的HTTPSWeb服务,但我没有让它工作。我编写了以下简单代码块来对其进行端到端测试:require'rubygems'require'em-http'EventMachine.rundourl='https://foobar.com/'ssl_opts={:private_key_file=>'/tmp/private.key',:cert_chain_file=>'/tmp/ca.pem',:verify_peer=>false}http=EventMachine::HttpRequest.new(url).g
我正在为需要与API建立SSL连接的客户端开发应用程序。我得到了三个文件;一个信任根证书(.cer)文件、一个中间证书(.cer)文件和一个签名的响应文件。我得到的安装说明与IIS或Javakeytool程序有关;我正在用RubyonRails构建应用程序,所以这两种方法都不是一个选项(据我所知)。证书由运行API服务的组织自签名,看来我获得了客户端证书以相互验证https连接。我不确定如何使用我的应用程序中的证书连接和使用API签名响应文件的作用我读过"Usingaself-signedcertificate"和thisarticleonOpenSSLinRuby但两者似乎都不是很到