扫二维码与项目经理沟通
我们在微信上24小时期待你的声音
解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流
Discuz不用多说,大家应该很多人都了解,国内知名社区论坛程序,为腾讯旗下康盛公司开发(被腾讯收购的),已经有十多年的发展,确实是一款优秀的产品。虽然它的主要卖点是论坛功能,但同时还有sns功能、门户功能。
随着很多中小企业也开始做互联网,也想着用自己的东西来做一个自己客户的互动平台,因此现在他们有很多使用Discuz的机会,而很多甚至就还利用门户功能,搭建自己所属行业的细分门户网站。而这里我们要讲的就是跟门户模块有关的一个细节——Discuz门户文章8个标签属性的数据库存储分析。
用过Discuz门户功能的人也知道,一篇文章有8个标签,分别是原创、热点、组图 、爆料、 头条、幻灯、滚动、推荐。那么当我们进行二次开发的时候,如果要利用这8个标签进行筛选文章的时候该怎么办呢?这个时候我们就得先知道这几个标签是怎么被保存的。比如下面是我们项目中的实例:可以通过标签来筛选数据,同时在每个文章下面显示他所属的标签。
通过查阅数据词典我们发现,这8个标签在文章表中只有一个字段,也就是一个字段保存了标签的信息,首先我们来看下DZ的数据词典的这部分:
那么这一个字段是怎么保存这8个标签信息的,我们可以联想到论坛帖子的高亮属性,也同样的是多个属性保存在一个字段中,那是通过进制转化来的;那么这又是什么呢?我们查看这些字段的值,也许能发现一些规律,但要我们得出一个规则那实在太难了。于是我们还是去分析源代码来倒推吧。
我们可以在source/function/function_portalcp.php中找到两个方法 article_parse_tags和article_make_tag:
function article_parse_tags($tag) { $tag = intval($tag); $article_tags = array(); for($i=1; $i<=8; $i++) { $k = pow(2, $i-1); $article_tags[$i] = ($tag & $k) ? 1 : 0; } return $article_tags; } function article_make_tag($tags) { $tags = (array)$tags; $tag = 0; for($i=1; $i<=8; $i++) { if(!empty($tags[$i])) { $tag += pow(2, $i-1); } } return $tag; }
其实这两个可以理解为一组方法,前者是把数据库读取到值传送进去,依次判断,然后得到返回一个数组,而这个数组就有八个键值,每个键值即代表一个标签属性。而下面一个方法就是把编写文章的时候表单的值传送给这个方法,然后依次执行,进行2的n次方累加,得到一个十进制的值然后返回$tag,这个$tag的值最终就是保存在数据库中的。
其实以上实际上可以完成我上面提到那个例子的一部分,也就是通过取得的值然后把对应的标签显示在文章下面,当然上面的第二个方法实际上为了解释整个原理附带来的。那么还有一部分,也就是怎么通过标签来筛选呢?
这个时候我们可以参考Discuz的源代码中source/class/block/portal/block_article.php中的block_article类,这个大概就是我们进行门户文章DIY调用的一个类,我们可以知道在DIY调用中我们是可以通过标签来进行数据过滤的,那么里面就一定有现成的方法来做到这点了。在里面有这么一段代码:
if(is_array($tag)) { $article_tags = array(); foreach($tag as $k) { $article_tags[$k] = 1; } include_once libfile('function/portalcp'); $v=article_make_tag($article_tags); if($v > 0) { $wheres[] = "(at.tag & $v) = $v"; } }
上面的参数$tag就是我们要筛选的目标标签数组,比如里面同样是八个键,每个键值可能为0或1(也就是勾选了或者没有勾选),然后一次判断,得到SQL查询语句中where条件约束的数组$where,这样就实现了对数据的筛选或者说过滤。以上就是我们查看Discuz源代码获取的一些信息,也就是我们知道了其中的原理,其实大概就是用2的n次方,然后&位运算来进行判断(PHP文档)。
============================华丽的分割线============================
好了,上面进摆出了几段代码实际上已经阐述了原理,那么我们以一个实际的例子来说明。比如说一篇文章已经选中了原创和推荐两个标签,也就是第一个和最后一个:
也就是这个对应的article_make_tag方法中
$tags=array(0=>1,1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,7=>1);
通过该方法返回的值$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
$article_tags=array(0=>1,1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,7=>1);
反过来就是我们最开始提到例子中需要做的,也就是说已经知道了某个筛选条件,比如说是要原创的,那么对应的数组会是
$tag=array(0=>1,1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,7=>0);
同样的我们可以得到相应的约束条件。
我们在微信上24小时期待你的声音
解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流