CSS、JS动态加载,加强版

var dcLoader = {
    load: function(option,callback){
        var _type = "",_url = "";
        var _callback = callback
        option.type? _type = option.type:"";
        option.url? _url = option.url:"";
        typeof option.filtration === "boolean"? filtration = option.filtration:"";
        switch(_type){
            case "js":
            case "javascript": this.js(_url,_callback); break;
            case "css": this.link(_url,_callback); break;
        }
        return this;
    },
    js: function(url,callback){
        var urls = url.replace(/[,]\s*$/ig,"").split(",");
        var scripts = [];
        var completeNum = 0;
        for( var i = 0; i < urls.length; i++ ){
            scripts[i] = document.createElement("script");
            scripts[i].type = "text/javascript";
            scripts[i].src = urls[i];
            document.getElementsByTagName("head")[0].appendChild(scripts[i]);
            if(!this.func(callback)){return this;}
            if(scripts[i].readyState){
                scripts[i].onreadystatechange = function(){
                    if( this.readyState === "loaded" || this.readyState === "complete" ){
                        this.onreadystatechange = null;
                        completeNum++;
                        completeNum >= urls.length ? callback () : '';
                    }
                }
            }else{
                scripts[i].onload = function(){
                    completeNum ++;
                    completeNum >= urls.length ? callback () : '';
                }
            }

        }
        return this;
    },
    link : function(url,callback){
        var urls = url.replace(/[,]\s*$/ig,"").split(",");
        var links = [];
        for( var i = 0; i < urls.length; i++ ){
            links[i] = document.createElement("link");
            links[i].rel = "stylesheet";
            links[i].href = urls[i];
            document.getElementsByTagName("head")[0].appendChild(links[i]);
        }
        this.func(callback) ? callback() : '';
        return this;
    },
    func : function isFunction(fn) {
        return Object.prototype.toString.call(fn)=== '[object Function]';
    }
};

Continue reading...

MySQL InnoDB 备份与恢复

MySQL InnoDB 备份与恢复

MysqlDump

这种方式不仅适用于InnoDB,还适用于其它类型的存储引擎,如MyISAM。备份的时候将数据库备份成SQL(包含drop,create,insert等语句),恢复的时候直接导入即可。属于逻辑备份。

Copy Files拷贝文件

我们知道InnoDB底层存储的时候会将数据和元信息存在下列文件中:
ibdata*, *.ibd, *.frm, .ib_logfile,所以备份这些文件即可备份InnoDB的数据,另外别忘了将MySQL的配置文件my.cnf也一并备份起来。
如果my.cnf中开启了innodb_file_per_table那么InnoDB在存储的时候也会像MyISAM那样每一个表都会有相应的文件,你也可以只备份其中的某些文件(也就是表)。
注意copy这些files的时候,最好能确保InnoDB的所有事务都commit了。
比如你可以将MySQL暂时关闭掉,如果不能忍受一些down time的话,先不要考虑这种方式。
另外还可以先获取Table的read lock, 比如Lock Tables Customers READ.
你可以用show innodb status看看还有没有活动的事务没有commit。属于物理备份。

下面讲一下这种备份方式怎么恢复:

a. 停止任何关于这个表的写,lock tables customers write;

b. 删除这个表的空间,alter table customers discard tablespace;

c. 将你备份的.ibd文件拷贝到相应的目录下。

d. 重新建立表空间,alter table customers import tablespace;

e. 释放写锁,unlock tables customers;

ibbackup

这是一个商业化的工具,将你线上的my.cnf配置文件复制一份出来到比如说/etc/my.backup.cnf,更改里边的datadir等比如说到/data/backup,然后运行 ibbackup /etc/my.cnf /etc/my.backup.cnf, ibbackup就会讲my.cnf所指向的的数据内容备份到my.backup.cnf指向的数据目录。恢复的时候也很奇葩,因为你现在等于说是拥有了两个数据目录,所以你重新运行mysql,safe_mysqld --defaults-file=/etc/my.backup.cnf. 这种方式仅局限于你的数据库全部使用InnoDB存储引擎。

innobackup

如果你还有其它数据表用的是MyISAM,那么你可以使用innobackup,它不但会将InnoDB的相关文件备份起来,还会将MyISAM的比如MYI, MYD文件也备份起来。innobackup使用了ibbackup作为InnoDB的备份子工具。

MySQL Administrator

binary logs

需要开启log-bin, 所有的更新操作都会被写到binary file里。恢复的时候mysqlbinlog binlog_file | mysql,这种方式基本上可以用在在线备份上。属于逻辑备份。

XtraBackup

对InnoDB做数据备份的开源工具,支持在线热备,备份时不影响数据读写。属于物理备份。
项目地址https://github.com/percona/percona-xtrabackup,可以结合使用Innobackupex进行全备,xtrabackup进行增量备份。

个人觉得第一种,第二种和第六种和第七种都非常不错,各有自己的特点。第一种方式比较适合小型的数据库,由于是SQL,所以不仅可以跨存储引擎,还能跨不同的数据库,比如PostgreSQL。数据量稍大一些就可以用第二种方式了,由于是文件传输,所以效率就是文件传输的效率了。第六种方式将数据库本身与备份方式利用binlog进行解耦,既不会对在线数据库造成太大的overhead,而且因为所有的更新操作都在binlog里了,所以可以利用此数据做很多的事情。第七种方式比较特殊,结合了第三种和第四种,所以功能上比较强大,商业上可以选用。

Continue reading...

两个时间段重叠判断算法

在日常做项目中我们会遇到很多与时间相关的一些检测。
如一个酒店的钟占房订单系统,须要计算出一个时间段内是否与一个或多个时间段是否重叠的例子。
用php的实现方法,直接看源代码

<?php
/**
 * [Detection_Overlap_DateTime description]
 * @param  [datetime] $starttime        [开始时间]
 * @param  [datetime] $endtime          [结束时间]
 * @param  [array]  $exist_time_array   [已存在时间数组]
 * @return [Boole]                      [时间冲突返回true]
 * @author [陶之11] <11@pao11.com>
 */
function Detection_Overlap_DateTime($starttime = null, $endtime = null, $exist_time_array = array())
{
    //将时间转换成unix时间戳
    $gstime = ctype_digit($starttime) && $starttime <= 2147483647 ? $starttime : strtotime($starttime);
    $getime = ctype_digit($endtime) && $endtime <= 2147483647 ? $endtime : strtotime($endtime);

    $i = 0;
    //将新加入时间做new标记
    $_array = array($gstime=>'new',$getime=>'new');

    //循环已存在时间并转成unix时间戳
    foreach($exist_time_array as  $v){
        $i++;
        //对已存在时间做标记,把时间戳设为数组键名
        $_array[strtotime($v['stime'])] = $i;
        $_array[strtotime($v['etime'])] = $i;
    }

    ksort($_array);//然后按时间戳排序
    reset($_array);//重置数组
    print_r($_array);
    while(key($_array) != $gstime) next($_array);
    if(next($_array)!='new'){
        return true;
    }

    reset($_array);
    while(key($_array) != $getime) next($_array);
    if(prev($_array)!='new'){
        return true;
    }
    return false;
}

以上带代码用数组排序功能实现了整个算法,其中数组的ksort next prev等方法是该算法的核心。
可以看看调用示例

<?php
//已存在时间段数组
$datetime  = array(
        array('stime'=>'2015-06-26 10:12:12','etime'=>'2015-06-26 10:50:15'),
        array('stime'=>'2015-06-26 11:00:00','etime'=>'2015-06-26 11:20:00'),
        array('stime'=>'2015-06-26 12:11:00','etime'=>'2015-06-26 13:30:00')
               );


//假如要插入一段时间
$stime = '2015-06-26 13:30:01';
$etime = '2015-06-26 14:00:00';

//检查时间段是否冲突
var_dump(Detection_Overlap_DateTime($stime, $etime, $datetime));

结果将返回false 因为$stime中的13:30:01与$datetime中的最后一条有重叠。

Continue reading...

逆向思维PHP中判断是否是Unix时间戳

在php中一个有效的时间是从 1970-01-01 07:00:00 – 2038-01-19 03:14:07. 这个的
关于php中unix时间戳的产生

echo strtotime('2038-01-19 03:14:07'); // 2147454847
echo strtotime('1970-01-01 07:00:00'); // 0

官方说明:http://php.net/manual/zh/function.strtotime.php

那如何判断定一个时间戳能顺利返回正确的时间通常有两种方法
逆向思维方法

function is_timestamp($timestamp) {
    if(strtotime(date('m-d-Y H:i:s', $timestamp)) === $timestamp) {
        return $timestamp;
    } else {
        return false;
    }
}

普通的整数范围判断

$is_unixtime = ctype_digit($str) && $str <= 2147483647;

这种方法比较简洁

Continue reading...

MySQL中用GKB来让UTF-8字段中的中文按照拼音排序

UTF-8 中的中文不是按照拼音排序的,因此对于 使用 UTF-8 编码集的字段就无法按照拼音进行排序,最简单的解决方法就是转成 GBK 编码。
实例代码:

SELECT *
FROM `test`
ORDER BY CONVERT( `test`.`name`
USING GBK )
LIMIT 0 , 30

数据库表结构:

CREATE TABLE IF NOT EXISTS `test` (
`id` INT(3) NOT NULL AUTO_INCREMENT,
`name` VARCHAR(10) COLLATE utf8_bin NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB  DEFAULT CHARSET=utf8 COLLATE=utf8_bin AUTO_INCREMENT=1 ;

测试环境:Windows 2003 Standard Edition,MySQL 5.1.22-rc-community。
对于我正在使用的 Symfony 框架,因为使用了 propel 做 ORM,比较难直接操作SQL,除非使用 RAW SQL,而且使用 RAW SQL会导致很多高级特性无法使用,解决方式:$criteria->addAscendingOrderByColumn('CONVERT(' . TestPeer::NAME . ' USING GBK)');

Continue reading...



about me

  • 陶之11's Blog Name:陶之11 OICQ:14779023 Site:Pao11.com

分类

快捷入口