当前位置:首页 > 技巧大全 > 网站建设 > 正文内容

ThinkPHP5.0 数据库的备份、下载、还原、删除

Git开源网2022-03-20 22:53:34网站建设753

ThinkPHP5.0 数据库的备份、下载、还原、删除

效果图:

ThinkPHP5.0 数据库的备份、下载、还原、删除.png

一、下载类文件
https://github.com/joytom/db_backup
按照下图存放类文件

我对Baksql.php文件进行了一点改动,代码如下:
主要修改的是get_dbdata方法,我也做了标记

<?php
/**
 * 备份数据库的扩展类
*/
namespace org;

class Baksql {
    private $config=[];
    private $handler;
    private $tables = array();//需要备份的表
    private $begin; //开始时间
    private $error;//错误信息

    public function __construct($config) {
        /记住创建这个存放备份文件目录,不然会报错
        $config['path']=ROOT_PATH . 'public' . DS .'static'. DS .'data/'; //默认目录
        $config["sqlbakname"]=date("YmdHis",time()).".sql";//默认保存文件
        $this->config = $config;
        
        $this->begin = microtime(true);
        header("Content-type: text/html;charset=utf-8");
        $this->connect();
    }
    //首次进行pdo连接
    private function connect() {
        try{
           $this->handler =new \PDO("{$this->config['type']}:host={$this->config['hostname']};port={$this->config['hostport']};dbname={$this->config['database']};",
                $this->config['username'],
                $this->config['password'], 
                array(
                    \PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES {$this->config['charset']};",
                    \PDO::ATTR_ERRMODE =>  \PDO::ERRMODE_EXCEPTION, 
                    \PDO::ATTR_DEFAULT_FETCH_MODE => \PDO::FETCH_ASSOC
                )); 
        }catch (PDOException $e) {
            die ("Error!: " . $e->getMessage() . "<br/>");
        }
        
    }
     /**
     * 查询
     * @param string $sql
     * @return mixed
     */
    private function query($sql = '')
    {
        $stmt = $this->handler->query($sql);
        $stmt->setFetchMode(\PDO::FETCH_NUM);
        $list = $stmt->fetchAll();
        return $list;
    }
     /**
     * 获取全部表
     * @param string $dbName
     * @return array
     */
    private function get_dbname($dbName = '*') {
         $sql = 'SHOW TABLES';
        $list = $this->query($sql);
        $tables = array();
        foreach ($list as $value)
        {   
            $tables[] = $value[0];
        }
        return $tables;        
    }
    /**
     * 获取表定义语句
     * @param string $table
     * @return mixed
     */
    private function get_dbhead($table = '')
    {
        $sql = "SHOW CREATE TABLE `{$table}`";
        $ddl = $this->query($sql)[0][1] . ';';
        return $ddl;
    }
    /**
     * 获取表数据
     * @param string $table
     * @return mixed
     */
    private function get_dbdata($table = '')
    {
        $sql = "SHOW COLUMNS FROM `{$table}`";
        $list = $this->query($sql);
        //字段
        $columns = '';
        //需要返回的SQL
        $query = '';
        foreach ($list as $value)
        {
            $columns .= "`{$value[0]}`,";
        }
        $columns = substr($columns, 0, -1);
        $data = $this->query("SELECT * FROM `{$table}`");
        foreach ($data as $value)
        {
            $dataSql = '';
      /我的数据库中有 上传图片的路径带有 \特殊字符 备份时消失了
      /数据库时候路径中的反斜杠会丢失,所以导出的时候要再给他加一个反斜杠///
            $value=array_map('addslashes', $value);//给预定义字符加反斜杠
            //原来的代码
//            foreach ($value as $v)
//            {
//                $dataSql .= "'{$v}',";
//            }

           
	/1.程序生成的代码,字段没有字符集;主要出现在varchar类型
    /2.程序生成的代码,int字段的数据被加单引号''//
             //修改
     		/**
             * 2019-07-25 xiaoxia  修改后
             * */
            //cp循环次数
            $cp_loop_time=0;
            foreach ($value as $v)
            {
                if(strpos($list[$cp_loop_time][1],'int') !== false || strpos($list[$cp_loop_time][1],'double') !== false )
                {
                    if($v==null)
                    {
                        $dataSql .= "NULL,";
                    }
                    else{
                        $dataSql .= "{$v},";
                    }
                }
                else{
                    $dataSql .= "'{$v}',";
                }
                $cp_loop_time++;
            }
            /**
             * 2019-07-25 xiaoxia 修改结束
             * */


            $dataSql = substr($dataSql, 0, -1);
            $query .= "INSERT INTO `{$table}` ({$columns}) VALUES ({$dataSql});\r\n";
        }

        return $query;
    }
    /**
     * 写入文件
     * @param array $tables
     * @param array $ddl
     * @param array $data
     */
    private function writeToFile($tables = array(), $ddl = array(), $data = array())
    {
        $str = "/*\r\nMySQL Database Backup Tools\r\n";
        $str .= "Server:{$this->config['hostname']}:{$this->config['hostport']}\r\n";
        $str .= "Database:{$this->config['database']}\r\n";
        $str .= "Data:" . date('Y-m-d H:i:s', time()) . "\r\n*/\r\n";
        $str .= "SET FOREIGN_KEY_CHECKS=0;\r\n";
        $i = 0;
        foreach ($tables as $table)
        {
            $str .= "-- ----------------------------\r\n";
            $str .= "-- Table structure for {$table}\r\n";
            $str .= "-- ----------------------------\r\n";
            $str .= "DROP TABLE IF EXISTS `{$table}`;\r\n";
            $str .= $ddl[$i] . "\r\n";
            $str .= "-- ----------------------------\r\n";
            $str .= "-- Records of {$table}\r\n";
            $str .= "-- ----------------------------\r\n";
            $str .= $data[$i] . "\r\n";
            $i++;
        }
              
        if(!file_exists($this->config['path'])){mkdir($this->config['path']);}
        return file_put_contents($this->config['path'].$this->config['sqlbakname'], $str) ? '备份成功!花费时间' . round(microtime(true) - $this->begin,2) . 'ms' : '备份失败!';
    }
    /**
     * 设置要备份的表
     * @param array $tables
     */
    private function setTables($tables = array())
    {
        if (!empty($tables) && is_array($tables))
        {
            //备份指定表
            $this->tables = $tables;
        }
        else
        {
            //备份全部表
            $this->tables = $this->get_dbname();
        }
    }
    /**
     * 备份
     * @param array $tables
     * @return bool
     */
    public function backup($tables = array())
    {
        //存储表定义语句的数组
        $ddl = array();
        //存储数据的数组
        $data = array();
        $this->setTables($tables);
        if (!empty($this->tables))
        {
            foreach ($this->tables as $table)
            {
                $ddl[] = $this->get_dbhead($table);
                $data[] = $this->get_dbdata($table);
            }
            
            //开始写入
            return $this->writeToFile($this->tables, $ddl, $data);
        }
        else
        {
            $this->error = '数据库中没有表!';
            return false;
        }
    }
    /**
     * 错误信息
     * @return mixed
     */
    public function getError()
    {
        return $this->error;
    }
    public function restore($filename = '')
    {
        $path=$this->config['path'].$filename;
        if (!file_exists($path))
        {
            $this->error('SQL文件不存在!');
            return false;
        }
        else
        {
            $sql = $this->parseSQL($path);
            //dump($sql);die;
            try
            {
                $this->handler->exec($sql);
                echo '还原成功!花费时间', round(microtime(true) - $this->begin,2) . 'ms';
            }
            catch (PDOException $e)
            {
                $this->error = $e->getMessage();
                return false;
            }
        }
    }
 
    /**
     * 解析SQL文件为SQL语句数组
     * @param string $path
     * @return array|mixed|string
     */
    private function parseSQL($path = '')
    {
        $sql = file_get_contents($path);
        $sql = explode("\r\n", $sql);
        //先消除--注释
        $sql = array_filter($sql, function ($data)
        {
            if (empty($data) || preg_match('/^--.*/', $data))
            {
                return false;
            }
            else
            {
                return true;
            }
        });
        $sql = implode('', $sql);
        //删除/**/注释
        $sql = preg_replace('/\/\*.*\*\//', '', $sql);
        return $sql;
    }
    /**
     * 下载备份
     * @param string $fileName
     * @return array|mixed|string
     */
    public function downloadFile($fileName) {
        $fileName=$this->config['path'].$fileName;
        if (file_exists($fileName)){
            ob_end_clean();
            header("Cache-Control: must-revalidate, post-check=0, pre-check=0");
            header('Content-Description: File Transfer');
            header('Content-Type: application/octet-stream');
            header('Content-Length: ' . filesize($fileName));
            header('Content-Disposition: attachment; filename=' . basename($fileName));
            readfile($fileName);
        }else{
            $this->error="文件有错误!";
        }

    }
    /**
     * 获取文件是时间
     * @param string $file
     * @return string
     */
    private function getfiletime($file){
        $path=$this->config['path'].$file;
        $a = filemtime($path);
        $time = date("Y-m-d H:i:s", $a);
        return $time;
    }
    /**
     * 获取文件是大小
     * @param string $file
     * @return string
     */
    private function getfilesize($file){
        $perms=stat($this->config['path'].$file);
        $size = $perms['size'];
        $a    = ['B', 'KB', 'MB', 'GB', 'TB'];
        $pos  = 0;
        while ($size >= 1024) {
            $size /= 1024;
            $pos++;
        }
        return round($size, 2). $a[$pos];
    }
    
    /**
     * 获取文件列表
     * @param string $Order 级别
     * @return array
     */
    public function get_filelist($Order = 0) {
        $FilePath=$this->config['path'];
//        print_r($FilePath);die;
        $FilePath = opendir($FilePath);
//        $FilePath = scandir($FilePath);
        $FileAndFolderAyy=array();
        $i=1;
        while (false !== ($filename = readdir($FilePath))) {
            if ($filename!="." && $filename!=".."){
                $i++;
                $FileAndFolderAyy[$i]['name'] = $filename;
                $FileAndFolderAyy[$i]['time'] = $this->getfiletime($filename);
                $FileAndFolderAyy[$i]['size'] = $this->getfilesize($filename);
            }
            
        }
        $Order == 0 ? sort($FileAndFolderAyy) : rsort($FileAndFolderAyy);
        return $FileAndFolderAyy;
    }
    public function delfilename($filename){
        $path=$this->config['path'].$filename; 
        if (@unlink($path)) {return '删除成功';}
    }
}

?>

二、控制器实例化该类文件:
当前我的控制器是admin2/System
System.php

<?php

namespace app\admin2\controller;

use think\Controller;
use org\Baksql;
use think\Config;
class System extends Controller
{
    //数据库备份
    public function mysqllist()
    {
        //获取操作内容:(备份/下载/还原/删除)数据库
        $type = input("type");
        //获取需要操作的数据库名字
        $name = input("name");
        $backup=new Baksql(Config::get("database"));
        //$backup = new \org\Baksql(\think\Config::get("database"));
        switch ($type) {
            //备份
            case "backup":
                $info = $backup->backup();
                $this->success("$info", 'admin2/system/mysqllist');
                break;
            //下载
            case "dowonload":
                $info = $backup->downloadFile($name);
                $this->success("$info", 'admin2/system/mysqllist');
                break;
            //还原
            case "restore":
                $info = $backup->restore($name);
                $this->success("$info", 'admin2/system/mysqllist');
                break;
            //删除
            case "del":
                $info = $backup->delfilename($name);
                $this->success("$info", 'admin2/system/mysqllist');
                break;
            //如果没有操作,则查询已备份的所有数据库信息
            default:
                $len=count($backup->get_filelist());
                return $this->fetch("system/mysqllist", ["backuplist" => array_reverse($backup->get_filelist()),"len"=>$len]);//将信息由新到老排序
        }
    }
}

三、视图层
mysqllist.html

1.H-ui框架写法 (有分页搜索功能,框架文件自己官网看文档自行引入)

<div class="ibox-content">
    <div class="">
        <a class="btn btn-primary radius" href="{:url('mysqllist',['type'=>'backup'])}"οnclick="return confirm('备份数据的时间较长,确定要备份所有数据吗?')"><i class="Hui-iconfont">&#xe600;</i> <span class="bold">添加备份</span></a>
    </div>
    <table class="table table-border table-bordered table-bg table-sort"  style="text-align: center">
        <thead>
        <tr class="text-c">
            <td>序号</td>
            <th style="display: none" >排序</th>
            <td>备份名称</td>
            <td>备份时间</td>
            <td>备份大小</td>
            <td>操作</td>
        </tr>
        </thead>
        <tbody>
        {volist name="backuplist" id="vo"}
        <tr class="text-c" >
            <td>{$key+1}</td>
            <td style="display: none"><input type="text" class="input-text text-c" value="1"></td>
            <td>{$vo.name}</td>
            <td>{$vo.time}</td>
            <td>{$vo.size}</td>
            <td width="25%">
                <a  href="{:url('mysqllist',['type'=>'dowonload','name'=>$vo.name])}" class="btn btn-success radius"><i class="Hui-iconfont">&#xe640;</i> <span class="bold">下载</span></a>
                <a  href="{:url('mysqllist',['type'=>'restore','name'=>$vo.name])}" class="btn btn-secondary   radius" onclick="return confirm('备份还原后仅会显示当前备份的数据库的信息,您确定还原备份吗 ?')"><i class="Hui-iconfont">&#xe68f;</i> <span class="bold">还原</span></a>
                <a  href="{:url('mysqllist',['type'=>'del','name'=>$vo.name])}"  class="btn btn-warning radius" onclick="return confirm('数据库备份一旦删除不可找回,您确定操作吗?')"><i class="Hui-iconfont">&#xe6e2;</i> <span class="bold">删除</span></a>
            </td>
        </tr>
        {/volist}
        </tbody>
    </table>
</div>
<script>
    $('.table-sort').dataTable({
        "aaSorting": [[ 1, "desc" ]],//默认第几个排序
        "bStateSave": true,//状态保存
        "aoColumnDefs": [
            //{"bVisible": false, "aTargets": [ 3 ]} //控制列的隐藏显示
            {"orderable":false,"aTargets":[1,"desc"]}// 制定列不参与排序
        ]
    });
</script>

//或
2.Bootstrap框架视图:

<div class="ibox-content">
    <div class="">
        <a class="btn btn-primary " href="{:url('mysqllist',['type'=>'backup'])}"οnclick="return confirm('备份数据的时间较长,确定要备份所有数据吗?')"><i class="glyphicon glyphicon-plus"></i> <span class="bold">添加备份</span></a>
    </div>
    <table class="table table-striped table-bordered table-hover dataTables-example"  style="text-align: center">
        <thead>
        <tr>
            <td>序号</td>
            <td>备份名称</td>
            <td>备份时间</td>
            <td>备份大小</td>
            <td>操作</td>
        </tr>
        </thead>
        <tbody>
        {volist name="backuplist" id="vo"}
        <tr class="gradeX" >
            <td>{$key+1}</td>
            <td>{$vo.name}</td>
            <td>{$vo.time}</td>
            <td>{$vo.size}</td>
            <td width="25%">
                <a  href="{:url('mysqllist',['type'=>'dowonload','name'=>$vo.name])}" class="btn btn-success "><i class="glyphicon glyphicon-download-alt"></i> <span class="bold">下载</span></a>
                <a  href="{:url('mysqllist',['type'=>'restore','name'=>$vo.name])}" class="btn btn-info " onclick="return confirm('备份还原后仅会显示当前备份的数据库的信息,您确定还原备份吗 ?')"><i class="glyphicon glyphicon-repeat"></i> <span class="bold">还原</span></a>
                <a  href="{:url('mysqllist',['type'=>'del','name'=>$vo.name])}"  class="btn btn-warning" onclick="return confirm('数据库备份一旦删除不可找回,您确定操作吗?')"><i class="fa fa-warning"></i> <span class="bold">删除</span></a>
            </td>
        </tr>
        {/volist}
        </tbody>
    </table>
</div>

注意这里的:

 //mysqllist指的是控制器里的mysqllist方法名
 href="{:url('mysqllist',['type'=>'del','name'=>$vo.name])}"
$this->success("$info", 'admin2/system/mysqllist');//指的是成功之后跳转的页面路由

参考文档:

https://blog.csdn.net/qq_42455095/article/details/84313754#commentBox

https://blog.csdn.net/qq_42249896/article/details/85238523

https://blog.csdn.net/cplvfx/article/details/86494487

自己整理了一个文档~~~


原文:https://blog.csdn.net/xiaxia_Lin/article/details/97270696

扫描二维码推送至手机访问。

版权声明:本文由Git开源网_git开源代码资源网_git开源博客发布,如需转载请注明出处。

本文链接:http://gitoscc.com/?id=861

相关文章

Ex_DirectUI自绘会计从业题库

Ex_DirectUI自绘会计从业题库

Ex_DirectUI自绘会计从业题库[Download]资源名称:Ex_DirectUI自绘会计从业题库下载地址:https://www.lanzoui.com/i25w9ad [/Download]...

上传MYSQL数据库、EXCEL导入导出超级列表框

上传MYSQL数据库、EXCEL导入导出超级列表框

上传MYSQL数据库、EXCEL导入导出超级列表框本地下载上传MYSQL数据库、EXCEL导入导出超级列表框.zip...

MySql连接池支持库1.0 2.0版(Mysql_Connet.fne)

MySql连接池支持库1.0 2.0版(Mysql_Connet.fne)mysql连接池 提供多线程操作MySql解决方案。操作系统支持: Windows[Download]资源名称:MySql连接池支持库1.02.0版(Mysql_Co...

Navicat Products 注册机 v3.4

Navicat Products 注册机 v3.4

Navicat Products 注册机是Navicat Premium 12 Windows 提供激活码的小工具,Navicat Premium 12可以让你以单一程式同时连线到 MySQL、SQLite、Oracle 及 Postgre...

MySQL批量更新替换字段内容

update table set num = replace(num,num,num/100)第一个num表示 num字段 第二个num表示这个字段的原始值, 第三个num表示要修改成什么值...

Discuz!X3.4最新版完美解决帖子标题80个字符的限制

一个很烦人的问 题,discuz!x3.4的帖子标题长度(字数)有限制,只能是80个字符!UTF-8的一个汉字就要占用 三个字符!80个字符只能输入26个汉字左右!而GBK版本80个字符能输入40个汉字,UTF8的怎么能够用呢!为了让UTF...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。