WordPress函数:query_posts(查询文章)

一旦有人认为你混得不错,你就必须得混好。很多情况下,人真的不是为自己而活。

描述

query_posts() 可以用来控制在循环(Loop)中显示哪些文章。它可以替换主查询,而使用一个新的查询来获取文章。要清除 query_posts() 的新查询,可以使用 wp_reset_query() 来重置查询。

我们强烈建议您使用 pre_get_posts 来替代 query_posts() ,同时使用 is_main_query 来检查主查询。

举个例子,在首页(homepage),你可以看到最新的10篇文章。如果你想仅仅显示 5 篇文章(不在乎分页),你可以像这样使用 query_posts() :

1
query_posts( 'posts_per_page=5' );

query_posts( 'posts_per_page=5' );

下面是在 functions.php 中使用  pre_get_posts 实现类型的结果:

1
2
3
4
5
6
function five_posts_on_homepage( $query ) {
    if ( $query->is_home() && $query->is_main_query() ) {
        $query->set( 'posts_per_page', '5' );
    }
}
add_action( 'pre_get_posts', 'five_posts_on_homepage' );

function five_posts_on_homepage( $query ) { if ( $query->is_home() && $query->is_main_query() ) { $query->set( 'posts_per_page', '5' ); } } add_action( 'pre_get_posts', 'five_posts_on_homepage' );

注:pre_get_posts 动作无法正常工作于 页面(Page) 查询。

注意事项

query_posts() 只是一种查询数据库获取文章列表的方法。在使用 query_posts() 之前,你需要了解它的弊端。

改变主循环(Alters Main Loop)

query_posts() 是通过替换查询条件来改变主循环生成的内容的。一旦你使用 query_posts(), 你文章的相关的全局变量和模板标签都将改变。条件标签(Conditional tags)在你调用  query_posts() 之后也会发生改变——这可能是也可能不是预期的结果。

二级循环(Secondary Loops)

要创建 二级列表(例如:在页面底部显示相关文章列表,或者在边栏小工具中显示一个列表), 尝试使用 WP_Query 或 get_posts() 来获取文章。

如果你一定要用 query_posts() ,请确保你在上一个使用了 query_posts() 的循环结束的地方,使用 wp_reset_query() 来重置查询。

分页(Pagination)

分页将无法正常工作,除非你适当设置 'paged' 查询: adding the paged parameter

额外的SQL查询

如果你在模板页面中使用 query_posts ,WordPress 就会在 query_posts 时执行数据库查询和检索记录(即它是如何知道是服务于哪个模板页面!)所以,当你使用 query_posts() 覆盖了默认的查询,你基本上就扔掉了默认的查询以及它的结果,并重新对数据库执行了另一个查询。

这也许不是一个问题,尤其是如果你正在处理一个较小的基于博客的网站。但是对于大型网站、大型数据库以及较高的访客流量来说,你就要考虑使用其他查询方式来减少 SQL查询了,比如直接修改默认的查询(在它被请求前)。使用请求(request)过滤器可是实现这点。

'parse_query' 和 'pre_get_posts' 过滤器也可用来修改内部用于生成SQL查询数据库的 $query 对象。

用法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
<?php
// 查询
query_posts( $args );

// 循环
while ( have_posts() ) : the_post();
    echo '<li>';
    the_title();
    echo '</li>';
endwhile;

// 重置查询
wp_reset_query();
?>

<?php // 查询 query_posts( $args ); // 循环 while ( have_posts() ) : the_post(); echo '<li>'; the_title(); echo '</li>'; endwhile; // 重置查询 wp_reset_query(); ?>

在模板文件的循环前面添加 query_posts() ,wp_query 对象将根据你给定的参数生成一个新的 SQL 查询。当你这样做时,WordPress 将忽视通过URL检索到的其他参数(例如 页码或分类)。

保留现有的查询参数

如果你想保留原来的查询参数作为当前查询的一部分,然后添加或重写一些参数,你可以在 query_posts() 中使用全局变量 $query_string 。

例如,要修改文章的显示顺序并且不影响接下来的查询,你可以在循环(The Loop)前面使用下面的代码:

1
2
global $query_string;
query_posts( $query_string . '&order=ASC' );

global $query_string; query_posts( $query_string . '&order=ASC' );

当你通过这种方式使用 query_posts() , 参数的前面必须添加一个 & 。或者你可以合并原来的查询数组到到你的参数数组:

1
2
3
global $wp_query;
$args = array_merge( $wp_query->query_vars, array( 'post_type' => 'product' ) );
query_posts( $args );

global $wp_query; $args = array_merge( $wp_query->query_vars, array( 'post_type' => 'product' ) ); query_posts( $args );

结合参数

从上面的例子中,你可能已经注意到,你可以使用 & 来结合参数:

1
query_posts( 'cat=3&year=2004' );

query_posts( 'cat=3&year=2004' );

在首页查询当前月份的分类ID为 13 的文章:

1
2
3
if ( is_home() ) {
	query_posts( $query_string . '&cat=13&monthnum=' . date( 'n', current_time( 'timestamp' ) ) );
}

if ( is_home() ) { query_posts( $query_string . '&cat=13&monthnum=' . date( 'n', current_time( 'timestamp' ) ) ); }

下面的例子,将返回一组文章(分类ID为 1 和 3 、显示 2 篇文章、根据标题降序排列):

1
query_posts( array( 'category__and' => array(1,3), 'posts_per_page' => 2, 'orderby' => 'title', 'order' => 'DESC' ) );

query_posts( array( 'category__and' => array(1,3), 'posts_per_page' => 2, 'orderby' => 'title', 'order' => 'DESC' ) );

下面的例子将返回分类ID为 1 和 标签为 “apples”的文章:

1
query_posts( 'cat=1&tag=apples' );

query_posts( 'cat=1&tag=apples' );

你可以使用加号 + 来添加更多标签:

1
query_posts( 'cat=1&tag=apples+apples' );

query_posts( 'cat=1&tag=apples+apples' );

参数

参数详情请阅读 Parameter section of the WP_Query class article.

下面的例子同样可以用于 WP_Query 对象。

例子

在首页排除分类

将下面的代码添加到 index.php 文件,这样首页将显示 除了分类ID为 3 以外的其他所有文章:

1
2
3
4
5
<?php
if ( is_home() ) {
	query_posts( 'cat=-3' );
}
?>

<?php if ( is_home() ) { query_posts( 'cat=-3' ); } ?>

你还可以以此类推添加更多要排除的分类:

1
2
3
4
5
<?php
if ( is_home() ) {
	query_posts( 'cat=-1,-2,-3' );
}
?>

<?php if ( is_home() ) { query_posts( 'cat=-1,-2,-3' ); } ?>

检索某一篇文章

要检索某一篇文章,可以使用下面的代码:

1
query_posts( 'p=5' );

query_posts( 'p=5' );

注:如果检索的是一个附件,那你应该使用 attachment_id 代替 p :

1
query_posts( 'attachment_id=5' );

query_posts( 'attachment_id=5' );

如果你想使用“阅读更多(Read More)”功能,你就需要在查询中设置全局变量 $more 的值为 0 :

1
2
3
4
5
6
7
8
9
10
11
12
13
<?php
// 检索 ID 为 5 的文章
query_posts( 'p=5' );

// 设置 $more 为 0 以便仅仅获取文章的第一部分
global $more;
$more = 0;

// 循环
while (have_posts()) : the_post();
	the_content( '阅读全文 »' );
endwhile;
?>

<?php // 检索 ID 为 5 的文章 query_posts( 'p=5' ); // 设置 $more 为 0 以便仅仅获取文章的第一部分 global $more; $more = 0; // 循环 while (have_posts()) : the_post(); the_content( '阅读全文 »' ); endwhile; ?>

一个分类的所有文章

在 设置 > 阅读 中,“博客页面最多显示”的设置可以影响你的结果。为了克服这个问题,你可以添加 'posts_per_page' 参数,例如:

1
query_posts( array ( 'category_name' => 'my-category-slug', 'posts_per_page' => -1 ) );

query_posts( array ( 'category_name' => 'my-category-slug', 'posts_per_page' => -1 ) );

这样将显示这个分类的所有文章。

但是,对于子分类(subcategories or child categories), 'The Category Name' 可能就没办法生效,请使用 'category-slug' 替代。阅读 Function_Reference/is_category。

1
2
3
4
5
if (is_category('category-slug')): 

	 query_posts(array('category_name' => 'my-category-slug', 'posts_per_page' => -1 )); 

endif;

if (is_category('category-slug')): query_posts(array('category_name' => 'my-category-slug', 'posts_per_page' => -1 )); endif;

聚合 Feeds

在 设置 > 阅读 中,“Feed 中显示最近”的设置 或者  'posts_per_rss' 参数将在 Feed 中覆盖任何查询的 'posts_per_page' 参数。

为了克服这个问题(例如在自定义的 ics Feed 中,输出所有符合条件的文章),可以使用 "posts_limit" 过滤器:

1
2
3
4
if ( isset ( $query->query_vars['feed'] ) and ( $query->query_vars['feed'] == 'ics' ) )
{
	add_filter( 'post_limits', '__return_empty' );
}

if ( isset ( $query->query_vars['feed'] ) and ( $query->query_vars['feed'] == 'ics' ) ) { add_filter( 'post_limits', '__return_empty' ); }

传递变量到 query_posts

有两种方法可以传递一个变量到查询中,你可以根据自己的需求选择。就像其他例子一样,将下面的代码添加到循环的上面:

例 1

在这个例子中,我们在查询运行前串连查询条件。首先分配变量,然后串连,然后运行它。在这里,我们从其他地方拉来一个分类变量:

1
2
3
4
5
6
7
8
// 分配参数作为当前分类
$categoryvariable = $cat;

// 串连查询
$args = 'cat=' . $categoryvariable . '&orderby=date&order=ASC';

// 运行查询
query_posts( $args );

// 分配参数作为当前分类 $categoryvariable = $cat; // 串连查询 $args = 'cat=' . $categoryvariable . '&orderby=date&order=ASC'; // 运行查询 query_posts( $args );

例 2

在这个例子中,双引号是告诉 PHP 这是一个闭合。对于这个例子,我们获取当前 月份 和 年份,然后告诉 query_posts() 显示当前 年/月 的文章。此外,我们使用升序排列(ASE) ,这样,最老的文章将显示在页面的顶部:

1
2
3
4
$current_year = date('Y');
$current_month = date('m');

query_posts( "cat=22&year=$current_year&monthnum=$current_month&order=ASC" );

$current_year = date('Y'); $current_month = date('m'); query_posts( "cat=22&year=$current_year&monthnum=$current_month&order=ASC" );

例 3

这个例子解析如何删除一个完整的文章列表,并且进行分页处理。我们可以使用默认的 $query_string 告诉 query_posts() 显示所有文章列表。我们还可以自定义 posts_per_page 这个查询参数为 –1 到 任何数量 来设置每页显示的文章数。最后,你可能想使用  posts_nav_link() 来导航生成的文章存档。

1
query_posts( $query_string . '&posts_per_page=-1' );

query_posts( $query_string . '&posts_per_page=-1' );

例 4

如果你不需要使用 $query_string 变量,另一种方法是将所有的参数添加到一个数组中,即使在更加复杂的情况下,也清晰明了。上面 例 2 可以使用下面的代码实现:

1
2
3
4
5
6
7
$args = array(
	'cat'      => 22,
	'year'     => $current_year,
	'monthnum' => $current_month,
	'order'    => 'ASC'
);
query_posts( $args );

$args = array( 'cat' => 22, 'year' => $current_year, 'monthnum' => $current_month, 'order' => 'ASC' ); query_posts( $args );

正如你所见,通过这种方式,所有的变量都放置在独立的行中,这就非常容易阅读了。

例 5

它甚至可以使用数组方式(例 4)来查询多个分类。只需提供分类别名(taxonomy slug),多个别名使用逗号隔开。在下面的例子中,我们会得到 Bruce Campbell 或 Chuck Norris 的movie文章:

1
2
3
4
5
6
$args = array(
	'post_type'=> 'movie',
	'actor'    => 'Bruce Campbell, Chuck Norris',
	'order'    => 'ASC'
);
query_posts( $args );

$args = array( 'post_type'=> 'movie', 'actor' => 'Bruce Campbell, Chuck Norris', 'order' => 'ASC' ); query_posts( $args );

所在文件

query_posts() 包含在  wp-includes/query.php.

其他资源

为了更深入的讨论 WordPress 生成和处理查询,请阅读这些文章:

  • Query Overview 和 Custom Queries
  • 正确使用'pre_get_posts的自定义默认查询 - Bill Erickson - Customize the Wordpress Query 或 John James Jacoby - Querying Posts Without query_posts
  • 你不知道的查询 - Slides from WordCamp Netherlands 2012 by Andrew Nacin
  • WordPress函数 之 query_posts用法汇总

相关函数

类: WP_Query - 详细的 WP_Query 类概述

类: $wpdb - 使用 $wpdb 对象的概述

函数: get_query_var()

函数: get_posts() - 一个专门返回文章数组的函数

函数: get_pages() - 一个专门返回页面数组的函数

函数: have posts() - 一个确定查询是否返回文章的条件

函数: the_post() - 用于自动设置查询后的循环

函数: rewind_posts() - 重置当前循环

函数: setup_postdata() - 为循环中的私有查询设置查询数据

函数: wp_reset_postdata() - 恢复以前的查询(通常在一环内循环)

函数: wp_reset_query()

函数: is_main_query() - 确保被改变的仅仅是主查询

动作钩子: pre_get_posts - 在执行之前修改 WordPress 的查询

过滤钩子: found_posts - 修改 WP_Query 对象的 found_posts 的值

好代码教程: Displaying Posts Using a Custom Select Query

好代码教程: Making Advanced Taxonomy Queries

好代码教程: Making Custom Queries using Offset and Pagination

  • 原文:http://codex.wordpress.org/Function_Reference/query_posts
  • 翻译:倡萌@WordPress大学 - WordPress函数:query_posts(查询文章)

本文WordPress函数:query_posts(查询文章)到此结束。我不怕千万人阻挡,只怕自己投降。逆风的方向,更适合飞翔。小编再次感谢大家对我们的支持!

您可能有感兴趣的文章
WordPress站点Gravatar头像前后台不显示的如何解决办法

WordPress主题需要支持https吗?WordPress站点如何如何实现https?

WordPress站点的页面/标签/分类URL地址如何添加.html?

WordPress站点更换了域名后数据库应该如何操作替换新旧域名?

WordPress安装在主机空间的什么目录里面?根目录在哪里?