AppSearch

AppSearch 是一款用于管理本地存储的结构化数据的高性能设备搜索解决方案。它包含用于索引数据和使用全文搜索检索数据的 API。应用程序可以使用 AppSearch 提供自定义的应用内搜索功能,使用户即使在离线状态下也可以搜索内容。

Diagram illustrating indexing and searching within AppSearch

AppSearch 提供以下功能

  • 具有低 I/O 使用率的快速、移动优先的存储实现
  • 对大型数据集进行高效的索引和查询
  • 多语言支持,例如英语和西班牙语
  • 相关性排名和使用评分

由于 I/O 使用率较低,与 SQLite 相比,AppSearch 为大型数据集的索引和搜索提供了更低的延迟。AppSearch 通过支持单一查询简化了跨类型查询,而 SQLite 合并了来自多个表的查询结果。

为了说明 AppSearch 的功能,让我们以一个管理用户收藏歌曲并允许用户轻松搜索歌曲的音乐应用程序为例。用户可以欣赏来自世界各地的音乐,歌曲标题使用不同的语言,AppSearch 原生支持索引和查询。当用户按标题或艺术家姓名搜索歌曲时,应用程序只需将请求传递给 AppSearch 以快速有效地检索匹配的歌曲。应用程序显示结果,使用户能够快速开始播放他们喜欢的歌曲。

设置

要在您的应用程序中使用 AppSearch,请将以下依赖项添加到应用程序的 build.gradle 文件中

Groovy

dependencies {
    def appsearch_version = "1.1.0-alpha06"

    implementation "androidx.appsearch:appsearch:$appsearch_version"
    // Use kapt instead of annotationProcessor if writing Kotlin classes
    annotationProcessor "androidx.appsearch:appsearch-compiler:$appsearch_version"

    implementation "androidx.appsearch:appsearch-local-storage:$appsearch_version"
    // PlatformStorage is compatible with Android 12+ devices, and offers additional features
    // to LocalStorage.
    implementation "androidx.appsearch:appsearch-platform-storage:$appsearch_version"
}

Kotlin

dependencies {
    val appsearch_version = "1.1.0-alpha06"

    implementation("androidx.appsearch:appsearch:$appsearch_version")
    // Use annotationProcessor instead of kapt if writing Java classes
    kapt("androidx.appsearch:appsearch-compiler:$appsearch_version")

    implementation("androidx.appsearch:appsearch-local-storage:$appsearch_version")
    // PlatformStorage is compatible with Android 12+ devices, and offers additional features
    // to LocalStorage.
    implementation("androidx.appsearch:appsearch-platform-storage:$appsearch_version")
}

AppSearch 概念

下图说明了 AppSearch 概念及其交互。

客户端应用程序及其与以下 AppSearch 概念的交互的图表概述:AppSearch 数据库、模式、模式类型、文档、会话和搜索。 图 1. AppSearch 概念图:AppSearch 数据库、模式、模式类型、文档、会话和搜索。

数据库和会话

AppSearch 数据库是符合数据库模式的文档集合。客户端应用程序通过提供其应用程序上下文和数据库名称来创建数据库。数据库只能由创建它们的应用程序打开。打开数据库时,会返回一个会话以与数据库交互。会话是调用 AppSearch API 的入口点,并在客户端应用程序关闭它之前保持打开状态。

模式和模式类型

模式表示 AppSearch 数据库中数据的组织结构。

模式由表示唯一数据类型的模式类型组成。模式类型包含包含名称、数据类型和基数的属性。将模式类型添加到数据库模式后,就可以创建该模式类型的文档并将其添加到数据库中。

文档

在 AppSearch 中,数据单元表示为文档。AppSearch 数据库中的每个文档都由其命名空间和 ID 唯一标识。当只需要查询一个来源时,例如用户帐户,命名空间用于将来自不同来源的数据分开。

文档包含创建时间戳、生存时间 (TTL) 和可在检索期间用于排名的分数。文档还分配了一个模式类型,该类型描述了文档必须具有的其他数据属性。

文档类是文档的抽象。它包含注释字段,这些字段表示文档的内容。默认情况下,文档类的名称设置模式类型的名称。

文档被索引,可以通过提供查询来搜索。如果文档包含查询中的术语或匹配其他搜索规范,则该文档将匹配并包含在搜索结果中。结果根据其分数和排名策略排序。搜索结果由您可以顺序检索的页面表示。

AppSearch 提供了 搜索自定义,例如过滤器、页面大小配置和摘要。

平台存储与本地存储

AppSearch 提供两种存储解决方案:LocalStorage 和 PlatformStorage。使用 LocalStorage,您的应用程序管理一个位于应用程序数据目录中的特定于应用程序的索引。使用 PlatformStorage,您的应用程序会贡献到系统范围的中央索引。中央索引中的数据访问仅限于您的应用程序已贡献的数据以及其他应用程序已明确与您共享的数据。LocalStorage 和 PlatformStorage 共享相同的 API,并且可以根据设备的版本进行互换

Kotlin

if (BuildCompat.isAtLeastS()) {
    appSearchSessionFuture.setFuture(
        PlatformStorage.createSearchSession(
            PlatformStorage.SearchContext.Builder(mContext, DATABASE_NAME)
               .build()
        )
    )
} else {
    appSearchSessionFuture.setFuture(
        LocalStorage.createSearchSession(
            LocalStorage.SearchContext.Builder(mContext, DATABASE_NAME)
                .build()
        )
    )
}

Java

if (BuildCompat.isAtLeastS()) {
    mAppSearchSessionFuture.setFuture(PlatformStorage.createSearchSession(
            new PlatformStorage.SearchContext.Builder(mContext, DATABASE_NAME)
                    .build()));
} else {
    mAppSearchSessionFuture.setFuture(LocalStorage.createSearchSession(
            new LocalStorage.SearchContext.Builder(mContext, DATABASE_NAME)
                    .build()));
}

使用 PlatformStorage,您的应用程序可以安全地与其他应用程序共享数据,以允许它们搜索您的应用程序数据。只读应用程序数据共享通过证书握手授予,以确保其他应用程序有权读取数据。有关此 API 的更多信息,请参阅 setSchemaTypeVisibilityForPackage() 文档。

此外,已编入索引的数据可以显示在系统 UI 表面上。应用程序可以选择不将其部分或全部数据显示在系统 UI 表面上。有关此 API 的更多信息,请参阅 setSchemaTypeDisplayedBySystem() 文档。

功能 LocalStorage(与 Android 4.0+ 兼容) PlatformStorage(与 Android 12+ 兼容)
高效的全文搜索
多语言支持
减少二进制文件大小
应用程序之间的数据共享
能够在系统 UI 表面上显示数据
可以索引无限数量的文档大小和数量
更快的操作,无需额外的绑定程序延迟

在 LocalStorage 和 PlatformStorage 之间进行选择时,还需要考虑其他权衡。因为 PlatformStorage 在 AppSearch 系统服务之上包装了 Jetpack API,所以与使用 LocalStorage 相比,APK 大小影响最小。但是,这也意味着在调用 AppSearch 系统服务时,AppSearch 操作会产生额外的绑定程序延迟。使用 PlatformStorage,AppSearch 会限制应用程序可以索引的文档数量和文档大小,以确保高效的中央索引。

开始使用 AppSearch

本节中的示例展示了如何使用 AppSearch API 与假设的笔记应用程序集成。

编写文档类

与 AppSearch 集成的第一步是编写一个文档类来描述要插入数据库中的数据。使用 @Document 注释将类标记为文档类。您可以使用文档类的实例将文档放入数据库并从数据库中检索文档。

以下代码定义了一个 Note 文档类,其中包含一个使用 @Document.StringProperty 注解的字段,用于索引 Note 对象的文本。

Kotlin

@Document
public data class Note(

    // Required field for a document class. All documents MUST have a namespace.
    @Document.Namespace
    val namespace: String,

    // Required field for a document class. All documents MUST have an Id.
    @Document.Id
    val id: String,

    // Optional field for a document class, used to set the score of the
    // document. If this is not included in a document class, the score is set
    // to a default of 0.
    @Document.Score
    val score: Int,

    // Optional field for a document class, used to index a note's text for this
    // document class.
    @Document.StringProperty(indexingType = AppSearchSchema.StringPropertyConfig.INDEXING_TYPE_PREFIXES)
    val text: String
)

Java

@Document
public class Note {

  // Required field for a document class. All documents MUST have a namespace.
  @Document.Namespace
  private final String namespace;

  // Required field for a document class. All documents MUST have an Id.
  @Document.Id
  private final String id;

  // Optional field for a document class, used to set the score of the
  // document. If this is not included in a document class, the score is set
  // to a default of 0.
  @Document.Score
  private final int score;

  // Optional field for a document class, used to index a note's text for this
  // document class.
  @Document.StringProperty(indexingType = StringPropertyConfig.INDEXING_TYPE_PREFIXES)
  private final String text;

  Note(@NonNull String id, @NonNull String namespace, int score, @NonNull String text) {
    this.id = Objects.requireNonNull(id);
    this.namespace = Objects.requireNonNull(namespace);
    this.score = score;
    this.text = Objects.requireNonNull(text);
  }

  @NonNull
  public String getNamespace() {
    return namespace;
  }

  @NonNull
  public String getId() {
    return id;
  }

  public int getScore() {
    return score;
  }

  @NonNull
  public String getText() {
     return text;
  }
}

打开数据库

在处理文档之前,必须先创建一个数据库。以下代码创建一个名为 notes_app 的新数据库,并获取一个 ListenableFuture,该对象表示与数据库的连接,并提供数据库操作的 API。

Kotlin

val context: Context = getApplicationContext()
val sessionFuture = LocalStorage.createSearchSession(
    LocalStorage.SearchContext.Builder(context, /*databaseName=*/"notes_app")
    .build()
)

Java

Context context = getApplicationContext();
ListenableFuture<AppSearchSession> sessionFuture = LocalStorage.createSearchSession(
       new LocalStorage.SearchContext.Builder(context, /*databaseName=*/ "notes_app")
               .build()
);

设置模式

在将文档放入数据库和从数据库中检索文档之前,必须设置模式。数据库模式由不同类型的结构化数据组成,称为“模式类型”。以下代码通过提供文档类作为模式类型来设置模式。

Kotlin

val setSchemaRequest = SetSchemaRequest.Builder().addDocumentClasses(Note::class.java)
    .build()
val setSchemaFuture = Futures.transformAsync(
    sessionFuture,
    { session ->
        session?.setSchema(setSchemaRequest)
    }, mExecutor
)

Java

SetSchemaRequest setSchemaRequest = new SetSchemaRequest.Builder().addDocumentClasses(Note.class)
       .build();
ListenableFuture<SetSchemaResponse> setSchemaFuture =
       Futures.transformAsync(sessionFuture, session -> session.setSchema(setSchemaRequest), mExecutor);

将文档放入数据库

添加模式类型后,您可以将该类型的文档添加到数据库中。以下代码使用 Note 文档类构建器构建一个模式类型为 Note 的文档。它将文档命名空间设置为 user1,以表示此示例的任意用户。然后将文档插入数据库,并附加一个监听器来处理 put 操作的结果。

Kotlin

val note = Note(
    namespace="user1",
    id="noteId",
    score=10,
    text="Buy fresh fruit"
)

val putRequest = PutDocumentsRequest.Builder().addDocuments(note).build()
val putFuture = Futures.transformAsync(
    sessionFuture,
    { session ->
        session?.put(putRequest)
    }, mExecutor
)

Futures.addCallback(
    putFuture,
    object : FutureCallback<AppSearchBatchResult<String, Void>?> {
        override fun onSuccess(result: AppSearchBatchResult<String, Void>?) {

            // Gets map of successful results from Id to Void
            val successfulResults = result?.successes

            // Gets map of failed results from Id to AppSearchResult
            val failedResults = result?.failures
        }

        override fun onFailure(t: Throwable) {
            Log.e(TAG, "Failed to put documents.", t)
        }
    },
    mExecutor
)

Java

Note note = new Note(/*namespace=*/"user1", /*id=*/
                "noteId", /*score=*/ 10, /*text=*/ "Buy fresh fruit!");

PutDocumentsRequest putRequest = new PutDocumentsRequest.Builder().addDocuments(note)
       .build();
ListenableFuture<AppSearchBatchResult<String, Void>> putFuture =
       Futures.transformAsync(sessionFuture, session -> session.put(putRequest), mExecutor);

Futures.addCallback(putFuture, new FutureCallback<AppSearchBatchResult<String, Void>>() {
   @Override
   public void onSuccess(@Nullable AppSearchBatchResult<String, Void> result) {

     // Gets map of successful results from Id to Void
     Map<String, Void> successfulResults = result.getSuccesses();

     // Gets map of failed results from Id to AppSearchResult
     Map<String, AppSearchResult<Void>> failedResults = result.getFailures();
   }

   @Override
   public void onFailure(@NonNull Throwable t) {
      Log.e(TAG, "Failed to put documents.", t);
   }
}, mExecutor);

您可以使用本节中介绍的搜索操作搜索已编制索引的文档。以下代码对数据库中属于 user1 命名空间的文档执行针对术语“fruit”的查询。

Kotlin

val searchSpec = SearchSpec.Builder()
    .addFilterNamespaces("user1")
    .build();

val searchFuture = Futures.transform(
    sessionFuture,
    { session ->
        session?.search("fruit", searchSpec)
    },
    mExecutor
)
Futures.addCallback(
    searchFuture,
    object : FutureCallback<SearchResults> {
        override fun onSuccess(searchResults: SearchResults?) {
            iterateSearchResults(searchResults)
        }

        override fun onFailure(t: Throwable?) {
            Log.e("TAG", "Failed to search notes in AppSearch.", t)
        }
    },
    mExecutor
)

Java

SearchSpec searchSpec = new SearchSpec.Builder()
       .addFilterNamespaces("user1")
       .build();

ListenableFuture<SearchResults> searchFuture =
       Futures.transform(sessionFuture, session -> session.search("fruit", searchSpec),
       mExecutor);

Futures.addCallback(searchFuture,
       new FutureCallback<SearchResults>() {
           @Override
           public void onSuccess(@Nullable SearchResults searchResults) {
               iterateSearchResults(searchResults);
           }

           @Override
           public void onFailure(@NonNull Throwable t) {
               Log.e(TAG, "Failed to search notes in AppSearch.", t);
           }
       }, mExecutor);

遍历搜索结果

搜索返回一个 SearchResults 实例,该实例可以访问 SearchResult 对象的页面。每个 SearchResult 都包含其匹配的 GenericDocument,这是所有文档都转换成的通用文档形式。以下代码获取搜索结果的第一页,并将结果转换回 Note 文档。

Kotlin

Futures.transform(
    searchResults?.nextPage,
    { page: List<SearchResult>? ->
        // Gets GenericDocument from SearchResult.
        val genericDocument: GenericDocument = page!![0].genericDocument
        val schemaType = genericDocument.schemaType
        val note: Note? = try {
            if (schemaType == "Note") {
                // Converts GenericDocument object to Note object.
                genericDocument.toDocumentClass(Note::class.java)
            } else null
        } catch (e: AppSearchException) {
            Log.e(
                TAG,
                "Failed to convert GenericDocument to Note",
                e
            )
            null
        }
        note
    },
    mExecutor
)

Java

Futures.transform(searchResults.getNextPage(), page -> {
  // Gets GenericDocument from SearchResult.
  GenericDocument genericDocument = page.get(0).getGenericDocument();
  String schemaType = genericDocument.getSchemaType();

  Note note = null;

  if (schemaType.equals("Note")) {
    try {
      // Converts GenericDocument object to Note object.
      note = genericDocument.toDocumentClass(Note.class);
    } catch (AppSearchException e) {
      Log.e(TAG, "Failed to convert GenericDocument to Note", e);
    }
  }

  return note;
}, mExecutor);

删除文档

当用户删除笔记时,应用程序会从数据库中删除相应的 Note 文档。这确保了笔记将不再出现在查询结果中。以下代码明确请求根据 ID 从数据库中删除 Note 文档。

Kotlin

val removeRequest = RemoveByDocumentIdRequest.Builder("user1")
    .addIds("noteId")
    .build()

val removeFuture = Futures.transformAsync(
    sessionFuture, { session ->
        session?.remove(removeRequest)
    },
    mExecutor
)

Java

RemoveByDocumentIdRequest removeRequest = new RemoveByDocumentIdRequest.Builder("user1")
       .addIds("noteId")
       .build();

ListenableFuture<AppSearchBatchResult<String, Void>> removeFuture =
       Futures.transformAsync(sessionFuture, session -> session.remove(removeRequest), mExecutor);

持久化到磁盘

通过调用 requestFlush(),应定期将数据库更新持久化到磁盘。以下代码调用 requestFlush() 并附加一个监听器以确定调用是否成功。

Kotlin

val requestFlushFuture = Futures.transformAsync(
    sessionFuture,
    { session -> session?.requestFlush() }, mExecutor
)

Futures.addCallback(requestFlushFuture, object : FutureCallback<Void?> {
    override fun onSuccess(result: Void?) {
        // Success! Database updates have been persisted to disk.
    }

    override fun onFailure(t: Throwable) {
        Log.e(TAG, "Failed to flush database updates.", t)
    }
}, mExecutor)

Java

ListenableFuture<Void> requestFlushFuture = Futures.transformAsync(sessionFuture,
        session -> session.requestFlush(), mExecutor);

Futures.addCallback(requestFlushFuture, new FutureCallback<Void>() {
    @Override
    public void onSuccess(@Nullable Void result) {
        // Success! Database updates have been persisted to disk.
    }

    @Override
    public void onFailure(@NonNull Throwable t) {
        Log.e(TAG, "Failed to flush database updates.", t);
    }
}, mExecutor);

关闭会话

当应用程序不再调用任何数据库操作时,应关闭 AppSearchSession。以下代码关闭先前打开的 AppSearch 会话并将所有更新持久化到磁盘。

Kotlin

val closeFuture = Futures.transform<AppSearchSession, Unit>(sessionFuture,
    { session ->
        session?.close()
        Unit
    }, mExecutor
)

Java

ListenableFuture<Void> closeFuture = Futures.transform(sessionFuture, session -> {
   session.close();
   return null;
}, mExecutor);

其他资源

要了解有关 AppSearch 的更多信息,请参阅以下其他资源

示例

提供反馈

通过以下资源与我们分享您的反馈和想法

问题跟踪器

报告错误,以便我们修复它们。