10

NextTokenを使用してSimpleDBデータをページ転送する方法を知っています。しかし、前のページをどの程度正確に処理しますか?私は.NETを使用していますが、それは重要ではないと思います。私は一般的な戦略にもっと興味があります。

MikeCulverのAnIntroductionto Amazon SimpleDBウェビナーでは、パンくずリストが使用されていると述べていますが、ビデオではそれらを実装していません。

編集:ビデオは、バックワードページングを実装するサンプルプロジェクトについて言及していますが、ダウンロードのURLが表示される前にビデオが終了します。私が見つけた1つのサンプルプロジェクトは、ページングを扱っていませんでした。

4

4 に答える 4

11

次のページに移動するときは、「次のページ」のみを許可し、任意のページングを許可しないことで、ユースケースを簡素化できる場合があります。SimpleDBでこれを行うには、LIMIT句を使用します。

SELECT title, summary, votecount FROM posts WHERE userid = '000022656' LIMIT 25

NextTokenの処理方法はすでに知っていますが、この戦術を使用すると、次のトークンのブレッドクラムトレイルを保存し(Webセッションなど)、前のNextTokenではなく前のNextTokenでクエリを再発行することで、「前のページ」をサポートできます。後続のもの。

ただし、SimpleDBで任意のページネーションを処理する一般的なケースは、前と次で同じです。一般的な場合、ユーザーは4ページまたは6ページにアクセスすることなく、5などの任意のページ番号をクリックできます。

SimpleDBでこれを処理するには、NextTokenが正しく機能するためにWHERE句が同じである必要があるだけであるという事実を使用します。したがって、すべてのページを順番にクエリして、介在するすべてのアイテムをプルダウンするのではなく、通常は2つのステップで実行できます。

  1. 目的のページを開始する場所の制限値と、必要な実際の属性の代わりにSELECT count(*)を使用してクエリを発行します。
  2. ステップ1のNextTokenを使用して、目的の属性とページサイズをLIMITとして使用して実際のページデータをフェッチします

したがって、擬似コードでは:

int targetPage, pageSize;
...
int jumpLimit = pageSize * (targetPage - 1);
String query = "SELECT %1 FROM posts WHERE userid = '000022656' LIMIT %2";
String output = "title, summary, votecount";
Result temp = sdb.select(query, "count(*)", jumpLimit);
Result data = sdb.select(query, output, pageSize, temp.getToken());

ここで、%1と%2は文字列置換であり、「sdb.select()」はSimpleDB呼び出しとともに文字列置換コードを含む架空のメソッドです。

SimpleDBへの2回の呼び出しでこれを実行できるかどうか(コードに示されているように)は、WHERE句の複雑さとデータセットのサイズによって異なります。上記のコードは、クエリの実行に5秒以上かかった場合に、一時的な結果が部分的なカウントを返す可能性があるという点で単純化されています。適切なカウントに達するまで、その行をループに入れたいと思うでしょう。コードをもう少し現実的にするために、メソッド内にコードを配置し、文字列の置換を取り除きます。

private Result fetchPage(String query, int targetPage)
{
    int pageSize = extractLimitValue(query);
    int skipLimit = pageSize * (targetPage - 1);
    String token = skipAhead(query, skipLimit);
    return sdb.select(query, token);
}

private String skipAhead(String query, int skipLimit)
{
    String tempQuery = replaceClause(query, "SELECT", "count(*)");
    int accumulatedCount = 0;
    String token = "";
    do {
        int tempLimit = skipLimit - accumulatedCount;
        tempQuery = replaceClause(tempQuery , "LIMIT", tempLimit + "");
        Result tempResult = sdb.select(query, token);
        token = tempResult.getToken();
        accumulatedCount += tempResult.getCount();
    } while (accumulatedCount < skipLimit);
    return token;
}

private int extractLimitValue(String query) {...}
private String replaceClause(String query, String clause, String value){...}

これはエラー処理なしの一般的な考え方であり、ページ1を除く任意のページで機能します。

于 2009-12-02T13:05:44.543 に答える
1

茶色のバッグのウェビナーの1つで、トークンを再送信すると、対応する結果セットが返されるとのことでした。

私はそれを試していません、そしてそれはただの考えです、しかしあなたが前に進んでいるときにトークンのリストを構築するのはどうですか?戻るには、リストを逆方向にトラバースして、トークンを再送信します(そしてステートメントを選択します)。

于 2009-11-29T15:49:04.160 に答える
0

トークンの取得に行き詰まっています-それはRequestIdと同じものですか?

私が使用しているPHPSimpleDBライブラリはそれを返さないようです。 http://sourceforge.net/projects/php-sdb/

このドキュメントを見つけました http://docs.amazonwebservices.com/AmazonSimpleDB/2009-04-15/DeveloperGuide/index.html?SDB_API_Select.html

これはnextToken要素があることを示しているようですが、サンプルの応答では、RequestId...と表示されています。

それを理解しました-私たちのPHPlibは、実際に、アクセスできる場所からネクストトークンを抽象化していました。ライブラリを掘り下げて見つけました。

于 2010-01-17T01:12:46.867 に答える
0

公式のSimpleDBAPIを使用して、上記で提案したサンプリングのJavaバージョンを作成しました。これは誰にとっても役立つかもしれません。

private static Set<String> getSdbAttributes(AmazonSimpleDBClient client,
            String domainName, int sampleSize) {
        if (!client.listDomains().getDomainNames().contains(domainName)) {
        throw new IllegalArgumentException("SimpleDB domain '" + domainName
                + "' not accessible from given client instance");
    }

    int domainCount = client.domainMetadata(
            new DomainMetadataRequest(domainName)).getItemCount();
    if (domainCount < sampleSize) {
        throw new IllegalArgumentException("SimpleDB domain '" + domainName
                + "' does not have enough entries for accurate sampling.");
    }

    int avgSkipCount = domainCount / sampleSize;
    int processedCount = 0;
    String nextToken = null;
    Set<String> attributeNames = new HashSet<String>();
    Random r = new Random();
    do {
        int nextSkipCount = r.nextInt(avgSkipCount * 2) + 1;

        SelectResult countResponse = client.select(new SelectRequest(
                "select count(*) from `" + domainName + "` limit "
                        + nextSkipCount).withNextToken(nextToken));

        nextToken = countResponse.getNextToken();

        processedCount += Integer.parseInt(countResponse.getItems().get(0)
                .getAttributes().get(0).getValue());

        SelectResult getResponse = client.select(new SelectRequest(
                "select * from `" + domainName + "` limit 1")
                .withNextToken(nextToken));

        nextToken = getResponse.getNextToken();

        processedCount++;

        if (getResponse.getItems().size() > 0) {
            for (Attribute a : getResponse.getItems().get(0)
                    .getAttributes()) {
                attributeNames.add(a.getName());
            }
        }
    } while (domainCount > processedCount);
    return attributeNames;
}
于 2012-02-14T14:28:35.443 に答える