JSON在PHP中的应用

来源:互联网 发布:筛选后的数据求和 编辑:程序博客网 时间:2024/05/22 07:03
互联网的今天,AJAX已经不是什么陌生的词汇了。说起AJAX,可能会立即想起因RSS而兴起的XML。XML的解析,恐怕已经不是什么难题了,特别是PHP5,大量的XML解析器的涌现,如最轻量级的SimpleXML。不过对于AJAX来说,XML的解析更倾向于前台Javascript的支持度。我想所有解析过XML的人,都会因树和节点而头大。不可否认,XML是很不错的数据存储方式,但是其灵活恰恰造成了其解析的困难。当然,这里所指的困难,是相对于本文的主角--JSON而言。
  JSON为何物?我就不重复概念了。通俗的说,它是一种数据的存储格式,就像PHP序列化后的字符串一样。它是一种数据描述。比如我们将一个数组序列化后存放,就可以很容易的反序列化后应用。JSON也是如此,只不过他搭建的是客户端Javascript和服务端PHP的交互桥梁。我们用PHP生成JSON后的字符串,然后把这个字符串传给前台Javascript,Javascirpt就可以很容易的将其反JSON然后应用。说通俗点,它真的很像数组。
  言归正传,如何使用JSON。PHP5.2开始内置了JSON的支持。当然,如果低于这个版本的话,那么市面上有很多PHP版本的实现,随便下一个用就OK啦。现在主要是说说PHP内置支持的JSON。很简单,两个函数:json_encodejson_decode(跟序列化很像啦)。一个编码,一个解码。先看看编码的使用:
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo $json_string;?>
  很简单的将一个数组JSON了。需要指出的是,在非UTF-8编码下,中文字符将不可被encode,结果会出来空值,所以,如果你使用gb2312编写PHP代码,那么就需要将包含中文的内容使用iconv或者mb转为UTF-8再进行json_encode,上面输出结果如下:

{"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}}
  我都说了和序列化很像,你还不信。编码后就要解码,PHP提供了相应的函数json_decode,json_decode执行后,将会得到一个对象,操作如下:

<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);$obj = json_decode($json_string);print_r($obj);?>
  访问对象内的属性会吧?$obj->name,这样子的,当然,也可以把它转位数组,方便调用啦:

$json_string = json_encode($arr);$obj = json_decode($json_string);$arr = (array) $obj;print_r($arr);
  PHP转来转去的用途不是特别大,除了缓存生成,感觉还不如直接存数组呢,不过,当你和前台交互的时候,它的作用就出来咯,下面看看我怎么用Javascript来使用这段字符:

<script type="text/javascript">var arr = {"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}};alert(arr.name)</script>
  上面中,直接将这个字符串赋给一个变量,它就变成一个Javascript数组了(专业化术语应该不叫数组,不过由于PHP的习惯问题,我就一直叫数组好了,方便理解)。这样,可以很方便的对arr进行遍历或者任意做你想做的事情了。写到这里,好像都没提到AJAX哦?是哦,联想一下,如果服务端返回的responseText用JSON过的字符串代替XML的话,前台Javascript处理起来是不是很方便呢?狗皮膏药就是这样用的。
  其实写到这里,除了数据的存储格式不太一样外,JSON和XML也没什么太大区别哦,不过下面我说的一点。虽然和XML没多大关系,不过,可以说明JSON更大范围的应用,那就是,跨域的数据调用。由于安全性问题,AJAX不支持跨域调用,这样要调用不同域名下的数据,很麻烦哦,虽然有解决方案(stone在他的讲座上提到过了代理啊什么的虽然听不懂但是知道能解决)。我写两个文件,足以展示跨域调用了。
  主调文件index.html
<script type="text/javascript">function getProfile(str) {    var arr = str;    document.getElementById('nick').innerHTML = arr.nick;}</script><body><div id="nick"></div></body><script type="text/javascript" src="http://www.openphp.cn/demo/profile.php"></script>
  被调文件profile.php
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo "getProfile($json_string)";?>
  很显然,当index.html调用profile.php时,JSON字符串生成,并作为参数传入getProfile,然后将昵称插入到div中,这样一次跨域数据交互就完成了,是不是特别简单。既然JSON这么简单易用而且好用,还等什么呢?^_^

 

多数据库支持的应用程序设计

2007年10月07日 09时32分

本站文章如未特殊注明,均为原创,严禁转载。

  以前做PHP应用,多数是单数据库数据查询和更新,顶多也是主从数据库的支持,实现起来相对简单。主从数据库的问题在于,当会话存储在数据库的时候,同步将可能出现问题,也就是说有可能出现会话的中断。所以我想在主从数据库设计上,应该将所有会话相关表进行特殊对待。即:所有的会话数据表都可以更新和查询,当一个用户访问站点的时候,即将此用户绑定到指定数据库,所有会话访问和查询操作都对此数据库进行。会话表不做同步,其他非会话类更新也从主数据库更新。这样做其实也逃脱不了会话更新时候的数据库切换,所以如果不想麻烦,还是将会话存放在文本中进行的好。
  分数据库设计,将可能从压力性能上会提升几个档次,当然单次执行效率不会比单数据库来的高的,毕竟存在着数据库切换的效率问题。分库以及主从数据库搭配是可以比较好改善数据库并发瓶颈的方案。原则:大数据量,分库;大访问量,主从。很多时候,都是这两者并行(本文不讨论cache)。
  我想,如果要实现分库以及主从关系,那么数据库服务器数量将是非常可观,在应用程序中随时切换到某一台服务器,将是非常头痛的问题,配置更换,变量名称,是不是会有一大堆呢?如何寻找更好的解决方案将是本文谈论的话题。
  首先是分库使得数据库颇多的问题。什么情况下分库?或许有些人还搞不明白为什么要分库,我就简要说一下自己的经验猜测。比如一个博客程序,一般设计是将日志存放在一张日志表中。假设是一个多用户博客,那么将会关联一个uid,如果数据量不大,这样设计是没有问题的,但是当日志量巨大,一天有几十万条日志记录录入的时候,而且访问量也比较可观的时候,我想不可能每个用户来访问日志列表,都去从这包含几千万条日志记录的数据表中去找那么几条,效率可见一斑。这个时候就该考虑到分库的问题。如何分?有一个很简单的分表方法,即,根据uid段,将日志记录在各个数据库中,当然,这个分布还是需要根据以往统计结果做出调整的,因为用户日志分布肯定不是均匀的。设置好uid段,然后根据uid索引到指定数据库配置,创建一个数据库对象即可。配置信息可能如下:

$configs['db_info']['blog'][0] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][2] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);//...还有很多
  至于选择哪一台服务器,只需要根据uid做一个简单的匹配就可以了。
  再谈到的就是主从数据库了。什么情况下使用主从数据库?比如某个名人博客,访问量相当的大,已经没有办法把他的数据再进行拆分了,这个时候就得考虑主从数据库服务器了,使用多台数据库来分流。这样要适用主从和分库,可能上面配置信息得稍微改动一下。
$configs['db_info']['blog'][0]['master'] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][0] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][1] = array(    'db_host' => '192.168.0.3',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['master'] = array(    'db_host' => '192.168.0.4',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][0] = array(    'db_host' => '192.168.0.5',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][1] = array(    'db_host' => '192.168.0.6',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][0]['master'] = array(    'db_host' => '192.168.0.7',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][1]['master'] = array(    'db_host' => '192.168.0.8',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);
  写到这里,我想都应该知道如何分表和分配你的数据库了吧,接下去我就来说一下如何轻松的读取这样的配置信息,如何将这些配置融入你的数据库驱动中。首先以单例摸式创建DB类:
<?phpif (!defined("DB_FETCH_ASSOC")) {    define("DB_FETCH_ASSOC", 1);}if (!defined("DB_FETCH_ROW")) {    define("DB_FETCH_ROW", 2);}if (!defined("DB_FETCH_ARRAY")) {    define("DB_FETCH_ARRAY", 3);}if (!defined("DB_FETCH_DEFAULT")) {    define("DB_FETCH_DEFAULT", DB_FETCH_ASSOC);}class DB {    function DB($dsn, $db_key, $p_conn, $fetch_mode) {        $this->dsn        = $dsn;        $this->db_key     = $db_key;        $this->sql        = '';        $this->sqls       = array();        $this->u_sqls     = array();        $this->q_sqls     = array();        $this->u_conn     = null;        $this->q_conn     = null;        $this->p_conn     = $p_conn;        $this->fecth_mode = $fetch_mode;        $this->query_num  = 0;        $this->update_num = 0;    }    function &init(& $dsn, $db_key, $p_conn = false, $fetch_mode = DB_FETCH_DEFAULT) {        static $db;        $db_key = explode('.', $db_key);        $db_key = "['" . implode("']['" , $db_key) . "']";        eval('$flag = isset($db' . $db_key . ');');        eval("/$db_info = /$dsn['db_info']" . $db_key . ";");        if (!$flag) {            $obj = new DB($db_info, $db_key, $p_conn, $fetch_mode);            eval('$db' . $db_key . ' = $obj;');            unset($obj);        }        return $db;    }}$db = &DB::init($configs, 'blog.0');print_r($db);?>
  从上面打印结果可以看出,blog数据库集群的第一组数据库服务器被载入到$this->dsn中了。这个下面就是简单的数据COPY的主从服务器,所以可以使用随机数来指定到某一台服务器。以下是一个简单的随机数实现:
class DB {    //....    function connectDB($type = "master") {        if ($type == "master") {            $db_host = $this->dsn["master"]["db_host"];            $db_name = $this->dsn["master"]["db_name"];            $db_user = $this->dsn["master"]["db_user"];            $db_pass = $this->dsn["master"]["db_pass"];            $this->u_conn = mysqli_connect($db_host, $db_user, $db_pass);            $this->selectDB($db_name, $this->conn);            if (!$this->u_conn) {                $message = "Update DataBase Connect False : ($db_host, $db_user, ******) !";                $this->error($message, 0);            }        } else {            if (empty( 互联网的今天,AJAX已经不是什么陌生的词汇了。说起AJAX,可能会立即想起因RSS而兴起的XML。XML的解析,恐怕已经不是什么难题了,特别是PHP5,大量的XML解析器的涌现,如最轻量级的SimpleXML。不过对于AJAX来说,XML的解析更倾向于前台Javascript的支持度。我想所有解析过XML的人,都会因树和节点而头大。不可否认,XML是很不错的数据存储方式,但是其灵活恰恰造成了其解析的困难。当然,这里所指的困难,是相对于本文的主角--JSON而言。
  JSON为何物?我就不重复概念了。通俗的说,它是一种数据的存储格式,就像PHP序列化后的字符串一样。它是一种数据描述。比如我们将一个数组序列化后存放,就可以很容易的反序列化后应用。JSON也是如此,只不过他搭建的是客户端Javascript和服务端PHP的交互桥梁。我们用PHP生成JSON后的字符串,然后把这个字符串传给前台Javascript,Javascirpt就可以很容易的将其反JSON然后应用。说通俗点,它真的很像数组。
  言归正传,如何使用JSON。PHP5.2开始内置了JSON的支持。当然,如果低于这个版本的话,那么市面上有很多PHP版本的实现,随便下一个用就OK啦。现在主要是说说PHP内置支持的JSON。很简单,两个函数:json_encodejson_decode(跟序列化很像啦)。一个编码,一个解码。先看看编码的使用:
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo $json_string;?>
  很简单的将一个数组JSON了。需要指出的是,在非UTF-8编码下,中文字符将不可被encode,结果会出来空值,所以,如果你使用gb2312编写PHP代码,那么就需要将包含中文的内容使用iconv或者mb转为UTF-8再进行json_encode,上面输出结果如下:

{"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}}
  我都说了和序列化很像,你还不信。编码后就要解码,PHP提供了相应的函数json_decode,json_decode执行后,将会得到一个对象,操作如下:

<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);$obj = json_decode($json_string);print_r($obj);?>
  访问对象内的属性会吧?$obj->name,这样子的,当然,也可以把它转位数组,方便调用啦:

$json_string = json_encode($arr);$obj = json_decode($json_string);$arr = (array) $obj;print_r($arr);
  PHP转来转去的用途不是特别大,除了缓存生成,感觉还不如直接存数组呢,不过,当你和前台交互的时候,它的作用就出来咯,下面看看我怎么用Javascript来使用这段字符:

<script type="text/javascript">var arr = {"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}};alert(arr.name)</script>
  上面中,直接将这个字符串赋给一个变量,它就变成一个Javascript数组了(专业化术语应该不叫数组,不过由于PHP的习惯问题,我就一直叫数组好了,方便理解)。这样,可以很方便的对arr进行遍历或者任意做你想做的事情了。写到这里,好像都没提到AJAX哦?是哦,联想一下,如果服务端返回的responseText用JSON过的字符串代替XML的话,前台Javascript处理起来是不是很方便呢?狗皮膏药就是这样用的。
  其实写到这里,除了数据的存储格式不太一样外,JSON和XML也没什么太大区别哦,不过下面我说的一点。虽然和XML没多大关系,不过,可以说明JSON更大范围的应用,那就是,跨域的数据调用。由于安全性问题,AJAX不支持跨域调用,这样要调用不同域名下的数据,很麻烦哦,虽然有解决方案(stone在他的讲座上提到过了代理啊什么的虽然听不懂但是知道能解决)。我写两个文件,足以展示跨域调用了。
  主调文件index.html
<script type="text/javascript">function getProfile(str) {    var arr = str;    document.getElementById('nick').innerHTML = arr.nick;}</script><body><div id="nick"></div></body><script type="text/javascript" src="http://www.openphp.cn/demo/profile.php"></script>
  被调文件profile.php
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo "getProfile($json_string)";?>
  很显然,当index.html调用profile.php时,JSON字符串生成,并作为参数传入getProfile,然后将昵称插入到div中,这样一次跨域数据交互就完成了,是不是特别简单。既然JSON这么简单易用而且好用,还等什么呢?^_^

 

多数据库支持的应用程序设计

2007年10月07日 09时32分

本站文章如未特殊注明,均为原创,严禁转载。

  以前做PHP应用,多数是单数据库数据查询和更新,顶多也是主从数据库的支持,实现起来相对简单。主从数据库的问题在于,当会话存储在数据库的时候,同步将可能出现问题,也就是说有可能出现会话的中断。所以我想在主从数据库设计上,应该将所有会话相关表进行特殊对待。即:所有的会话数据表都可以更新和查询,当一个用户访问站点的时候,即将此用户绑定到指定数据库,所有会话访问和查询操作都对此数据库进行。会话表不做同步,其他非会话类更新也从主数据库更新。这样做其实也逃脱不了会话更新时候的数据库切换,所以如果不想麻烦,还是将会话存放在文本中进行的好。
  分数据库设计,将可能从压力性能上会提升几个档次,当然单次执行效率不会比单数据库来的高的,毕竟存在着数据库切换的效率问题。分库以及主从数据库搭配是可以比较好改善数据库并发瓶颈的方案。原则:大数据量,分库;大访问量,主从。很多时候,都是这两者并行(本文不讨论cache)。
  我想,如果要实现分库以及主从关系,那么数据库服务器数量将是非常可观,在应用程序中随时切换到某一台服务器,将是非常头痛的问题,配置更换,变量名称,是不是会有一大堆呢?如何寻找更好的解决方案将是本文谈论的话题。
  首先是分库使得数据库颇多的问题。什么情况下分库?或许有些人还搞不明白为什么要分库,我就简要说一下自己的经验猜测。比如一个博客程序,一般设计是将日志存放在一张日志表中。假设是一个多用户博客,那么将会关联一个uid,如果数据量不大,这样设计是没有问题的,但是当日志量巨大,一天有几十万条日志记录录入的时候,而且访问量也比较可观的时候,我想不可能每个用户来访问日志列表,都去从这包含几千万条日志记录的数据表中去找那么几条,效率可见一斑。这个时候就该考虑到分库的问题。如何分?有一个很简单的分表方法,即,根据uid段,将日志记录在各个数据库中,当然,这个分布还是需要根据以往统计结果做出调整的,因为用户日志分布肯定不是均匀的。设置好uid段,然后根据uid索引到指定数据库配置,创建一个数据库对象即可。配置信息可能如下:

$configs['db_info']['blog'][0] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][2] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);//...还有很多
  至于选择哪一台服务器,只需要根据uid做一个简单的匹配就可以了。
  再谈到的就是主从数据库了。什么情况下使用主从数据库?比如某个名人博客,访问量相当的大,已经没有办法把他的数据再进行拆分了,这个时候就得考虑主从数据库服务器了,使用多台数据库来分流。这样要适用主从和分库,可能上面配置信息得稍微改动一下。
$configs['db_info']['blog'][0]['master'] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][0] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][1] = array(    'db_host' => '192.168.0.3',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['master'] = array(    'db_host' => '192.168.0.4',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][0] = array(    'db_host' => '192.168.0.5',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][1] = array(    'db_host' => '192.168.0.6',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][0]['master'] = array(    'db_host' => '192.168.0.7',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][1]['master'] = array(    'db_host' => '192.168.0.8',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);
  写到这里,我想都应该知道如何分表和分配你的数据库了吧,接下去我就来说一下如何轻松的读取这样的配置信息,如何将这些配置融入你的数据库驱动中。首先以单例摸式创建DB类:
<?phpif (!defined("DB_FETCH_ASSOC")) {    define("DB_FETCH_ASSOC", 1);}if (!defined("DB_FETCH_ROW")) {    define("DB_FETCH_ROW", 2);}if (!defined("DB_FETCH_ARRAY")) {    define("DB_FETCH_ARRAY", 3);}if (!defined("DB_FETCH_DEFAULT")) {    define("DB_FETCH_DEFAULT", DB_FETCH_ASSOC);}class DB {    function DB($dsn, $db_key, $p_conn, $fetch_mode) {        $this->dsn        = $dsn;        $this->db_key     = $db_key;        $this->sql        = '';        $this->sqls       = array();        $this->u_sqls     = array();        $this->q_sqls     = array();        $this->u_conn     = null;        $this->q_conn     = null;        $this->p_conn     = $p_conn;        $this->fecth_mode = $fetch_mode;        $this->query_num  = 0;        $this->update_num = 0;    }    function &init(& $dsn, $db_key, $p_conn = false, $fetch_mode = DB_FETCH_DEFAULT) {        static $db;        $db_key = explode('.', $db_key);        $db_key = "['" . implode("']['" , $db_key) . "']";        eval('$flag = isset($db' . $db_key . ');');        eval("/$db_info = /$dsn['db_info']" . $db_key . ";");        if (!$flag) {            $obj = new DB($db_info, $db_key, $p_conn, $fetch_mode);            eval('$db' . $db_key . ' = $obj;');            unset($obj);        }        return $db;    }}$db = &DB::init($configs, 'blog.0');print_r($db);?>
  从上面打印结果可以看出,blog数据库集群的第一组数据库服务器被载入到$this->dsn中了。这个下面就是简单的数据COPY的主从服务器,所以可以使用随机数来指定到某一台服务器。以下是一个简单的随机数实现:
___FCKpd___10
  至此,基本框架已经出来了,来看看调用方法:
<?php// 连接到第一组会话服务器$db = &DB::init($configs, 'session.0');//  执行一次查询$db['session'][0]->query('SELECT ...');//  再次连接BLOG服务器$db = &DB::init($configs, 'blog.1');//  执行一次更新$db['blog'][1]->update('UPDATE ...');//  再次调用会话更新$db['session'][0]->update('INSERT ...');?> 
COOKIE[ 互联网的今天,AJAX已经不是什么陌生的词汇了。说起AJAX,可能会立即想起因RSS而兴起的XML。XML的解析,恐怕已经不是什么难题了,特别是PHP5,大量的XML解析器的涌现,如最轻量级的SimpleXML。不过对于AJAX来说,XML的解析更倾向于前台Javascript的支持度。我想所有解析过XML的人,都会因树和节点而头大。不可否认,XML是很不错的数据存储方式,但是其灵活恰恰造成了其解析的困难。当然,这里所指的困难,是相对于本文的主角--JSON而言。
  JSON为何物?我就不重复概念了。通俗的说,它是一种数据的存储格式,就像PHP序列化后的字符串一样。它是一种数据描述。比如我们将一个数组序列化后存放,就可以很容易的反序列化后应用。JSON也是如此,只不过他搭建的是客户端Javascript和服务端PHP的交互桥梁。我们用PHP生成JSON后的字符串,然后把这个字符串传给前台Javascript,Javascirpt就可以很容易的将其反JSON然后应用。说通俗点,它真的很像数组。
  言归正传,如何使用JSON。PHP5.2开始内置了JSON的支持。当然,如果低于这个版本的话,那么市面上有很多PHP版本的实现,随便下一个用就OK啦。现在主要是说说PHP内置支持的JSON。很简单,两个函数:json_encodejson_decode(跟序列化很像啦)。一个编码,一个解码。先看看编码的使用:
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo $json_string;?>
  很简单的将一个数组JSON了。需要指出的是,在非UTF-8编码下,中文字符将不可被encode,结果会出来空值,所以,如果你使用gb2312编写PHP代码,那么就需要将包含中文的内容使用iconv或者mb转为UTF-8再进行json_encode,上面输出结果如下:

{"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}}
  我都说了和序列化很像,你还不信。编码后就要解码,PHP提供了相应的函数json_decode,json_decode执行后,将会得到一个对象,操作如下:

<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);$obj = json_decode($json_string);print_r($obj);?>
  访问对象内的属性会吧?$obj->name,这样子的,当然,也可以把它转位数组,方便调用啦:

$json_string = json_encode($arr);$obj = json_decode($json_string);$arr = (array) $obj;print_r($arr);
  PHP转来转去的用途不是特别大,除了缓存生成,感觉还不如直接存数组呢,不过,当你和前台交互的时候,它的作用就出来咯,下面看看我怎么用Javascript来使用这段字符:

<script type="text/javascript">var arr = {"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}};alert(arr.name)</script>
  上面中,直接将这个字符串赋给一个变量,它就变成一个Javascript数组了(专业化术语应该不叫数组,不过由于PHP的习惯问题,我就一直叫数组好了,方便理解)。这样,可以很方便的对arr进行遍历或者任意做你想做的事情了。写到这里,好像都没提到AJAX哦?是哦,联想一下,如果服务端返回的responseText用JSON过的字符串代替XML的话,前台Javascript处理起来是不是很方便呢?狗皮膏药就是这样用的。
  其实写到这里,除了数据的存储格式不太一样外,JSON和XML也没什么太大区别哦,不过下面我说的一点。虽然和XML没多大关系,不过,可以说明JSON更大范围的应用,那就是,跨域的数据调用。由于安全性问题,AJAX不支持跨域调用,这样要调用不同域名下的数据,很麻烦哦,虽然有解决方案(stone在他的讲座上提到过了代理啊什么的虽然听不懂但是知道能解决)。我写两个文件,足以展示跨域调用了。
  主调文件index.html
<script type="text/javascript">function getProfile(str) {    var arr = str;    document.getElementById('nick').innerHTML = arr.nick;}</script><body><div id="nick"></div></body><script type="text/javascript" src="http://www.openphp.cn/demo/profile.php"></script>
  被调文件profile.php
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo "getProfile($json_string)";?>
  很显然,当index.html调用profile.php时,JSON字符串生成,并作为参数传入getProfile,然后将昵称插入到div中,这样一次跨域数据交互就完成了,是不是特别简单。既然JSON这么简单易用而且好用,还等什么呢?^_^

 

多数据库支持的应用程序设计

2007年10月07日 09时32分

本站文章如未特殊注明,均为原创,严禁转载。

  以前做PHP应用,多数是单数据库数据查询和更新,顶多也是主从数据库的支持,实现起来相对简单。主从数据库的问题在于,当会话存储在数据库的时候,同步将可能出现问题,也就是说有可能出现会话的中断。所以我想在主从数据库设计上,应该将所有会话相关表进行特殊对待。即:所有的会话数据表都可以更新和查询,当一个用户访问站点的时候,即将此用户绑定到指定数据库,所有会话访问和查询操作都对此数据库进行。会话表不做同步,其他非会话类更新也从主数据库更新。这样做其实也逃脱不了会话更新时候的数据库切换,所以如果不想麻烦,还是将会话存放在文本中进行的好。
  分数据库设计,将可能从压力性能上会提升几个档次,当然单次执行效率不会比单数据库来的高的,毕竟存在着数据库切换的效率问题。分库以及主从数据库搭配是可以比较好改善数据库并发瓶颈的方案。原则:大数据量,分库;大访问量,主从。很多时候,都是这两者并行(本文不讨论cache)。
  我想,如果要实现分库以及主从关系,那么数据库服务器数量将是非常可观,在应用程序中随时切换到某一台服务器,将是非常头痛的问题,配置更换,变量名称,是不是会有一大堆呢?如何寻找更好的解决方案将是本文谈论的话题。
  首先是分库使得数据库颇多的问题。什么情况下分库?或许有些人还搞不明白为什么要分库,我就简要说一下自己的经验猜测。比如一个博客程序,一般设计是将日志存放在一张日志表中。假设是一个多用户博客,那么将会关联一个uid,如果数据量不大,这样设计是没有问题的,但是当日志量巨大,一天有几十万条日志记录录入的时候,而且访问量也比较可观的时候,我想不可能每个用户来访问日志列表,都去从这包含几千万条日志记录的数据表中去找那么几条,效率可见一斑。这个时候就该考虑到分库的问题。如何分?有一个很简单的分表方法,即,根据uid段,将日志记录在各个数据库中,当然,这个分布还是需要根据以往统计结果做出调整的,因为用户日志分布肯定不是均匀的。设置好uid段,然后根据uid索引到指定数据库配置,创建一个数据库对象即可。配置信息可能如下:

$configs['db_info']['blog'][0] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][2] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);//...还有很多
  至于选择哪一台服务器,只需要根据uid做一个简单的匹配就可以了。
  再谈到的就是主从数据库了。什么情况下使用主从数据库?比如某个名人博客,访问量相当的大,已经没有办法把他的数据再进行拆分了,这个时候就得考虑主从数据库服务器了,使用多台数据库来分流。这样要适用主从和分库,可能上面配置信息得稍微改动一下。
$configs['db_info']['blog'][0]['master'] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][0] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][1] = array(    'db_host' => '192.168.0.3',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['master'] = array(    'db_host' => '192.168.0.4',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][0] = array(    'db_host' => '192.168.0.5',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][1] = array(    'db_host' => '192.168.0.6',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][0]['master'] = array(    'db_host' => '192.168.0.7',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][1]['master'] = array(    'db_host' => '192.168.0.8',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);
  写到这里,我想都应该知道如何分表和分配你的数据库了吧,接下去我就来说一下如何轻松的读取这样的配置信息,如何将这些配置融入你的数据库驱动中。首先以单例摸式创建DB类:
<?phpif (!defined("DB_FETCH_ASSOC")) {    define("DB_FETCH_ASSOC", 1);}if (!defined("DB_FETCH_ROW")) {    define("DB_FETCH_ROW", 2);}if (!defined("DB_FETCH_ARRAY")) {    define("DB_FETCH_ARRAY", 3);}if (!defined("DB_FETCH_DEFAULT")) {    define("DB_FETCH_DEFAULT", DB_FETCH_ASSOC);}class DB {    function DB($dsn, $db_key, $p_conn, $fetch_mode) {        $this->dsn        = $dsn;        $this->db_key     = $db_key;        $this->sql        = '';        $this->sqls       = array();        $this->u_sqls     = array();        $this->q_sqls     = array();        $this->u_conn     = null;        $this->q_conn     = null;        $this->p_conn     = $p_conn;        $this->fecth_mode = $fetch_mode;        $this->query_num  = 0;        $this->update_num = 0;    }    function &init(& $dsn, $db_key, $p_conn = false, $fetch_mode = DB_FETCH_DEFAULT) {        static $db;        $db_key = explode('.', $db_key);        $db_key = "['" . implode("']['" , $db_key) . "']";        eval('$flag = isset($db' . $db_key . ');');        eval("/$db_info = /$dsn['db_info']" . $db_key . ";");        if (!$flag) {            $obj = new DB($db_info, $db_key, $p_conn, $fetch_mode);            eval('$db' . $db_key . ' = $obj;');            unset($obj);        }        return $db;    }}$db = &DB::init($configs, 'blog.0');print_r($db);?>
  从上面打印结果可以看出,blog数据库集群的第一组数据库服务器被载入到$this->dsn中了。这个下面就是简单的数据COPY的主从服务器,所以可以使用随机数来指定到某一台服务器。以下是一个简单的随机数实现:
___FCKpd___10
  至此,基本框架已经出来了,来看看调用方法:
___FCKpd___11
configs['cookie_prefix'] . 'db_no'])) { $db_no = array_rand($this->dsn["db_info"]["slave"]); } else { $db_no =  互联网的今天,AJAX已经不是什么陌生的词汇了。说起AJAX,可能会立即想起因RSS而兴起的XML。XML的解析,恐怕已经不是什么难题了,特别是PHP5,大量的XML解析器的涌现,如最轻量级的SimpleXML。不过对于AJAX来说,XML的解析更倾向于前台Javascript的支持度。我想所有解析过XML的人,都会因树和节点而头大。不可否认,XML是很不错的数据存储方式,但是其灵活恰恰造成了其解析的困难。当然,这里所指的困难,是相对于本文的主角--JSON而言。
  JSON为何物?我就不重复概念了。通俗的说,它是一种数据的存储格式,就像PHP序列化后的字符串一样。它是一种数据描述。比如我们将一个数组序列化后存放,就可以很容易的反序列化后应用。JSON也是如此,只不过他搭建的是客户端Javascript和服务端PHP的交互桥梁。我们用PHP生成JSON后的字符串,然后把这个字符串传给前台Javascript,Javascirpt就可以很容易的将其反JSON然后应用。说通俗点,它真的很像数组。
  言归正传,如何使用JSON。PHP5.2开始内置了JSON的支持。当然,如果低于这个版本的话,那么市面上有很多PHP版本的实现,随便下一个用就OK啦。现在主要是说说PHP内置支持的JSON。很简单,两个函数:json_encodejson_decode(跟序列化很像啦)。一个编码,一个解码。先看看编码的使用:
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo $json_string;?>
  很简单的将一个数组JSON了。需要指出的是,在非UTF-8编码下,中文字符将不可被encode,结果会出来空值,所以,如果你使用gb2312编写PHP代码,那么就需要将包含中文的内容使用iconv或者mb转为UTF-8再进行json_encode,上面输出结果如下:

{"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}}
  我都说了和序列化很像,你还不信。编码后就要解码,PHP提供了相应的函数json_decode,json_decode执行后,将会得到一个对象,操作如下:

<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);$obj = json_decode($json_string);print_r($obj);?>
  访问对象内的属性会吧?$obj->name,这样子的,当然,也可以把它转位数组,方便调用啦:

$json_string = json_encode($arr);$obj = json_decode($json_string);$arr = (array) $obj;print_r($arr);
  PHP转来转去的用途不是特别大,除了缓存生成,感觉还不如直接存数组呢,不过,当你和前台交互的时候,它的作用就出来咯,下面看看我怎么用Javascript来使用这段字符:

<script type="text/javascript">var arr = {"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}};alert(arr.name)</script>
  上面中,直接将这个字符串赋给一个变量,它就变成一个Javascript数组了(专业化术语应该不叫数组,不过由于PHP的习惯问题,我就一直叫数组好了,方便理解)。这样,可以很方便的对arr进行遍历或者任意做你想做的事情了。写到这里,好像都没提到AJAX哦?是哦,联想一下,如果服务端返回的responseText用JSON过的字符串代替XML的话,前台Javascript处理起来是不是很方便呢?狗皮膏药就是这样用的。
  其实写到这里,除了数据的存储格式不太一样外,JSON和XML也没什么太大区别哦,不过下面我说的一点。虽然和XML没多大关系,不过,可以说明JSON更大范围的应用,那就是,跨域的数据调用。由于安全性问题,AJAX不支持跨域调用,这样要调用不同域名下的数据,很麻烦哦,虽然有解决方案(stone在他的讲座上提到过了代理啊什么的虽然听不懂但是知道能解决)。我写两个文件,足以展示跨域调用了。
  主调文件index.html
<script type="text/javascript">function getProfile(str) {    var arr = str;    document.getElementById('nick').innerHTML = arr.nick;}</script><body><div id="nick"></div></body><script type="text/javascript" src="http://www.openphp.cn/demo/profile.php"></script>
  被调文件profile.php
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo "getProfile($json_string)";?>
  很显然,当index.html调用profile.php时,JSON字符串生成,并作为参数传入getProfile,然后将昵称插入到div中,这样一次跨域数据交互就完成了,是不是特别简单。既然JSON这么简单易用而且好用,还等什么呢?^_^

 

多数据库支持的应用程序设计

2007年10月07日 09时32分

本站文章如未特殊注明,均为原创,严禁转载。

  以前做PHP应用,多数是单数据库数据查询和更新,顶多也是主从数据库的支持,实现起来相对简单。主从数据库的问题在于,当会话存储在数据库的时候,同步将可能出现问题,也就是说有可能出现会话的中断。所以我想在主从数据库设计上,应该将所有会话相关表进行特殊对待。即:所有的会话数据表都可以更新和查询,当一个用户访问站点的时候,即将此用户绑定到指定数据库,所有会话访问和查询操作都对此数据库进行。会话表不做同步,其他非会话类更新也从主数据库更新。这样做其实也逃脱不了会话更新时候的数据库切换,所以如果不想麻烦,还是将会话存放在文本中进行的好。
  分数据库设计,将可能从压力性能上会提升几个档次,当然单次执行效率不会比单数据库来的高的,毕竟存在着数据库切换的效率问题。分库以及主从数据库搭配是可以比较好改善数据库并发瓶颈的方案。原则:大数据量,分库;大访问量,主从。很多时候,都是这两者并行(本文不讨论cache)。
  我想,如果要实现分库以及主从关系,那么数据库服务器数量将是非常可观,在应用程序中随时切换到某一台服务器,将是非常头痛的问题,配置更换,变量名称,是不是会有一大堆呢?如何寻找更好的解决方案将是本文谈论的话题。
  首先是分库使得数据库颇多的问题。什么情况下分库?或许有些人还搞不明白为什么要分库,我就简要说一下自己的经验猜测。比如一个博客程序,一般设计是将日志存放在一张日志表中。假设是一个多用户博客,那么将会关联一个uid,如果数据量不大,这样设计是没有问题的,但是当日志量巨大,一天有几十万条日志记录录入的时候,而且访问量也比较可观的时候,我想不可能每个用户来访问日志列表,都去从这包含几千万条日志记录的数据表中去找那么几条,效率可见一斑。这个时候就该考虑到分库的问题。如何分?有一个很简单的分表方法,即,根据uid段,将日志记录在各个数据库中,当然,这个分布还是需要根据以往统计结果做出调整的,因为用户日志分布肯定不是均匀的。设置好uid段,然后根据uid索引到指定数据库配置,创建一个数据库对象即可。配置信息可能如下:

$configs['db_info']['blog'][0] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][2] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);//...还有很多
  至于选择哪一台服务器,只需要根据uid做一个简单的匹配就可以了。
  再谈到的就是主从数据库了。什么情况下使用主从数据库?比如某个名人博客,访问量相当的大,已经没有办法把他的数据再进行拆分了,这个时候就得考虑主从数据库服务器了,使用多台数据库来分流。这样要适用主从和分库,可能上面配置信息得稍微改动一下。
$configs['db_info']['blog'][0]['master'] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][0] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][1] = array(    'db_host' => '192.168.0.3',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['master'] = array(    'db_host' => '192.168.0.4',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][0] = array(    'db_host' => '192.168.0.5',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][1] = array(    'db_host' => '192.168.0.6',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][0]['master'] = array(    'db_host' => '192.168.0.7',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][1]['master'] = array(    'db_host' => '192.168.0.8',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);
  写到这里,我想都应该知道如何分表和分配你的数据库了吧,接下去我就来说一下如何轻松的读取这样的配置信息,如何将这些配置融入你的数据库驱动中。首先以单例摸式创建DB类:
<?phpif (!defined("DB_FETCH_ASSOC")) {    define("DB_FETCH_ASSOC", 1);}if (!defined("DB_FETCH_ROW")) {    define("DB_FETCH_ROW", 2);}if (!defined("DB_FETCH_ARRAY")) {    define("DB_FETCH_ARRAY", 3);}if (!defined("DB_FETCH_DEFAULT")) {    define("DB_FETCH_DEFAULT", DB_FETCH_ASSOC);}class DB {    function DB($dsn, $db_key, $p_conn, $fetch_mode) {        $this->dsn        = $dsn;        $this->db_key     = $db_key;        $this->sql        = '';        $this->sqls       = array();        $this->u_sqls     = array();        $this->q_sqls     = array();        $this->u_conn     = null;        $this->q_conn     = null;        $this->p_conn     = $p_conn;        $this->fecth_mode = $fetch_mode;        $this->query_num  = 0;        $this->update_num = 0;    }    function &init(& $dsn, $db_key, $p_conn = false, $fetch_mode = DB_FETCH_DEFAULT) {        static $db;        $db_key = explode('.', $db_key);        $db_key = "['" . implode("']['" , $db_key) . "']";        eval('$flag = isset($db' . $db_key . ');');        eval("/$db_info = /$dsn['db_info']" . $db_key . ";");        if (!$flag) {            $obj = new DB($db_info, $db_key, $p_conn, $fetch_mode);            eval('$db' . $db_key . ' = $obj;');            unset($obj);        }        return $db;    }}$db = &DB::init($configs, 'blog.0');print_r($db);?>
  从上面打印结果可以看出,blog数据库集群的第一组数据库服务器被载入到$this->dsn中了。这个下面就是简单的数据COPY的主从服务器,所以可以使用随机数来指定到某一台服务器。以下是一个简单的随机数实现:
___FCKpd___10
  至此,基本框架已经出来了,来看看调用方法:
___FCKpd___11
COOKIE[ 互联网的今天,AJAX已经不是什么陌生的词汇了。说起AJAX,可能会立即想起因RSS而兴起的XML。XML的解析,恐怕已经不是什么难题了,特别是PHP5,大量的XML解析器的涌现,如最轻量级的SimpleXML。不过对于AJAX来说,XML的解析更倾向于前台Javascript的支持度。我想所有解析过XML的人,都会因树和节点而头大。不可否认,XML是很不错的数据存储方式,但是其灵活恰恰造成了其解析的困难。当然,这里所指的困难,是相对于本文的主角--JSON而言。
  JSON为何物?我就不重复概念了。通俗的说,它是一种数据的存储格式,就像PHP序列化后的字符串一样。它是一种数据描述。比如我们将一个数组序列化后存放,就可以很容易的反序列化后应用。JSON也是如此,只不过他搭建的是客户端Javascript和服务端PHP的交互桥梁。我们用PHP生成JSON后的字符串,然后把这个字符串传给前台Javascript,Javascirpt就可以很容易的将其反JSON然后应用。说通俗点,它真的很像数组。
  言归正传,如何使用JSON。PHP5.2开始内置了JSON的支持。当然,如果低于这个版本的话,那么市面上有很多PHP版本的实现,随便下一个用就OK啦。现在主要是说说PHP内置支持的JSON。很简单,两个函数:json_encodejson_decode(跟序列化很像啦)。一个编码,一个解码。先看看编码的使用:
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo $json_string;?>
  很简单的将一个数组JSON了。需要指出的是,在非UTF-8编码下,中文字符将不可被encode,结果会出来空值,所以,如果你使用gb2312编写PHP代码,那么就需要将包含中文的内容使用iconv或者mb转为UTF-8再进行json_encode,上面输出结果如下:

{"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}}
  我都说了和序列化很像,你还不信。编码后就要解码,PHP提供了相应的函数json_decode,json_decode执行后,将会得到一个对象,操作如下:

<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);$obj = json_decode($json_string);print_r($obj);?>
  访问对象内的属性会吧?$obj->name,这样子的,当然,也可以把它转位数组,方便调用啦:

$json_string = json_encode($arr);$obj = json_decode($json_string);$arr = (array) $obj;print_r($arr);
  PHP转来转去的用途不是特别大,除了缓存生成,感觉还不如直接存数组呢,不过,当你和前台交互的时候,它的作用就出来咯,下面看看我怎么用Javascript来使用这段字符:

<script type="text/javascript">var arr = {"name":"/u9648/u6bc5/u946b","nick":"/u6df1/u7a7a","contact":{"email":"shenkong at qq dot com","website":"http:////www.chenyixin.com"}};alert(arr.name)</script>
  上面中,直接将这个字符串赋给一个变量,它就变成一个Javascript数组了(专业化术语应该不叫数组,不过由于PHP的习惯问题,我就一直叫数组好了,方便理解)。这样,可以很方便的对arr进行遍历或者任意做你想做的事情了。写到这里,好像都没提到AJAX哦?是哦,联想一下,如果服务端返回的responseText用JSON过的字符串代替XML的话,前台Javascript处理起来是不是很方便呢?狗皮膏药就是这样用的。
  其实写到这里,除了数据的存储格式不太一样外,JSON和XML也没什么太大区别哦,不过下面我说的一点。虽然和XML没多大关系,不过,可以说明JSON更大范围的应用,那就是,跨域的数据调用。由于安全性问题,AJAX不支持跨域调用,这样要调用不同域名下的数据,很麻烦哦,虽然有解决方案(stone在他的讲座上提到过了代理啊什么的虽然听不懂但是知道能解决)。我写两个文件,足以展示跨域调用了。
  主调文件index.html
<script type="text/javascript">function getProfile(str) {    var arr = str;    document.getElementById('nick').innerHTML = arr.nick;}</script><body><div id="nick"></div></body><script type="text/javascript" src="http://www.openphp.cn/demo/profile.php"></script>
  被调文件profile.php
<?php$arr = array(    'name' => '陈毅鑫',    'nick' => '深空',    'contact' => array(        'email' => 'shenkong at qq dot com',        'website' => 'http://www.chenyixin.com',    ));$json_string = json_encode($arr);echo "getProfile($json_string)";?>
  很显然,当index.html调用profile.php时,JSON字符串生成,并作为参数传入getProfile,然后将昵称插入到div中,这样一次跨域数据交互就完成了,是不是特别简单。既然JSON这么简单易用而且好用,还等什么呢?^_^

 

多数据库支持的应用程序设计

2007年10月07日 09时32分

本站文章如未特殊注明,均为原创,严禁转载。

  以前做PHP应用,多数是单数据库数据查询和更新,顶多也是主从数据库的支持,实现起来相对简单。主从数据库的问题在于,当会话存储在数据库的时候,同步将可能出现问题,也就是说有可能出现会话的中断。所以我想在主从数据库设计上,应该将所有会话相关表进行特殊对待。即:所有的会话数据表都可以更新和查询,当一个用户访问站点的时候,即将此用户绑定到指定数据库,所有会话访问和查询操作都对此数据库进行。会话表不做同步,其他非会话类更新也从主数据库更新。这样做其实也逃脱不了会话更新时候的数据库切换,所以如果不想麻烦,还是将会话存放在文本中进行的好。
  分数据库设计,将可能从压力性能上会提升几个档次,当然单次执行效率不会比单数据库来的高的,毕竟存在着数据库切换的效率问题。分库以及主从数据库搭配是可以比较好改善数据库并发瓶颈的方案。原则:大数据量,分库;大访问量,主从。很多时候,都是这两者并行(本文不讨论cache)。
  我想,如果要实现分库以及主从关系,那么数据库服务器数量将是非常可观,在应用程序中随时切换到某一台服务器,将是非常头痛的问题,配置更换,变量名称,是不是会有一大堆呢?如何寻找更好的解决方案将是本文谈论的话题。
  首先是分库使得数据库颇多的问题。什么情况下分库?或许有些人还搞不明白为什么要分库,我就简要说一下自己的经验猜测。比如一个博客程序,一般设计是将日志存放在一张日志表中。假设是一个多用户博客,那么将会关联一个uid,如果数据量不大,这样设计是没有问题的,但是当日志量巨大,一天有几十万条日志记录录入的时候,而且访问量也比较可观的时候,我想不可能每个用户来访问日志列表,都去从这包含几千万条日志记录的数据表中去找那么几条,效率可见一斑。这个时候就该考虑到分库的问题。如何分?有一个很简单的分表方法,即,根据uid段,将日志记录在各个数据库中,当然,这个分布还是需要根据以往统计结果做出调整的,因为用户日志分布肯定不是均匀的。设置好uid段,然后根据uid索引到指定数据库配置,创建一个数据库对象即可。配置信息可能如下:

$configs['db_info']['blog'][0] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][2] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);//...还有很多
  至于选择哪一台服务器,只需要根据uid做一个简单的匹配就可以了。
  再谈到的就是主从数据库了。什么情况下使用主从数据库?比如某个名人博客,访问量相当的大,已经没有办法把他的数据再进行拆分了,这个时候就得考虑主从数据库服务器了,使用多台数据库来分流。这样要适用主从和分库,可能上面配置信息得稍微改动一下。
$configs['db_info']['blog'][0]['master'] = array(    'db_host' => '192.168.0.1',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][0] = array(    'db_host' => '192.168.0.2',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][0]['slave'][1] = array(    'db_host' => '192.168.0.3',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['master'] = array(    'db_host' => '192.168.0.4',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][0] = array(    'db_host' => '192.168.0.5',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['blog'][1]['slave'][1] = array(    'db_host' => '192.168.0.6',    'db_name' => 'blog',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][0]['master'] = array(    'db_host' => '192.168.0.7',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);$configs['db_info']['session'][1]['master'] = array(    'db_host' => '192.168.0.8',    'db_name' => 'session',    'db_user' => 'root',    'db_pass' => '',);
  写到这里,我想都应该知道如何分表和分配你的数据库了吧,接下去我就来说一下如何轻松的读取这样的配置信息,如何将这些配置融入你的数据库驱动中。首先以单例摸式创建DB类:
<?phpif (!defined("DB_FETCH_ASSOC")) {    define("DB_FETCH_ASSOC", 1);}if (!defined("DB_FETCH_ROW")) {    define("DB_FETCH_ROW", 2);}if (!defined("DB_FETCH_ARRAY")) {    define("DB_FETCH_ARRAY", 3);}if (!defined("DB_FETCH_DEFAULT")) {    define("DB_FETCH_DEFAULT", DB_FETCH_ASSOC);}class DB {    function DB($dsn, $db_key, $p_conn, $fetch_mode) {        $this->dsn        = $dsn;        $this->db_key     = $db_key;        $this->sql        = '';        $this->sqls       = array();        $this->u_sqls     = array();        $this->q_sqls     = array();        $this->u_conn     = null;        $this->q_conn     = null;        $this->p_conn     = $p_conn;        $this->fecth_mode = $fetch_mode;        $this->query_num  = 0;        $this->update_num = 0;    }    function &init(& $dsn, $db_key, $p_conn = false, $fetch_mode = DB_FETCH_DEFAULT) {        static $db;        $db_key = explode('.', $db_key);        $db_key = "['" . implode("']['" , $db_key) . "']";        eval('$flag = isset($db' . $db_key . ');');        eval("/$db_info = /$dsn['db_info']" . $db_key . ";");        if (!$flag) {            $obj = new DB($db_info, $db_key, $p_conn, $fetch_mode);            eval('$db' . $db_key . ' = $obj;');            unset($obj);        }        return $db;    }}$db = &DB::init($configs, 'blog.0');print_r($db);?>
  从上面打印结果可以看出,blog数据库集群的第一组数据库服务器被载入到$this->dsn中了。这个下面就是简单的数据COPY的主从服务器,所以可以使用随机数来指定到某一台服务器。以下是一个简单的随机数实现:
___FCKpd___10
  至此,基本框架已经出来了,来看看调用方法:
___FCKpd___11
configs['cookie_prefix'] . 'db_no']; } $db_info = $this->dsn["slave"][$db_no]; $db_host = $db_info["db_host"]; $db_name = $db_info["db_name"]; $db_user = $db_info["db_user"]; $db_pass = $db_info["db_pass"]; $this->q_conn = mysqli_connect($db_host, $db_user, $db_pass); if (!$this->q_conn) { if (!$this->u_conn) { $this->connectDB(); } $this->q_conn = $this->u_conn; if (!$this->q_conn) { $message = "Query DataBase Connect False : ($db_host, $db_user, ******) !"; $this->error($message, 0); } } else { $this->selectDB($db_name, $this->q_conn); } } } function selectDB($db_name, $conn) { if ($db_name != null) { if(! mysqli_select_db($conn, $db_name)) { $code = mysqli_errno($conn); $message = mysqli_error($conn); $this->error($message, $code); } return true; } return false; } function query($sql, $limit = 1, $quick = false) { if ($limit != null) { $sql = $sql . " LIMIT " . $limit; } $this->sqls[] = $sql; $this->q_sqls[] = $sql; $this->sql = $sql; if (empty($this->q_conn)) { $this->connectDB("slave"); } $this->qrs = mysqli_query($this->q_conn, $sql, $quick ? MYSQLI_USE_RESULT : MYSQLI_STORE_RESULT); if (!$this->qrs) { $code = mysqli_errno($this->q_conn); $message = mysqli_error($this->q_conn); $this->error($message, $code); } $this->query_num++; return $this->qrs; } function update($sql) { $this->sql = $sql; $this->sqls[] = $this->sql; $this->u_sqls[] = $this->sql; if ($this->u_conn == null) { $this->connectDB("master"); } $this->urs = mysqli_query($this->u_conn, $sql, MYSQLI_USE_RESULT); $this->update_num++; if (!$this->urs) { return false; } else { return true; } }}
  至此,基本框架已经出来了,来看看调用方法:
___FCKpd___11
原创粉丝点击