利用python解决mysql视图导入导出依赖的问题


Posted in Python onDecember 17, 2017

视图

视图是一个虚拟表(非真实存在),其本质是根据SQL语句获取动态的数据集,并为其命名,用户使用时只需使用名称即可获取结果集,并可以将其当作表来使用。

创建视图

创建一个名称为v1的视图,其功能就是查询color表中的所有数据

CREATE VIEW v1 AS SELECT * FROM color;

查看视图

使用视图时,将其当作表进行操作即可,由于视图是虚拟表,所以无法使用其对真实表进行创建、更新和删除操作,仅能做查询用。

select * from v1; -- 等于执行SELECT * FROM color

输出结果

+-----+--------+
| nid | title |
+-----+--------+
| 1 | red |
| 2 | yellow |
+-----+--------+
2 rows in set (0.00 sec)

修改视图

ALTER VIEW v1 AS SELECT * FROM color WHERE nid = 1;

删除视图

DROP VIEW v1;

引用

navicat是mysql可视化工具中最棒的,但是,在处理视图的导入导出方面,它是按照视图名称的字母顺序来处理的,若视图存在依赖,在导入过程中就会报错。这个问题一直困绕我,一度因为我使用docker来部署mysql而绕过了这个问题。最近不得不直面这个问题,因此,写了一个小工具来解决它。

整体思路

在mysql很容易查出所有视图和其定义,因此可以写一个视图导出工具,存储时对各视图的保存位置进行调整,处理好它们之间的依赖关系,被依赖的放前面,这样就解决了导入时的依赖问题。

获取视图信息

运行以下查询语句,就能获得该数据库中所有视图的信息。

select * from information_schema.VIEWS where TABLE_SCHEMA = DatabaseName

查询结果字段说明:

  • TABLE_NAME : 数所库中视图名称
  • VIEW_DEFINITION : 视图的定义代码,只有查询语句部分
  • DEFINER : 视图定义(建立)者名称
  • SECURITY : 安全级别

总之,所有视图的信息都在这个表中保存,我要完成任务,只需要TABLE_NAME和VIEW_DEFINITION就可以了。

算法描述

  • 将查询结果放到dict中,视图名称为key;视图定义为value;
  • 编写处理依赖关系的函数process_rely,输入参数中的rely_old为保存所有视图名称的数组;返回参数为按依赖关系调整顺序后的视图名称数组。之所以这样做,是一开始考虑到,依赖关系复杂时,可能一次迭代处理不好,需要递归调用或多次调用。

process_rely函数算法描述:

    第一层循环,从rely_old中取一个视图名称

        第二层循环,从dict中取出一个键值

            若键值被第一层元素的定义所依赖

                若键值还不在结果数组中

                    若第一层元素不在结果数组中

                        追加键值到结果数组中

                    第一层元素在结果数组中

                        将键值插入到第一层元素前

                键值在结果数组中

                    第一层元素在结果数组中

                        查找各自在结果数组中的位置

                        若第一层元素在键值的后

                            将键值移动到第一层元素前

        第二层循环结束时,若第一层元素还不在结果集中

            将第一层元素追加到结果集中

    返回结果集

上面的说明,是按python代码模式给出的。很幸运,算法一次就能将复杂的依赖关系处理好了。我在编写的过程中,刚开始依赖算法不完善时,通过多次迭代也能处理好复杂的依赖关系。因此,坚定了必胜的信心,完成了这个任务。

完整代码

import pymysql
conn = pymysql.connect(host='172.17.0.1', port=3306, user='root',
      passwd='123456', db='database', charset='utf8mb4')
def process_rely(parmas={}, rely_old=[]):
 _rely = []
 _keys = list(parmas.keys())
 for k in rely_old:
  for bl in _keys:
   if str(parmas[k]).find(bl) > -1:
    if bl not in _rely:
     if k not in _rely:
      _rely.append(bl)
     else:
      i = _rely.index(k)
      _rely.insert(i, bl)
    else:
     if k in _rely:
      i = _rely.index(k)
      j = _rely.index(bl)
      if i < j:
       del _rely[j]
       _rely.insert(i, bl)
  if k not in _rely:
   _rely.append(k)
 return _rely
cur = conn.cursor()
cur.execute('select TABLE_NAME, VIEW_DEFINITION from information_schema.VIEWS where TABLE_SCHEMA = %s ', 'database')
rs = cur.fetchall()
cur.close()
conn.close()
ps = {}
for al in rs:
 ps['`' + al[0] + '`'] = al[1]
rely = process_rely(ps, list(ps.keys()))
# rely = process_rely(ps, rely1)
file_object = open('view.sql', 'w')
for al in rely:
 file_object.write('DROP VIEW IF EXISTS ' + al + ';\n')
 file_object.write('CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`%` SQL SECURITY DEFINER VIEW ' + al +
      ' AS ' + ps[al] + ';\n\n')
file_object.close()

小结

思路要清晰,代码要一步步的向最终目标靠近,积跬步以至千里。在做这个工具时,一开始觉得很麻烦,依赖关系若是深层次的,可能一次处理不好,正因为采用的迭代的思想,最后才完成了一次迭代解决问题的完美结局。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
python网络爬虫采集联想词示例
Feb 11 Python
深入解析Python中的线程同步方法
Jun 14 Python
Python获取当前函数名称方法实例分享
Jan 18 Python
python中的变量如何开辟内存
Jun 26 Python
Python操作word常见方法示例【win32com与docx模块】
Jul 17 Python
基于Python3.6+splinter实现自动抢火车票
Sep 25 Python
Django 拆分model和view的实现方法
Aug 16 Python
opencv-python 读取图像并转换颜色空间实例
Dec 09 Python
python支持多继承吗
Jun 19 Python
Python中Selenium库使用教程详解
Jul 23 Python
只用50行Python代码爬取网络美女高清图片
Jun 02 Python
python基础之模块的导入
Oct 24 Python
python 3.5实现检测路由器流量并写入txt的方法实例
Dec 17 #Python
python中闭包Closure函数作为返回值的方法示例
Dec 17 #Python
django模板语法学习之include示例详解
Dec 17 #Python
详解python string类型 bytes类型 bytearray类型
Dec 16 #Python
python使用os.listdir和os.walk获得文件的路径的方法
Dec 16 #Python
python读取与写入csv格式文件的示例代码
Dec 16 #Python
浅谈用VSCode写python的正确姿势
Dec 16 #Python
You might like
浅析Yii中使用RBAC的完全指南(用户角色权限控制)
2013/06/20 PHP
经典PHP加密解密函数Authcode()修复版代码
2015/04/05 PHP
PHP生成器简单实例
2015/05/13 PHP
PHP版微信第三方实现一键登录及获取用户信息的方法
2016/10/14 PHP
为Extjs加加速(javascript加速)
2010/08/19 Javascript
修改好的jquery滚动字幕效果实现代码
2011/06/22 Javascript
JavaScript面向对象知识串结(读JavaScript高级程序设计(第三版))
2012/07/17 Javascript
JavaScript SHA512&amp;SHA256加密算法详解
2015/08/11 Javascript
JS响应鼠标点击实现两个滑块区间拖动效果
2015/10/26 Javascript
Node.js本地文件操作之文件拷贝与目录遍历的方法
2016/02/16 Javascript
js编写三级联动简单案例
2016/12/21 Javascript
简单实现JS倒计时效果
2016/12/23 Javascript
vue组件实例解析
2017/01/10 Javascript
Vue系列:通过vue-router如何传递参数示例
2017/01/16 Javascript
JavaScript中的子窗口与父窗口的互相调用问题
2017/02/08 Javascript
基于jQuery的表单填充实例
2017/08/22 jQuery
原生JS封装animate运动框架的实例
2017/10/12 Javascript
Vue 父子组件数据传递的四种方式( inheritAttrs + $attrs + $listeners)
2018/05/04 Javascript
vue组件开发props验证的实现
2019/02/12 Javascript
ECharts地图绘制和钻取简易接口详解
2019/07/12 Javascript
在vue项目中使用codemirror插件实现代码编辑器功能
2019/08/27 Javascript
layer弹窗在键盘按回车将反复刷新的实现方法
2019/09/25 Javascript
vue实践---vue不依赖外部资源实现简单多语操作
2020/09/21 Javascript
Anaconda2下实现Python2.7和Python3.5的共存方法
2018/06/11 Python
wtfPython—Python中一组有趣微妙的代码【收藏】
2018/08/31 Python
详解python多线程之间的同步(一)
2019/04/03 Python
解决windows下python3使用multiprocessing.Pool出现的问题
2020/04/08 Python
Pyinstaller 打包发布经验总结
2020/06/02 Python
python+django+selenium搭建简易自动化测试
2020/08/19 Python
PyCharm上安装Package的实现(以pandas为例)
2020/09/18 Python
美体小铺美国官网:The Body Shop美国
2017/11/10 全球购物
教师自荐书
2013/10/08 职场文书
自荐书范文
2013/12/08 职场文书
见习期自我鉴定
2014/01/31 职场文书
节约用电标语
2014/06/17 职场文书
Netty客户端接入流程NioSocketChannel创建解析
2022/03/25 Java/Android