这几天有在重组、改建我的几个小站,又涉及到了 WordPress 的相关技术。本文说说关于“相关文章”的代码,网上有不少插件实现这个功能,不过个人还是喜欢用纯代码来实现。

<?php
//相关文章,这段是放 functions.php 里的。
function get_relate_posts($num_limit)
{
$theSeedID=get_the_ID();
$tags = wp_get_post_tags($theSeedID);
$tagIDs = array();
$return_str='相关文章:';
if ($tags)
{
$tagcount=count($tags);
for ($i=0;$i<$tagcount;$i++)
{
$tagIDs[$i]=$tags[$i]->term_id;
}
$args=array('tag__in'=>$tagIDs,'post__not_in'=>array($theSeedID),'showposts'=>$num_limit,'caller_get_posts'=>1);
$my_query=new WP_Query($args);

if ($my_query->have_posts())
{
$return_str.='<ul>';
while ($my_query->have_posts())
{
$my_query->the_post();
$return_str.='<li><a href="'.get_permalink().'" rel="bookmark" title="'.get_the_title().'">'.get_the_title().'</a></li>';
}
$return_str.='</ul>';
wp_reset_query();
}
else
{
$return_str.='<ul>';
$return_str.=get_random_posts($num_limit,get_the_ID()).'</ul>';
}
}
else //<!-- 没有相关文章显示随机文章 -->
{
$return_str.='好像还木有咛...';
}
return $return_str.'';
}
?>

//下面这段直接放 single.php 文件里,试用过,不好,因为它是随机选取一个 tag 来匹配的。

<ul id="tags_related">
<?php
global $post;
$post_tags = wp_get_post_tags($post->ID);
if ($post_tags) {
foreach ($post_tags as $tag) {
// 获取标签列表
$tag_list[] .= $tag->term_id;
}
// 随机获取标签列表中的一个标签
$post_tag = $tag_list[ mt_rand(0, count($tag_list) - 1) ];
// 该方法使用 query_posts() 函数来调用相关文章,以下是参数列表
$args = array(
'tag__in' => array($post_tag),
'category__not_in' => array(NULL), // 不包括的分类ID
'post__not_in' => array($post->ID),
'showposts' => 6, // 显示相关文章数量
'caller_get_posts' => 1
);
query_posts($args);
if (have_posts()) {
while (have_posts()) {
the_post(); update_post_caches($posts); ?>
<li><a href="<?php the_permalink(); ?>" rel="bookmark" title="<?php the_title_attribute(); ?>"><?php the_title(); ?></a></li>
<?php
}
}
else {
echo '<li>好像还木有咛...</li>';
}
wp_reset_query();
}
else {
echo '<li>好像还木有咛...</li>';
}
?>
</ul>

最后请出一段最优的代码,如下:

<ul id="tags_related">
<?php
global $post, $wpdb;
$post_tags = wp_get_post_tags($post->ID);
if ($post_tags) {
$tag_list = '';
foreach ($post_tags as $tag) {
// 获取标签列表
$tag_list .= $tag->term_id.',';
}
$tag_list = substr($tag_list, 0, strlen($tag_list)-1);

$related_posts = $wpdb->get_results("
SELECT DISTINCT ID, post_title
FROM {$wpdb->prefix}posts, {$wpdb->prefix}term_relationships, {$wpdb->prefix}term_taxonomy
WHERE {$wpdb->prefix}term_taxonomy.term_taxonomy_id = {$wpdb->prefix}term_relationships.term_taxonomy_id
AND ID = object_id
AND taxonomy = 'post_tag'
AND post_status = 'publish'
AND post_type = 'post'
AND term_id IN (" . $tag_list . ")
AND ID != '" . $post->ID . "'
ORDER BY RAND()
LIMIT 6");
// 以上代码中的 6 为限制只获取6篇相关文章
// 通过修改数字 6,可修改你想要的文章数量

if ( $related_posts ) {
foreach ($related_posts as $related_post) {
?>
<li><a href="<?php echo get_permalink($related_post->ID); ?>" rel="bookmark" title="<?php echo $related_post->post_title; ?>"><?php echo $related_post->post_title; ?></a></li>
<?php }
}
else {
echo '<li>好像还木有咛……</li>';
}
}
else {
echo '<li>好像还木有咛……</li>';
}
?>
</ul>

  为什么说这段是最优的呢?虽然获取相关文章的原理与方法一相似,但它在获取文章的时候是以SQL语句来直接读取数据库,从而随机获取6篇相关文章记录,而不是WordPress的函数query_posts()。而且有网友进行了测试,把它和第二段做了比较,结果前者执行时间为:0.18067908287048 秒,后者为:0.037126064300537 秒,显然更快!