Typical DP Contest G: 辞書順 (動的計画法)

解法

dp[i][c]:= i文字目以降の部分列で文字cから始まるものの個数、とする。i=N-1から順々にdpを埋めていき、埋め終わったらそこから文字列を復元すれば良い。

コード

import java.io.BufferedReader;
import java.io.InputStreamReader;

public class Main {

	public static void main(String[] args) throws Exception {
		BufferedReader r = new BufferedReader(new InputStreamReader(System.in), 1);

		char[] s = r.readLine().toCharArray();
		long K = Long.parseLong(r.readLine());

		int N = s.length;

		// dp[i][c]:= i文字目以降の部分列で文字cから始まるものの個数
		long[][] dp = new long[N + 1][26];
		for (int i = N - 1; i >= 0; i--) {
			for (int j = 0; j < 26; j++) {
				if (s[i] == (char) j + 'a') {
					dp[i][j] = 1;

					// i文字目以降の部分文字列の数は、i+1文字目以降の部分文字列の数の和+1である
					for (int k = 0; k < 26; k++) {
						dp[i][j] = Math.min(dp[i][j] + dp[i + 1][k], K + 1);
						if (dp[i][j] == K + 1) {
							break;
						}
					}
				} else {
					dp[i][j] = dp[i + 1][j];
				}
			}
		}

		long sum = 0;
		for (int i = 0; i < 26; i++) {
			sum = Math.min(sum + dp[0][i], K + 1);
		}
		if (K > sum) {
			System.out.println("Eel");
			return;
		}

		StringBuilder sb = new StringBuilder();
		for (int i = 0; i < dp.length; i++) {
			for (int c = 0; c < 26; c++) {
				if (K > dp[i][c]) {
					K -= dp[i][c];
				} else {
					while (s[i] != (char) c + 'a') {
						i++;
					}
					sb.append(s[i]);
					K--;// 部分文字列s[i]を出したので1つ減る
					break;
				}
			}
			if (K == 0) {
				break;
			}
		}
		System.out.println(sb.toString());

	}
}

大きいテストケースはBufferedReaderとInputStreamReaderを使わないと通らないなど……