🧠 了解 REST API 的 populate
参数
¥🧠 Understanding the populate
parameter for the REST API
此页面的内容可能尚未与 Strapi 5 完全同步。
¥The content of this page might not be fully up-to-date with Strapi 5 yet.
此页面的内容可能尚未与 Strapi 5 完全同步:
¥The content of this page might not be fully up-to-date with Strapi 5 yet:
-
所有概念信息和解释都是正确且最新的。
¥All the conceptual information and explanations are correct and up-to-date.
-
但是,在示例中,响应内容可能略有不同。
¥However, in the examples, the response content might be slightly different.
Strapi 5.0.0(稳定版)发布后,示例将完全更新,并且 FoodAdvisor 示例应用升级到 Strapi 5 后即可更新。
¥Examples will be fully up-to-date after the Strapi 5.0.0 (stable version) release and as soon as the FoodAdvisor example application is upgraded to Strapi 5.
但是,响应示例略有不同不应妨碍你掌握本页所教授的基本概念。
¥However, having slightly different response examples should not prevent you from grasping the essential concepts taught in this page.
当使用 Strapi 的 REST API 查询内容类型时,默认情况下,响应仅包含顶层字段,不包含任何关系、媒体字段、组件或动态区域。
¥When querying content-types with Strapi's REST API, by default, responses only include top-level fields and do not include any relations, media fields, components, or dynamic zones.
在 Strapi REST API 上下文中填充意味着通过返回比默认返回的字段更多的字段,在响应中包含其他内容。你可以使用 populate
参数 来实现此目的。
¥Populating in the context of the Strapi REST API means including additional content with your response by returning more fields than the ones returned by default. You use the populate
parameter to achieve this.
在本指南中,示例都是使用从 FoodAdvisor 示 例应用附带的服务器查询的真实数据构建的。要自行测试示例,请设置 FoodAdvisor,在 /api/
文件夹中启动服务器,并确保在发送查询之前为查询的内容类型授予适当的 find
权限。
¥Throughout this guide, examples are built with real data queried from the server included with the FoodAdvisor example application. To test examples by yourself, setup FoodAdvisor, start the server in the /api/
folder, and ensure that proper find
permissions are given for the queried content-types before sending your queries.
本指南将详细解释以下用例:
¥The present guide will cover detailed explanations for the following use cases:
-
填充 所有字段和关系,深 1 层,
¥populate all fields and relations, 1 level deep,
-
填充 某些字段和关系,深 1 层,
¥populate some fields and relations, 1 level deep,
-
填充 一些字段和关系,深层次,
-
填充 components,
¥populate components,
-
填充 动态区域。
¥populate dynamic zones.
填充几个深度级别通常称为 "深度填充"。
¥Populating several levels deep is often called "deep populate".
除了在查询中使用 populate
参数的各种方式之外,你还可以构建自定义控制器作为填充创建者字段(例如 createdBy
和 updatedBy
)的解决方法。专门的 如何填充创建者字段 指南对此进行了解释。
¥In addition to the various ways of using the populate
parameter in your queries, you can also build a custom controller as a workaround to populate creator fields (e.g., createdBy
and updatedBy
). This is explained in the dedicated How to populate creator fields guide.
填充所有关系和字段,深 1 层
¥Populate all relations and fields, 1 level deep
你可以通过单个查询返回所有关系、媒体字段、组件和动态区域。对于关系来说,这只适用于 1 层深度,以防止性能问题和较长的响应时间。
¥You can return all relations, media fields, components and dynamic zones with a single query. For relations, this will only work 1 level deep, to prevent performance issues and long response times.
要填充 1 级深度的所有内容,请将 populate=*
参数添加到你的查询中。
¥To populate everything 1 level deep, add the populate=*
parameter to your query.
下图比较了 FoodAdvisor 示例应用在填充和不填充 1 级深度的所有内容时返回的数据:
¥The following diagram compares data returned by the FoodAdvisor example application with and without populating everything 1 level deep:
让我们比较并解释使用和不使用此查询参数时会发生什么:
¥Let's compare and explain what happens with and without this query parameter:
示例:没有 populate
¥Example: Without populate
如果没有 populate 参数,GET
对 /api/articles
的请求仅返回默认属性,不会返回任何媒体字段、关系、组件或动态区域。
¥Without the populate parameter, a GET
request to /api/articles
only returns the default attributes and does not return any media fields, relations, components or dynamic zones.
以下示例是 articles
内容类型中所有 4 个条目的完整响应。
¥The following example is the full response for all 4 entries from the articles
content-types.
请注意响应如何仅包含 title
、slug
、createdAt
、updatedAt
、publishedAt
和 locale
字段,以及由 CKEditor 插件处理的文章的字段内容(ckeditor_content
,为了简洁而被截断):
¥Notice how the response only includes the title
, slug
, createdAt
, updatedAt
, publishedAt
, and locale
fields, and the field content of the article as handled by the CKEditor plugin (ckeditor_content
, truncated for brevity):
GET /api/articles
{
"data": [
{
"id": 1,
"documentId": "t3q2i3v1z2j7o8p6d0o4xxg",
"title": "Here's why you have to try basque cuisine, according to a basque chef",
"slug": "here-s-why-you-have-to-try-basque-cuisine-according-to-a-basque-chef",
"createdAt": "2021-11-09T13:33:19.948Z",
"updatedAt": "2023-06-02T10:57:19.584Z",
"publishedAt": "2022-09-22T09:30:00.208Z",
"locale": "en",
"ckeditor_content": // truncated content
},
{
"id": 2,
"documentId": "k2r5l0i9g3u2j3b4p7f0sed",
"title": "What are chinese hamburgers and why aren't you eating them?",
"slug": "what-are-chinese-hamburgers-and-why-aren-t-you-eating-them",
"createdAt": "2021-11-11T13:33:19.948Z",
"updatedAt": "2023-06-01T14:32:50.984Z",
"publishedAt": "2022-09-22T12:36:48.312Z",
"locale": "en",
"ckeditor_content": // truncated content
},
{
"id": 3,
"documentId": "k6m6l9q0n6v9z2m3i0z5jah"
"title": "7 Places worth visiting for the food alone",
"slug": "7-places-worth-visiting-for-the-food-alone",
"createdAt": "2021-11-12T13:33:19.948Z",
"updatedAt": "2023-06-02T11:30:00.075Z",
"publishedAt": "2023-06-02T11:30:00.075Z",
"locale": "en",
"ckeditor_content": // truncated content
},
{
"id": 4,
"documentId": "d5m4b6z6g5d9e3v1k9n5gbn",
"title": "If you don't finish your plate in these countries, you might offend someone",
"slug": "if-you-don-t-finish-your-plate-in-these-countries-you-might-offend-someone",
"createdAt": "2021-11-15T13:33:19.948Z",
"updatedAt": "2023-06-02T10:59:35.148Z",
"publishedAt": "2022-09-22T12:35:53.899Z",
"locale": "en",
"ckeditor_content": // truncated content
}
],
"meta": {
"pagination": {
"page": 1,
"pageSize": 25,
"pageCount": 1,
"total": 4
}
}
}
示例:与 populate=*
¥Example: With populate=*
使用 populate=*
参数,对 /api/articles
的 GET
请求也会返回所有媒体字段、第一级关系、组件和动态区域。
¥With the populate=*
parameter, a GET
request to /api/articles
also returns all media fields, first-level relations, components and dynamic zones.
以下示例是 articles
内容类型中所有 4 个条目中第一个条目的完整响应(为简洁起见,ID 为 2、3 和 4 的文章中的数据被截断)。
¥The following example is the full response for the first of all 4 entries from the articles
content-types (the data from articles with ids 2, 3, and 4 is truncated for brevity).
向下滚动可以看到响应大小比没有填充时大得多。响应现在包含其他字段(请参阅高亮的行),例如:
¥Scroll down to see that the response size is much bigger than without populate. The response now includes additional fields (see highlighted lines) such as:
-
image
媒体字段(存储有关文章封面的所有信息,包括所有不同的格式),¥the
image
media field (which stores all information about the article cover, including all its different formats), -
blocks
动态区域和seo
组件的第一级字段,¥the first-level fields of the
blocks
dynamic zone and theseo
component, -
category
关系及其字段,¥the
category
relation and its fields, -
甚至还有一些有关其他语言翻 译的文章的信息,如
localizations
对象所示。¥and even some information about the articles translated in other languages, as shown by the
localizations
object.
GET /api/articles?populate=*
{
"data": [
{
"id": 1,
"title": "Here's why you have to try basque cuisine, according to a basque chef",
"slug": "here-s-why-you-have-to-try-basque-cuisine-according-to-a-basque-chef",
"createdAt": "2021-11-09T13:33:19.948Z",
"updatedAt": "2023-06-02T10:57:19.584Z",
"publishedAt": "2022-09-22T09:30:00.208Z",
"locale": "en",
"ckeditor_content": // truncated content
"image": {
"data": {
"id": 12,
"documentId": "o5d4b0l4p8l4o4k5n1l3rxa",
"name": "Basque dish",
"alternativeText": "Basque dish",
"caption": "Basque dish",
"width": 758,
"height": 506,
"formats": {
"thumbnail": {
"name": "thumbnail_https://4d40-2a01-cb00-c8b-1800-7cbb-7da-ea9d-2011.ngrok.io/uploads/basque_cuisine_17fa4567e0.jpeg",
"hash": "thumbnail_basque_cuisine_17fa4567e0_f033424240",
"ext": ".jpeg",
"mime": "image/jpeg",
"width": 234,
"height": 156,
"size": 11.31,
"path": null,
"url": "/uploads/thumbnail_basque_cuisine_17fa4567e0_f033424240.jpeg"
},
"medium": {
"name": "medium_https://4d40-2a01-cb00-c8b-1800-7cbb-7da-ea9d-2011.ngrok.io/uploads/basque_cuisine_17fa4567e0.jpeg",
"hash": "medium_basque_cuisine_17fa4567e0_f033424240",
"ext": ".jpeg",
"mime": "image/jpeg",
"width": 750,
"height": 501,
"size": 82.09,
"path": null,
"url": "/uploads/medium_basque_cuisine_17fa4567e0_f033424240.jpeg"
},
"small": {
"name": "small_https://4d40-2a01-cb00-c8b-1800-7cbb-7da-ea9d-2011.ngrok.io/uploads/basque_cuisine_17fa4567e0.jpeg",
"hash": "small_basque_cuisine_17fa4567e0_f033424240",
"ext": ".jpeg",
"mime": "image/jpeg",
"width": 500,
"height": 334,
"size": 41.03,
"path": null,
"url": "/uploads/small_basque_cuisine_17fa4567e0_f033424240.jpeg"
}
},
"hash": "basque_cuisine_17fa4567e0_f033424240",
"ext": ".jpeg",
"mime": "image/jpeg",
"size": 58.209999999999994,
"url": "/uploads/basque_cuisine_17fa4567e0_f033424240.jpeg",
"previewUrl": null,
"provider": "local",
"provider_metadata": null,
"createdAt": "2021-11-23T14:05:33.460Z",
"updatedAt": "2021-11-23T14:05:46.084Z"
}
}
},
"blocks": [
{
"id": 2,
"__component": "blocks.related-articles"
},
{
"id": 2,
"documentId": "w8r5k8o8v0t9l9e0d7y6vco",
"__component": "blocks.cta-command-line",
"theme": "primary",
"title": "Want to give a try to a Strapi starter?",
"text": "❤️",
"commandLine": "git clone https://github.com/strapi/nextjs-corporate-starter.git"
}
],
"seo": {
"id": 1,
"documentId": "h7c8d0u3i3q5v1j3j3r4cxf",
"metaTitle": "Articles - FoodAdvisor",
"metaDescription": "Discover our articles about food, restaurants, bars and more! - FoodAdvisor",
"keywords": "food",
"metaRobots": null,
"structuredData": null,
"metaViewport": null,
"canonicalURL": null
},
"category": {
"data": {
"id": 4,
"documentId": "t1t3d9k6n1k5a6r8l7f8rox",
"name": "European",
"slug": "european",
"createdAt": "2021-11-09T13:33:20.123Z",
"updatedAt": "2021-11-09T13:33:20.123Z"
}
},
"localizations": {
"data": [
{
"id": 10,
"documentId": "h7c8d0u3i3q5v1j3j3r4cxf",
"title": "Voici pourquoi il faut essayer la cuisine basque, selon un chef basque",
"slug": "voici-pourquoi-il-faut-essayer-la-cuisine-basque-selon-un-chef-basque",
"createdAt": "2021-11-18T13:33:19.948Z",
"updatedAt": "2023-06-02T10:57:19.606Z",
"publishedAt": "2022-09-22T13:00:00.069Z",
"locale": "fr-FR",
"ckeditor_content": // truncated content
}
]
}
}
},
{
"id": 2,
// truncated content
},
{
"id": 3,
// truncated content
},
{
"id": 4,
// truncated content
}
],
"meta": {
"pagination": {
"page": 1,
"pageSize": 25,
"pageCount": 1,
"total": 4
}
}
}