MySQL好玩新特性:离线模式
Posted GreatSQL
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了MySQL好玩新特性:离线模式相关的知识,希望对你有一定的参考价值。
- GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源。
- GreatSQL是MySQL的国产分支版本,使用上与MySQL一致。
- 作者:Yejinrong/叶金荣
- 文章来源:GreatSQL社区原创
继续吹MySQL 8.0~
在以前,当需要对MySQL数据库进行维护操作时,通常需要先进行主从切换,然后修改设置并重启实例,关闭网络监听,只允许从本地socket方式登入,再进行相应的维护操作;有时候甚至还要修改相应的防火墙,或者干脆关闭前端业务服务,总体比较麻烦。
从MySQL 5.7开始,支持设置为离线模式(offline_mode),再有维护操作需求就不用这么麻烦了。只需在线动态修改,可立即生效,非常的简单粗暴:
mysql> set global offline_mode = on; -- 打开离线模式,拒绝外部请求
mysql> set global offline_mode = off; -- 关闭离线模式,允许外部连接请求
当设置为离线模式后,普通用户将无法继续发起连接请求,甚至当前正在执行的SQL也会立即被终止并被断开连接。
1. 无法创建新连接
$ mysql -h127.0.0.1 -uyejr -pxx -P4306 sbtest
mysql: [Warning] Using a password on the command line interface can be insecure.
ERROR 3032 (HY000): The server is currently in offline mode
2. 即便是普通用户通过本地socket连接,当启用离线模式后,也会被断开
$ mysql -S/data/MySQL/mysql.sock
...
mysql> show processlist;
+-----+------+-----------+------+---------+------+-------+------------------+
| Id | User | Host | db | Command | Time | State | Info |
+-----+------+-----------+------+---------+------+-------+------------------+
| 304 | yejr | localhost | NULL | Query | 0 | init | show processlist |
+-----+------+-----------+------+---------+------+-------+------------------+
1 row in set (0.00 sec)
[sbtest]>select *,sleep(10) from t1 limit 3; -- 正在运行的SQL会立即被终止,并断开连接
ERROR 2013 (HY000): Lost connection to MySQL server during query
No connection. Trying to reconnect...
ERROR 3032 (HY000): The server is currently in offline mode
ERROR:
Can\'t connect to the server
3. 正在运行中的sysbench压测,也会被立即断开
[ 1s ] thds: 16 tps: 442.02 qps: 9078.28 (r/w/o: 6382.37/1795.94/899.96) lat (ms,99%,99%,99.9%): 150.29/150.29/150.29 err/s: 0.00 reconn/s: 0.00
[ 2s ] thds: 16 tps: 471.23 qps: 9387.56 (r/w/o: 6576.19/1868.91/942.46) lat (ms,99%,99%,99.9%): 61.08/61.08/65.65 err/s: 0.00 reconn/s: 0.00
[ 3s ] thds: 16 tps: 386.03 qps: 7712.68 (r/w/o: 5399.48/1541.14/772.07) lat (ms,99%,99%,99.9%): 82.96/82.96/84.47 err/s: 0.00 reconn/s: 0.00
[ 4s ] thds: 16 tps: 547.00 qps: 10894.97 (r/w/o: 7609.98/2190.99/1094.00) lat (ms,99%,99%,99.9%): 65.65/65.65/68.05 err/s: 0.00 reconn/s: 0.00
FATAL: mysql_drv_query() returned error 2013 (Lost connection to MySQL server during query) for query \'COMMIT\'
(last message repeated 1 times)
FATAL: mysql_drv_query() returned error 2013 (Lost connection to MySQL server during query) for query \'SELECT c FROM sbtest1 WHERE id=4822870\'
FATAL: mysql_drv_query() returned error 2013 (Lost connection to MySQL server during query) for query \'COMMIT\'
FATAL: mysql_drv_query() returned error 2013 (Lost connection to MySQL server during query) for query \'UPDATE sbtest1 SET k=k+1 WHERE id=2265001\'
FATAL: mysql_drv_query() returned error 2013 (Lost connection to MySQL server during query) for query \'SELECT c FROM sbtest8 WHERE id BETWEEN 3389984 AND 3390083\'
另外,从MySQL 8.0开始,对于离线模式又做了些改进和完善,比如新引入 CONNECTION_ADMIN
权限等,细化离线模式的权限管理模式。
简单几点小结关于离线模式:
- 必须要有
CONNECTION_ADMIN
以及CONNECTION_ADMIN
权限 或者SUPER
权限(SUPER
权限在未来会被废弃,而细分成更多细粒度权限),才能在线设置离线模式。 - 复制线程不会受到离线模式影响,还能正常工作。
- 当设置为离线模式时,没有授予
CONNECTION_ADMIN
或SUPER
权限的普通用户,正在执行的SQL会被立即终止,连接也会被立即断开。 - 当设置为离线模式时,拥有
CONNECTION_ADMIN
或SUPER
权限的用户,不会被断开连接。 - 当设置离线模式的用户不具备
SYSTEM_USER
权限(只拥有CONNECTION_ADMIN
以及CONNECTION_ADMIN
权限)的话,拥有SYSTEM_USER
权限的活跃用户连接不会被断开(因为想要断开SYSTEM_USER
权限级别用户连接同样需要至少有SYSTEM_USER
权限),详见下面的案例。
有 u1 和 u2 两个用户,授权模式不同
mysql> show grants for u1;
+----------------------------------------+
| Grants for u1@% |
+----------------------------------------+
| GRANT USAGE ON *.* TO `u1`@`%` |
| GRANT SELECT ON `sbtest`.* TO `u1`@`%` |
+----------------------------------------+
mysql> show grants for u2;
+----------------------------------------+
| Grants for u2@% |
+----------------------------------------+
| GRANT USAGE ON *.* TO `u2`@`%` |
| GRANT SYSTEM_USER ON *.* TO `u2`@`%` |
| GRANT SELECT ON `sbtest`.* TO `u2`@`%` |
+----------------------------------------+
用户 yejr 的授权模式如下
+--------------------------------------------------------------------+
| Grants for yejr@% |
+--------------------------------------------------------------------+
| GRANT USAGE ON *.* TO `yejr`@`%` |
| GRANT CONNECTION_ADMIN,SYSTEM_VARIABLES_ADMIN ON *.* TO `yejr`@`%` |
| GRANT ALL PRIVILEGES ON `sbtest`.* TO `yejr`@`%` |
+--------------------------------------------------------------------+
当 yejr 用户设置离线模式后,u2 用户的连接不会被断开(但不能再建立新连接),而 u1 用户的连接会被断开
# 三个用户先分别建立连接
$ jobs
[1] Stopped mysql -h127.0.0.1 -uyejr -pxx -P4306 sbtest
[2]- Stopped mysql -h127.0.0.1 -uu2 -pxx -P4306 sbtest
[3]+ Stopped mysql -h127.0.0.1 -uu1 -pxx -P4306 sbtest
# 设置离线模式
$ fg 1
mysql -h127.0.0.1 -uyejr -pxx -P4306 sbtest
[yejr@db160] [sbtest]>set global offline_mode=on;
Query OK, 0 rows affected (0.00 sec)
# u1用户被断开连接
$ fg 3
mysql -h127.0.0.1 -uu1 -pxx -P4306 sbtest
[u1@db160] [sbtest]>select 1;
ERROR 2013 (HY000): Lost connection to MySQL server during query
No connection. Trying to reconnect...
ERROR 3032 (HY000): The server is currently in offline mode
ERROR:
Can\'t connect to the server
# u2用户不会被断开连接
$ fg 2
mysql -h127.0.0.1 -uu2 -pxx -P4306 sbtest
[u2@db160] [sbtest]>select 1;
+---+
| 1 |
+---+
| 1 |
+---+
# 但u1/u2用户均不能再建立新链接
$ mysql -h127.0.0.1 -uu2 -pxx -P4306 sbtest
mysql: [Warning] Using a password on the command line interface can be insecure.
ERROR 3032 (HY000): The server is currently in offline mode
$ mysql -h127.0.0.1 -uu1 -pxx -P4306 sbtest
mysql: [Warning] Using a password on the command line interface can be insecure.
ERROR 3032 (HY000): The server is currently in offline mode
是不是有点好玩呀~
结合前面的两篇文章 MySQL 8.0不再担心被垃圾SQL搞爆内存 以及 InnoDB buffer pool size进度更透明 可以看到MySQL 8.0在各个细节方面做的是越来越好了。
延伸阅读
- #sysvar_offline_mode, https://dev.mysql.com/doc/refman/8.0/en/server-system-variables.html
- Changes in MySQL 8.0.31, https://dev.mysql.com/doc/relnotes/mysql/8.0/en/news-8-0-31.html
Enjoy GreatSQL
C# 7.0 新特性3: 模式匹配
本文参考Roslyn项目Issue:#206,及Docs:#patterns。
1. C# 7.0 新特性1: 基于Tuple的“多”返回值方法
模式匹配也许能算的上C#本次更新最重量级的升级,也是最受关注的特性(也许没有之一),通过模式匹配,我们可以简化大量的条件代码。
Switch语句
大家也许遇到过这样的情景,假设你的代码中,有一个Nullable<int>的值,需要对其在正整数,非正整数,Null三种情况下分别作不同的逻辑处理。大多数童鞋直接想到是类似于下面的逻辑:

1 void Foo(int? num) 2 { 3 if (!num.HasValue) 4 /* null logic */ 5 else if (num.Value > 0) 6 /* positive int logic */ 7 else 8 /* negative int & zero logic */ 9 }
请大家思考一下,这个逻辑是否可以用switch-case语句来做,在VB及很多非C系的语言中,答案是肯定的,比如VB.NET中可以这样写:

1 Sub Foo(Num As Integer?)
2 Select Case Num
3 Case Not Num.HasValue
4 \'null logic
5 Case Num > 0
6 \'positive Int logic
7 Case Num <= 0
8 \'negative Int() & zero logic
9 Case Else
10
11 End Select
12 End Sub
说到这里,在具体讨论模式匹配在switch-case中的应用之前,先淡淡的吐槽一下C#,本来理所应当的一个简单的小语法,到了C#7.0才加入。
看看C#7.0加入的类型模式(Type Pattern):
1 void Foo(int? num) 2 { 3 switch (num) 4 { 5 case null: 6 //null logic 7 break; 8 case int n when n > 0: 9 //positive Int logic 10 break; 11 case int n when n <= 0: 12 //negative Int() & zero logic 13 break; 14 } 15 }
这个不多说了,大家自己体会,单纯的在Nullable<int>下,可能体现的不是很清晰,个人认为这个小变动其实意义并不是很大,同样场景下,或许if-if else-else会让代码更清晰易读些。
如果说模式匹配仅仅是完善了一下switch-case,那可真是太大才小用了,下面我们看一个好玩的。
Match表达式
虽然把match带到C#中看起来并不是什么大事,但是会引起的代码简化还是非常爽的。
就像很多人说三元表达式(<condition>?<true result> : <false result> )将if-else简化一样。match表达式,是将switch-case结构简化到了一个新限度。
看match表达式代码前,我们先来看一行略坑的三元表达式。
var reuslt = x == null ? default(int) : (x is Func<int> ? (x as Func<int>)() : (x is int ? Convert.ToInt32(x) : default(int)));
好吧,我承认我是故意让你们抓狂的。^_^, 为了能稳住大家看完上面这行代码后的情绪,来一副match表达式消消火。
var result = x match( case Func<int> f: f(), case int i: i, case *: default(int) );
这两种写法效果上是等效的,有没有非常干净清爽的感觉?写过match表达式的码农,应该再也不想回去嵌套 <*>?<*>:<*> 了。 (注:目前这种写法还未确认,C#7.0发布后可能会有略微变动)
Is表达式
如果说上面两个变化是“语法糖”,那么is表达式可是要玩真的了。
说点题外话,其实对正则表达式熟悉的童鞋可能知道,本质上[模式匹配]和正则表达式要解决的问题逻辑类似,以一个确定的模式,来判断或查找一个确定的实例。只不过在正则表达式中,这里说的"模式"是正则表达式,"实例"指字符串。而[模式匹配]下,所针对的"实例"是对象,那么"模式",就可以理解成is表达式了。
举个例子,比如你要查找并列出 一组电子设备中,所有iPhone的IMEI串号,我们在C#6.0中,会这样做:
1 class Device 2 { 3 public ProductLineOption ProductLine { get; set; } 4 } 5 6 class MobiePhone : Device 7 { 8 public string IMEICode { get; set; } 9 } 10 11 IEnumerable<Device> GetAllDevices() { /* 获取并返回所有设备 */ }; 12 13 IEnumerable<string> GetAlliPhoneIMEI() 14 { 15 var deviceList = this.GetAllDevices(); 16 foreach (Device device in deviceList) 17 { 18 MobiePhone phone = device as MobiePhone; 19 if (phone == null) continue; 20 21 if (phone.ProductLine == ProductLineOption.IPhone) 22 { 23 yield return phone.IMEICode; 24 } 25 } 26 }
一个非常典型的传统方法,没什么好说的。我们直接来看C#7.0 中 is表达式怎么等效的实现这段逻辑:
1 IEnumerable<string> GetAlliPhoneIMEI() 2 { 3 List<Device> deviceList = this.GetAllDevices(); 4 foreach (Device device in deviceList) 5 { 6 if (device is MobiePhone { IMEICode is var imei, ProductLine is ProductLineOption.IPhone}) 7 { 8 yield return imei; 9 } 10 } 11 }
如果你还是觉得这没什么,那么,其实这个例子中,仅仅体现出模式匹配中的属性模式。
根据Doc:#patterns C#7.0会提供一下几种匹配方式:
- 类型模式
- 常量模式
- 变量模式
- 通配符模式
- 位置模式
- 属性模式
我们可以想象,如果模式匹配组合起来使用,会给现有的C#代码带来多大的便利和清静。
Okay,说了这么多,下面给大家一个相对完整的案例,自行体会。
案例
1 abstract class Animal 2 { 3 public string Name { get; set; } 4 } 5 6 class Dog : Animal 7 { 8 public string BarkLikeCrazy() => "WOOF WOOF WOOF"; 9 } 10 11 class Cat : Animal { } 12 class Swan : Animal { } 13 14 class Program 15 { 16 static void Main(string[] args) 17 { 18 var animals = new Animal[] { 19 new Dog { Name = "hola" }, 20 new Cat { Name = "tom" }, 21 new Swan { Name = "hacienda" } 22 }; 23 24 var organizedAnimals = from animal in animals 25 let sound = animal match( //Match语句 26 case Dog d: "woof... " + d.BarkLikeCrazy(), //类型匹配 27 case Cat c: "meow", 28 case * : "I\'m mute.." //通配符匹配 29 ) 30 select new { Type = animal, Sound = sound }; 31 32 foreach (var animal in organizedAnimals) 33 { 34 Console.WriteLine($"{animal.Type.ToString()} - {animal.Sound}"); 35 } 36 37 foreach (var a in animals) 38 { 39 if (a is Cat { Name is var name }) //类型及属性匹配,is表达式 40 { 41 Console.WriteLine($"Name of {nameof(Cat)} is {name}"); 42 } 43 44 string sound = ""; 45 switch (a) //匹配switch语句 46 { 47 case Dog d when d.Name == "hola": 48 sound = "woof... hola" + d.BarkLikeCrazy(); 49 break; 50 case Dog d: 51 sound = "woof..." + d.BarkLikeCrazy(); 52 break; 53 case Cat c: 54 sound = "meow"; 55 break; 56 case IEnumerable<Animal> l when l.Any(): 57 //TODO: any logic; 58 break; 59 case null: 60 sound = "no animal"; 61 break; 62 default: 63 sound = "I\'m mute.."; 64 break; 65 } 66 Console.WriteLine($"{a.ToString()} - {sound}"); 67 } 68 } 69 }
注1:模式匹配的部分高级feature,已经确认在C#7.0中移除,可能出现在后续C#版本中。(#10888)。
注2:目前(2016-06-15)VS15的最新Preview下,模式匹配的部分语法依然无法使用。
注3:由于目前仍然未在Roslyn中Release,后期有变动的可能,本文中涉及的样例代码以Mads Torgersen在#Build 2016上的演示的语法为准,本文涉及的案例有可能无法在VS15 RTM后正常使用,仅供参考。
(当然,如果笔者乐意,会及时把后期得到确认的变更更新到本文中 ^_^!)
本文链接:http://www.cnblogs.com/ylvict/p/5588613.html (转载请注明)
目前(2016年6月)C#7.0还未正式发布,大家如果想体验部分特性,可以去下载VS15预览版,最终发布的语法可能和本文中提及的有所不同,最新动态请大家关注Roslyn项目。
以上是关于MySQL好玩新特性:离线模式的主要内容,如果未能解决你的问题,请参考以下文章
TP3.2项目 MySQL5.7报错1055 group by新特性