兔源码网 - 专注DZ模版分享、DZ插件分享!

 找回密码
 立即注册

QQ登录

只需一步,快速开始

广告
温馨提示:本站所有DZ相关模板、插件、教程(包括DZ应用中心已下架插件、模板)所有用户可全部免费下载,无任何收费模式(不包括VIP版块非DZ资源,VIP版块需购买会员组方可下载),如本站没有您需要的插件或模板,请自行前往淘宝互站网送吗网等第三方站点搜索相关插件或模板进行低价格购买下载本站非常不建议用户浪费金钱前往DZ应用中心购买。

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

[复制链接]
tuyuanma 发表于 2017-4-6 09:59:20 | 显示全部楼层 |阅读模式
温馨提醒:禁止内容附带广告图片、二维码图片,否则将在不通知情况下永久禁言账号!
欢迎加入兔源码站长交流群,您的加入是我们前行的动力!

DZ站长建站交流群:493651246

[ 温馨提示:点击上方群文字,可快速加入QQ群 ]


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

& P" y4 _- v/ N- c0 q

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


6 t5 d% a4 Z/ Y% L0 T  `- [

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


) ?, V4 I6 ^0 q! l: ]- Q# d) W3 i" |" C  C

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

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


; d0 H5 k  s. s( i  f3 A0 [

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

1 f( ]# n! L/ c- r0 c- x

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

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


6 J& Q/ i+ B1 g

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

* c3 q  h' ?' _: J

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

6 _, M( K( j; b  D3 F! n5 s

  1. <font color="rgb(85, 85, 85)">function article_parse_tags($tag) {3 V* y( T. G% k& e& U6 d
  2.         $tag = intval($tag);
    / t6 g% k& S% t' T' x0 Z( f
  3.         $article_tags = array();
    6 X4 C2 [3 V3 a- Q
  4.         for($i=1; $i<=8; $i++) {  S6 j7 w# c7 s2 K" V+ j+ D4 d
  5.                 $k = pow(2, $i-1);
    4 q  V; c3 b4 ~& e0 M$ Y
  6.                 $article_tags[$i] = ($tag & $k) ? 1 : 0;7 G! @0 w- f+ s! V7 G+ ]% d  b, V
  7.         }
    6 H" k8 Q! {& L% P% I5 x; S
  8.         return $article_tags;
    ! ~, V8 V; k, n5 M7 v' r  b, I8 ^% m1 h
  9. }4 Q5 w' m) n; i: i. @6 w) t) n
  10. 0 Y0 k! ^# B5 {* S6 F" w/ U
  11. function article_make_tag($tags) {2 [; f& K3 ?' o5 V1 S' R+ m" R# j$ w2 u
  12.         $tags = (array)$tags;9 `2 ?- l( P& T7 B. t$ i( r" s
  13.         $tag = 0;% K# T3 `& ]$ u# e# e5 x
  14.         for($i=1; $i<=8; $i++) {2 T9 f3 Q4 R. G5 Y; R
  15.                 if(!empty($tags[$i])) {4 u: s3 f/ u9 B; `
  16.                         $tag += pow(2, $i-1);
    8 L# Q  ~. g# V. v# D
  17.                 }
    # u0 a" w9 W; ]5 J# g: I
  18.         }
    7 y) H6 o- d# d" E$ W. W
  19.         return $tag;
    % Z* \6 C, H  `+ ^
  20. }</font>
复制代码

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


9 \; v/ }& s6 y5 L: G' k

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

+ X; a; R' e% A- C  b+ J

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

1 K' C. B1 ]( K6 Y, S; v

  1. <font color="rgb(85, 85, 85)">if(is_array($tag)) {
      o4 C* K9 @/ ~4 K. S. ~
  2.         $article_tags = array();
      g" o0 q" j& [& K0 a
  3.         foreach($tag as $k) {
    0 _1 V* a# s7 K) E! R0 W
  4.                 $article_tags[$k] = 1;- l4 `5 s# F. C) R3 }1 z; S
  5.         }* h" [: p$ o- |4 o5 U
  6.         include_once libfile('function/portalcp');& `" D3 j  j) p6 F8 v  J
  7.         $v=article_make_tag($article_tags);
    7 }/ V- ^; O& O
  8.         if($v > 0) {
    / o+ w" Y0 {0 a) D
  9.                 $wheres[] = "(at.tag & $v) = $v";! [% S$ t% ], a! ?. s- c* P4 }. e, h. L3 ?
  10.         }
    8 b0 E6 f$ V5 X4 w5 X7 I! U
  11. }</font>
复制代码

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


4 V% V) M  h3 U# R+ L# I

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


2 E4 W- v* I0 D0 K4 A0 z0 K

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


% D* s" \2 D1 Q" m+ G& T5 Q1 i

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

Discuz门户文章8个标签属性的数据库存储分析
  [% w, M, T/ C& h# x* {

7 r! g; p) m3 I+ x3 g

也就是这个对应的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。

) U2 ]9 Q+ {9 o2 R- S$ I

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


8 b* \5 J- {- i5 T! N* S- W

  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>
复制代码

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


) `" B4 j$ P% B! G- D

  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>
复制代码
同样的我们可以得到相应的约束条件。4 U+ ^1 ?2 G. m6 b

* k$ C! d( j( x' Q; D  Z% m7 D" ]5 ~  y) J5 m. d. A1 S, H1 Z" D

版权声明
1、转摘或引用本站内容资源须注明原网址,并标明本站网址(兔源码www.tuyuanma.com);
2、对于转摘或引用本站内容资源而引起的民事纠纷、行政处罚或其他损失,本站不承担责任;
3、对于不遵守本声明或其他违法、恶意使用本站内容者,本站保留其追究法律责任的权利。




自动排版 | 高级模式
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

广告

兔源码网 ( 赣ICP备17001544号 公网安备36112102000036号 )

兔源码 QQ交流群:493651246  GMT+8, 2019-11-15 01:43
←站长统计工具 查看密码:tuyuanma

兔源码DZ模板下载 站长邮箱:tuyuanma@qq.com 站长QQ:3197813386 网站地图

© 2001-2013 本站内容皆来自互联网及用户分享,如需删除请提供软著或商标证书。

     
快速回复 返回顶部 返回列表