- 在线时间
- 4 小时
- 最后登录
- 2015-5-5
- 注册时间
- 2015-4-8
- 听众数
- 10
- 收听数
- 0
- 能力
- 0 分
- 体力
- 104 点
- 威望
- 0 点
- 阅读权限
- 20
- 积分
- 47
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 30
- 主题
- 12
- 精华
- 0
- 分享
- 0
- 好友
- 5
升级   44.21% TA的每日心情 | 慵懒 2015-5-5 09:30 |
|---|
签到天数: 12 天 [LV.3]偶尔看看II
- 自我介绍
- 学生
 |
过去我做的课题是蛋白质折叠,其中用到一个算法,前不久在做一个论文库的tag系统的时候意识到这两者其实是同一个问题,于是就把原来的优化算法改进了一下,不过想了一下,估计也没有什么系统会用吧,但是基于不想浪费的原则,就把这个优化写一下。
' z7 c* S4 ~6 w) `7 v1 M- }9 {- }; Q. B6 r1 n% r0 X' j
这个算法是用来计算两个主题在tag上面的相似度。假如一个主题的tag的数目是有限的(用在论文库上,还有一个限制就是主题词数目是一样多的),那么我们可以找出所有和它有至少一个tag相同的主题,然后在里面找出tag相同数目最多的那个主题作为最相似主题。( B! F( v) @. m& E) ]/ P3 j
, E* @- \1 [! [0 P1 B; C- r7 h
最简单的想法是做一个二叉树,第一级是有没有tag a,第二级是有没有tag b,但是这个和二叉树不同,二叉树的不同级优先级是不一样的,而这个问题里面优先级是一样的。, ~6 B/ C0 U! {8 m4 ?
然后的想法是仍然做一个二叉树,不过使用贪恋法找一个尝试解,然后其实根本不必算到最后就可以砍掉很多分支。因为假如最优解是5个相同,你现在得分只有1,然后下面只剩3层,可想最优情况也只能得到4,所以整个分支都可以砍掉。当找到更优解的时候就替换掉暂时的尝试最优解。0 C Q; Q, u( [3 R/ H, [3 e+ r/ y
. F. R7 M" n" w, ?7 J
这在速度上当然是最优解法,不过有没有空间上花费较少,而效率上下降不多的解法呢?- O2 d4 ~: E4 x, N( w u
我想了一下,使用这种方法:1 O7 y2 C+ j5 b! h* c
把每种tag按照有无,把一个主题的tag编码成一个二进制串,第n位为1则表示拥有第n个tag。里面的1的数目是固定的。然后把两个主题的二进制表示进行按位and操作,结果字节用查表法就可以查出里面1的数目,最后相加,可以节省一点if操作。
! E2 Z# t7 ?) r+ G- Z$ D/ x# q
! C! y5 S+ ]% G然后,我们把二进制表示为这样一种排序,最低位是0110,0110,0110重复,第二位是0011,1100,0011,1100重复,第三位是0000,1111,1111,0000,重复,简单说就是呈现一种回文的结构。这种排序表示为数字,就是0,1,3,2,6,7,5,4的顺序……! X; B% h3 M0 \( d
000, U ?, d: f- _/ a9 R) y3 A
0010 P& P E7 h# r0 u1 w7 j
011+ x+ B" x$ b" I: g8 z: w
010. W* g; f3 k+ p- G. h1 p
110
- K, O7 H+ [0 K* i2 y111) E. p: Z: H0 d: h% h4 X+ h0 t( r! x
101
/ X% @, }, K8 Y' x100
" V/ W/ b2 c. w& o) }- E) ]# O& O k" Y6 o1 C) h1 L! V6 s! ]
可以看到,任意相邻数字的二进制表示永远只相差一位,而且相邻的高位几乎总是相同的(用数学归纳法可以简单证明)。这就保证了,如果我找到一个优质解,那么很大概率在它的附近找到另外一个优质解(但只有一定概率,不是肯定保证)。我们把tag的二进制数表示按照这种方法排序后,我们就知道了所有相邻位置相同到第几位。一旦知道一个条目的二进制表示到倒数第四个字节得分1,那么肯定不会超过最高得分6,然后接下来所有和前一个位置的前四位相同的一系列条目都可以跳过。 由于每个条目和前一个条目相同到第几位是可以预先算好的,所以这个算法几乎不用存储中间结果。& [. O0 L: H* ]4 H J( w( C# y. o
% |2 L9 { h' u/ U0 f8 L3 n9 L
其实不按回文排列排序也可以,但是在密集区域这样可能更好,特别是可以用普通的对半分解排序函数更有效地找到试探解,这使得新数据的插入,删除都可以使用传统技术而不必把效率下降太多。' ~- M& M% K! E
7 B- m8 ^/ t- ]! o& r# U
这种表虽然效率不是最高,不过考虑到tag往往是幂数分形分布的(也就是最大类有1000个,次大类100个,次次大类10个),这种算法和二叉树只有常数倍的差距,即理论上是平均情况为O(log_2 N) 和O(N^[1/m],实际都退化到O(N^[1/m]。程序的复杂度更小,空间占用为O(1),而数据的插入,排序都可以用传统的线性排序技术,属于一种值得推荐的算法。
& w6 P4 z9 a6 n" G/ e
; V/ U% b: d# k" r5 @, |
2 }; g; X/ Q' f5 H( Q f可惜的是,由于限定了条件是tag数有限,所以应用范围不广。这种算法不知道有没有人发明过(我觉得应该有,但是不会是这个领域),在基于关键词的倒排表里面可能有一些用处。
/ Z3 d; J/ n4 g6 f- O$ q$ |
- E" n3 A9 M; h; f |
zan
|