Discuz门户文章8个标签属性的数据库存储分析-Discuz教程下载

Discuz门户文章8个标签属性的数据库存储分析

来自版块: Discuz教程发表于: 2017-4-6 09:59:20
40168
0
如本资源下载地址失效,请点击此处进行反馈
开通本站Svip会员,全站资源免费下

Discuz不用多说,大家应该很多人都了解,国内知名社区论坛程序,为腾讯旗下康盛公司开发(被腾讯收购的),已经有十多年的发展,确实是一款优秀的产品。虽然它的主要卖点是论坛功能,但同时还有sns功能、门户功能。


随着很多中小企业也开始做互联网,也想着用自己的东西来做一个自己客户的互动平台,因此现在他们有很多使用Discuz的机会,而很多甚至就还利用门户功能,搭建自己所属行业的细分门户网站。而这里我们要讲的就是跟门户模块有关的一个细节——Discuz门户文章8个标签属性的数据存储分析。


用过Discuz门户功能的人也知道,一篇文章有8个标签,分别是原创、热点、组图 、爆料、 头条、幻灯、滚动、推荐。那么当我们进行二次开发的时候,如果要利用这8个标签进行筛选文章的时候该怎么办呢?这个时候我们就得先知道这几个标签是怎么被保存的。比如下面是我们项目中的实例:可以通过标签来筛选数据,同时在每个文章下面显示他所属的标签。


Discuz门户文章8个标签属性的数据库存储分析

Discuz门户文章8个标签属性的数据库存储分析


通过查阅数据词典我们发现,这8个标签在文章表中只有一个字段,也就是一个字段保存了标签的信息,首先我们来看下dz的数据词典的这部分:


Discuz门户文章8个标签属性的数据库存储分析

Discuz门户文章8个标签属性的数据库存储分析


那么这一个字段是怎么保存这8个标签信息的,我们可以联想到论坛帖子的高亮属性,也同样的是多个属性保存在一个字段中,那是通过进制转化来的;那么这又是什么呢?我们查看这些字段的值,也许能发现一些规律,但要我们得出一个规则那实在太难了。于是我们还是去分析源代码来倒推吧。


我们可以在source/function/function_portalcp.php中找到两个方法 article_parse_tags和article_make_tag:


  1. <font color="rgb(85, 85, 85)">function article_parse_tags($tag) {
  2.         $tag = intval($tag);
  3.         $article_tags = array();
  4.         for($i=1; $i<=8; $i++) {
  5.                 $k = pow(2, $i-1);
  6.                 $article_tags[$i] = ($tag & $k) ? 1 : 0;
  7.         }
  8.         return $article_tags;
  9. }

  10. function article_make_tag($tags) {
  11.         $tags = (array)$tags;
  12.         $tag = 0;
  13.         for($i=1; $i<=8; $i++) {
  14.                 if(!empty($tags[$i])) {
  15.                         $tag += pow(2, $i-1);
  16.                 }
  17.         }
  18.         return $tag;
  19. }</font>
复制代码

其实这两个可以理解为一组方法,前者是把数据库读取到值传送进去,依次判断,然后得到返回一个数组,而这个数组就有八个键值,每个键值即代表一个标签属性。而下面一个方法就是把编写文章的时候表单的值传送给这个方法,然后依次执行,进行2的n次方累加,得到一个十进制的值然后返回$tag,这个$tag的值最终就是保存在数据库中的。


其实以上实际上可以完成我上面提到那个例子的一部分,也就是通过取得的值然后把对应的标签显示在文章下面,当然上面的第二个方法实际上为了解释整个原理附带来的。那么还有一部分,也就是怎么通过标签来筛选呢?


这个时候我们可以参考Discuz的源代码中source/class/block/portal/block_article.php中的block_article类,这个大概就是我们进行门户文章DIY调用的一个类,我们可以知道在DIY调用中我们是可以通过标签来进行数据过滤的,那么里面就一定有现成的方法来做到这点了。在里面有这么一段代码:


  1. <font color="rgb(85, 85, 85)">if(is_array($tag)) {
  2.         $article_tags = array();
  3.         foreach($tag as $k) {
  4.                 $article_tags[$k] = 1;
  5.         }
  6.         include_once libfile('function/portalcp');
  7.         $v=article_make_tag($article_tags);
  8.         if($v > 0) {
  9.                 $wheres[] = "(at.tag & $v) = $v";
  10.         }
  11. }</font>
复制代码

上面的参数$tag就是我们要筛选的目标标签数组,比如里面同样是八个键,每个键值可能为0或1(也就是勾选了或者没有勾选),然后一次判断,得到SQL查询语句中where条件约束的数组$where,这样就实现了对数据的筛选或者说过滤。以上就是我们查看Discuz源代码获取的一些信息,也就是我们知道了其中的原理,其实大概就是用2的n次方,然后&位运算来进行判断(PHP文档)。


============================华丽的分割线============================


好了,上面进摆出了几段代码实际上已经阐述了原理,那么我们以一个实际的例子来说明。比如说一篇文章已经选中了原创和推荐两个标签,也就是第一个和最后一个:  


Discuz门户文章8个标签属性的数据库存储分析

Discuz门户文章8个标签属性的数据库存储分析


也就是这个对应的article_make_tag方法中

  1. <font color="rgb(85, 85, 85)">$tags=array(0=>1,1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,7=>1);</font>
复制代码

通过该方法返回的值$tag=2^(1-1)+2^(8-1)=2^0+2^7+1+128=129,我们可以在数据库中查看portal_article_title表中的tag字段的值是129。


再比如我们通过查询到的129这个值传入article_parse_tags作为参数可以返回$article_tags


  1. <font color="rgb(85, 85, 85)">$article_tags=array(0=>1,1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,7=>1);</font>
复制代码

反过来就是我们最开始提到例子中需要做的,也就是说已经知道了某个筛选条件,比如说是要原创的,那么对应的数组会是


  1. <font color="rgb(85, 85, 85)">$tag=array(0=>1,1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,7=>0);</font>
复制代码
同样的我们可以得到相应的约束条件。


全部评论 0

您需要登录后才可以回帖 立即登录
登录
0
0
0
返回顶部