結果
問題 | No.2388 At Least K-Characters |
ユーザー | 👑 p-adic |
提出日時 | 2023-08-07 11:01:25 |
言語 | C++17 (gcc 13.3.0 + boost 1.87.0) |
結果 |
AC
|
実行時間 | 1,587 ms / 4,000 ms |
コード長 | 56,452 bytes |
コンパイル時間 | 4,525 ms |
コンパイル使用メモリ | 253,188 KB |
実行使用メモリ | 110,084 KB |
最終ジャッジ日時 | 2024-07-05 04:29:14 |
合計ジャッジ時間 | 34,848 ms |
ジャッジサーバーID (参考情報) |
judge2 / judge5 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 12 ms
56,932 KB |
testcase_01 | AC | 12 ms
56,800 KB |
testcase_02 | AC | 12 ms
56,808 KB |
testcase_03 | AC | 12 ms
56,672 KB |
testcase_04 | AC | 12 ms
56,676 KB |
testcase_05 | AC | 12 ms
56,800 KB |
testcase_06 | AC | 12 ms
56,800 KB |
testcase_07 | AC | 12 ms
56,676 KB |
testcase_08 | AC | 12 ms
56,680 KB |
testcase_09 | AC | 12 ms
56,804 KB |
testcase_10 | AC | 12 ms
56,932 KB |
testcase_11 | AC | 13 ms
56,672 KB |
testcase_12 | AC | 12 ms
56,928 KB |
testcase_13 | AC | 15 ms
56,832 KB |
testcase_14 | AC | 15 ms
56,824 KB |
testcase_15 | AC | 15 ms
56,828 KB |
testcase_16 | AC | 1,402 ms
109,888 KB |
testcase_17 | AC | 1,587 ms
109,960 KB |
testcase_18 | AC | 1,380 ms
109,968 KB |
testcase_19 | AC | 1,375 ms
109,892 KB |
testcase_20 | AC | 1,364 ms
109,960 KB |
testcase_21 | AC | 1,376 ms
110,000 KB |
testcase_22 | AC | 1,363 ms
110,040 KB |
testcase_23 | AC | 1,356 ms
109,920 KB |
testcase_24 | AC | 1,374 ms
110,084 KB |
testcase_25 | AC | 1,357 ms
109,984 KB |
testcase_26 | AC | 1,354 ms
109,896 KB |
testcase_27 | AC | 1,356 ms
110,008 KB |
testcase_28 | AC | 1,349 ms
109,944 KB |
testcase_29 | AC | 1,353 ms
109,912 KB |
testcase_30 | AC | 1,360 ms
109,904 KB |
testcase_31 | AC | 1,362 ms
109,984 KB |
testcase_32 | AC | 1,355 ms
109,924 KB |
testcase_33 | AC | 1,361 ms
109,896 KB |
testcase_34 | AC | 1,356 ms
110,008 KB |
testcase_35 | AC | 1,365 ms
109,884 KB |
testcase_36 | AC | 1,352 ms
110,064 KB |
ソースコード
#ifdef DEBUG #define _GLIBCXX_DEBUG #define UNTIE ios_base::sync_with_stdio( false ); cin.tie( nullptr ); signal( SIGABRT , &AlertAbort ) #define DEXPR( LL , BOUND , VALUE , DEBUG_VALUE ) CEXPR( LL , BOUND , DEBUG_VALUE ) #define CERR( MESSAGE ) cerr << MESSAGE << endl; #define COUT( ANSWER ) cout << ANSWER << endl #define ASSERT( A , MIN , MAX ) CERR( "ASSERTチェック: " << ( MIN ) << ( ( MIN ) <= A ? "<=" : ">" ) << A << ( A <= ( MAX ) ? "<=" : ">" ) << ( MAX ) ); assert( ( MIN ) <= A && A <= ( MAX ) ) #define LIBRARY_SEARCH bool searched_library = false; LibrarySearch( searched_library ); if( searched_library ){ QUIT; }; #define START_WATCH( PROCESS_NAME ) StartWatch( PROCESS_NAME ) #define STOP_WATCH( HOW_MANY_TIMES ) StopWatch( HOW_MANY_TIMES ) #else #pragma GCC optimize ( "O3" ) #pragma GCC optimize( "unroll-loops" ) #pragma GCC target ( "sse4.2,fma,avx2,popcnt,lzcnt,bmi2" ) #define UNTIE ios_base::sync_with_stdio( false ); cin.tie( nullptr ) #define DEXPR( LL , BOUND , VALUE , DEBUG_VALUE ) CEXPR( LL , BOUND , VALUE ) #define CERR( MESSAGE ) #define COUT( ANSWER ) cout << ANSWER << "\n" #define ASSERT( A , MIN , MAX ) assert( ( MIN ) <= A && A <= ( MAX ) ) #define LIBRARY_SEARCH #define START_WATCH( PROCESS_NAME ) #define STOP_WATCH( HOW_MANY_TIMES ) #endif // #define RANDOM_TEST #include <bits/stdc++.h> using namespace std; using uint = unsigned int; using ll = long long; using ull = unsigned long long; #define ATT __attribute__( ( target( "sse4.2,fma,avx2,popcnt,lzcnt,bmi2" ) ) ) #define TYPE_OF( VAR ) decay_t<decltype( VAR )> #define CEXPR( LL , BOUND , VALUE ) constexpr LL BOUND = VALUE #define CIN( LL , A ) LL A; cin >> A #define CIN_ASSERT( A , MIN , MAX ) TYPE_OF( MAX ) A; SET_ASSERT( A , MIN , MAX ) #define GETLINE( A ) string A; getline( cin , A ) #define GETLINE_SEPARATE( A , SEPARATOR ) string A; getline( cin , A , SEPARATOR ) #define FOR( VAR , INITIAL , FINAL_PLUS_ONE ) for( TYPE_OF( FINAL_PLUS_ONE ) VAR = INITIAL ; VAR < FINAL_PLUS_ONE ; VAR ++ ) #define FOREQ( VAR , INITIAL , FINAL ) for( TYPE_OF( FINAL ) VAR = INITIAL ; VAR <= FINAL ; VAR ++ ) #define FOREQINV( VAR , INITIAL , FINAL ) for( TYPE_OF( INITIAL ) VAR = INITIAL ; VAR >= FINAL ; VAR -- ) #define AUTO_ITR( ARRAY ) auto itr_ ## ARRAY = ARRAY .begin() , end_ ## ARRAY = ARRAY .end() #define FOR_ITR( ARRAY ) for( AUTO_ITR( ARRAY ) , itr = itr_ ## ARRAY ; itr_ ## ARRAY != end_ ## ARRAY ; itr_ ## ARRAY ++ , itr++ ) #define REPEAT( HOW_MANY_TIMES ) FOR( VARIABLE_FOR_REPEAT_ ## HOW_MANY_TIMES , 0 , HOW_MANY_TIMES ) #define QUIT return 0 #define SET_PRECISION( DECIMAL_DIGITS ) cout << fixed << setprecision( DECIMAL_DIGITS_ ) #ifdef DEBUG inline void AlertAbort( int n ) { CERR( "abort関数が呼ばれました。assertマクロのメッセージが出力されていない場合はオーバーフローの有無を確認をしてください。" ); } void StartWatch( const string& process_name = "nothing" ); void StopWatch( const int& how_many_times = 1 ); #endif #if defined( DEBUG ) && defined( RANDOM_TEST ) inline CEXPR( int , bound_random_test_num , 1000 ); #define START_MAIN FOR( random_test_num , 0 , bound_random_test_num ){ CERR( "(" << random_test_num << ")" ); ll GetRand( const ll& Rand_min , const ll& Rand_max ); #define SET_ASSERT( A , MIN , MAX ) CERR( #A << " = " << ( A = GetRand( MIN , MAX ) ) ) #define RETURN( ANSWER ) if( ( ANSWER ) == guchoku ){ CERR( ( ANSWER ) << " == " << guchoku ); continue; } else { CERR( ( ANSWER ) << " != " << guchoku ); QUIT; } #define FINISH_MAIN CERR( "" ); } #else #define START_MAIN #define SET_ASSERT( A , MIN , MAX ) cin >> A; ASSERT( A , MIN , MAX ) #define RETURN( ANSWER ) COUT( ( ANSWER ) ); QUIT #define FINISH_MAIN #endif template <typename T> inline T Absolute( const T& a ){ return a > 0 ? a : -a; } template <typename T> inline T Residue( const T& a , const T& p ){ return a >= 0 ? a % p : p - 1 - ( ( - ( a + 1 ) ) % p ); } #define POWER( ANSWER , ARGUMENT , EXPONENT ) \ static_assert( ! is_same<TYPE_OF( ARGUMENT ),int>::value && ! is_same<TYPE_OF( ARGUMENT ),uint>::value ); \ TYPE_OF( ARGUMENT ) ANSWER{ 1 }; \ { \ TYPE_OF( ARGUMENT ) ARGUMENT_FOR_SQUARE_FOR_POWER = ( ARGUMENT ); \ TYPE_OF( EXPONENT ) EXPONENT_FOR_SQUARE_FOR_POWER = ( EXPONENT ); \ while( EXPONENT_FOR_SQUARE_FOR_POWER != 0 ){ \ if( EXPONENT_FOR_SQUARE_FOR_POWER % 2 == 1 ){ \ ANSWER *= ARGUMENT_FOR_SQUARE_FOR_POWER; \ } \ ARGUMENT_FOR_SQUARE_FOR_POWER *= ARGUMENT_FOR_SQUARE_FOR_POWER; \ EXPONENT_FOR_SQUARE_FOR_POWER /= 2; \ } \ } \ #define POWER_MOD( ANSWER , ARGUMENT , EXPONENT , MODULO ) \ ll ANSWER{ 1 }; \ { \ ll ARGUMENT_FOR_SQUARE_FOR_POWER = ( ( MODULO ) + ( ( ARGUMENT ) % ( MODULO ) ) ) % ( MODULO ); \ TYPE_OF( EXPONENT ) EXPONENT_FOR_SQUARE_FOR_POWER = ( EXPONENT ); \ while( EXPONENT_FOR_SQUARE_FOR_POWER != 0 ){ \ if( EXPONENT_FOR_SQUARE_FOR_POWER % 2 == 1 ){ \ ANSWER = ( ANSWER * ARGUMENT_FOR_SQUARE_FOR_POWER ) % ( MODULO ); \ } \ ARGUMENT_FOR_SQUARE_FOR_POWER = ( ARGUMENT_FOR_SQUARE_FOR_POWER * ARGUMENT_FOR_SQUARE_FOR_POWER ) % ( MODULO ); \ EXPONENT_FOR_SQUARE_FOR_POWER /= 2; \ } \ } \ #define FACTORIAL_MOD( ANSWER , ANSWER_INV , INVERSE , MAX_INDEX , CONSTEXPR_LENGTH , MODULO ) \ static ll ANSWER[CONSTEXPR_LENGTH]; \ static ll ANSWER_INV[CONSTEXPR_LENGTH]; \ static ll INVERSE[CONSTEXPR_LENGTH]; \ { \ ll VARIABLE_FOR_PRODUCT_FOR_FACTORIAL = 1; \ ANSWER[0] = VARIABLE_FOR_PRODUCT_FOR_FACTORIAL; \ FOREQ( i , 1 , MAX_INDEX ){ \ ANSWER[i] = ( VARIABLE_FOR_PRODUCT_FOR_FACTORIAL *= i ) %= ( MODULO ); \ } \ ANSWER_INV[0] = ANSWER_INV[1] = INVERSE[1] = VARIABLE_FOR_PRODUCT_FOR_FACTORIAL = 1; \ FOREQ( i , 2 , MAX_INDEX ){ \ ANSWER_INV[i] = ( VARIABLE_FOR_PRODUCT_FOR_FACTORIAL *= INVERSE[i] = ( MODULO ) - ( ( ( ( MODULO ) / i ) * INVERSE[ ( MODULO ) % i ] ) % ( MODULO ) ) ) %= ( MODULO ); \ } \ } \ // 二分探索テンプレート // EXPRESSIONがANSWERの広義単調関数の時、EXPRESSION >= TARGETの整数解を格納。 #define BS( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , DESIRED_INEQUALITY , TARGET , INEQUALITY_FOR_CHECK , UPDATE_U , UPDATE_L , UPDATE_ANSWER ) \ static_assert( ! is_same<TYPE_OF( TARGET ),uint>::value && ! is_same<TYPE_OF( TARGET ),ull>::value ); \ ll ANSWER = MINIMUM; \ if( MINIMUM <= MAXIMUM ){ \ ll VARIABLE_FOR_BINARY_SEARCH_L = MINIMUM; \ ll VARIABLE_FOR_BINARY_SEARCH_U = MAXIMUM; \ ANSWER = ( VARIABLE_FOR_BINARY_SEARCH_L + VARIABLE_FOR_BINARY_SEARCH_U ) / 2; \ ll VARIABLE_FOR_DIFFERENCE_FOR_BINARY_SEARCH; \ while( VARIABLE_FOR_BINARY_SEARCH_L != VARIABLE_FOR_BINARY_SEARCH_U ){ \ VARIABLE_FOR_DIFFERENCE_FOR_BINARY_SEARCH = ( EXPRESSION ) - ( TARGET ); \ CERR( "二分探索中: " << VARIABLE_FOR_BINARY_SEARCH_L << "<=" << ANSWER << "<=" << VARIABLE_FOR_BINARY_SEARCH_U << ":" << EXPRESSION << "-" << TARGET << "=" << VARIABLE_FOR_DIFFERENCE_FOR_BINARY_SEARCH ); \ if( VARIABLE_FOR_DIFFERENCE_FOR_BINARY_SEARCH INEQUALITY_FOR_CHECK 0 ){ \ VARIABLE_FOR_BINARY_SEARCH_U = UPDATE_U; \ } else { \ VARIABLE_FOR_BINARY_SEARCH_L = UPDATE_L; \ } \ ANSWER = UPDATE_ANSWER; \ } \ CERR( "二分探索終了: " << VARIABLE_FOR_BINARY_SEARCH_L << "<=" << ANSWER << "<=" << VARIABLE_FOR_BINARY_SEARCH_U << ":" << EXPRESSION << ( EXPRESSION > TARGET ? ">" : EXPRESSION < TARGET ? "<" : "=" ) << TARGET ); \ CERR( ( EXPRESSION DESIRED_INEQUALITY TARGET ? "二分探索成功" : "二分探索失敗" ) ); \ assert( EXPRESSION DESIRED_INEQUALITY TARGET ); \ } else { \ CERR( "二分探索失敗: " << MINIMUM << ">" << MAXIMUM ); \ assert( MINIMUM <= MAXIMUM ); \ } \ // 単調増加の時にEXPRESSION >= TARGETの最小解を格納。 #define BS1( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , TARGET ) \ BS( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , >= , TARGET , >= , ANSWER , ANSWER + 1 , ( VARIABLE_FOR_BINARY_SEARCH_L + VARIABLE_FOR_BINARY_SEARCH_U ) / 2 ) \ // 単調増加の時にEXPRESSION <= TARGETの最大解を格納。 #define BS2( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , TARGET ) \ BS( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , <= , TARGET , > , ANSWER - 1 , ANSWER , ( VARIABLE_FOR_BINARY_SEARCH_L + 1 + VARIABLE_FOR_BINARY_SEARCH_U ) / 2 ) \ // 単調減少の時にEXPRESSION >= TARGETの最大解を格納。 #define BS3( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , TARGET ) \ BS( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , >= , TARGET , < , ANSWER - 1 , ANSWER , ( VARIABLE_FOR_BINARY_SEARCH_L + 1 + VARIABLE_FOR_BINARY_SEARCH_U ) / 2 ) \ // 単調減少の時にEXPRESSION <= TARGETの最小解を格納。 #define BS4( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , TARGET ) \ BS( ANSWER , MINIMUM , MAXIMUM , EXPRESSION , <= , TARGET , <= , ANSWER , ANSWER + 1 , ( VARIABLE_FOR_BINARY_SEARCH_L + VARIABLE_FOR_BINARY_SEARCH_U ) / 2 ) \ // 圧縮用 #define TE template #define TY typename #define US using #define ST static #define IN inline #define CL class #define PU public #define OP operator #define CE constexpr #define CO const #define NE noexcept #define RE return #define WH while #define VO void #define VE vector #define LI list #define BE begin #define EN end #define SZ size #define MO move #define TH this #define CRI CO int& #define CRUI CO uint& #define CRL CO ll& #define ASK_NUMBER( ... ) \ CERR( "" ); \ CERR( "問題の区分は以下の中で何番に該当しますか?" ); \ problems = { __VA_ARGS__ }; \ problems_size = problems.size(); \ FOR( i , 0 , problems_size ){ \ CERR( i << ": " << problems[i] ); \ } \ cin >> num; \ CERR( "" ); \ if( num < 0 || num >= problems_size ){ \ CERR( "返答は" << problems_size - 1 << "以下の非負整数にしてください。" ); \ CERR( "終了します。" ); \ CERR( "" ); \ return; \ } \ num_temp = 0; \ #define ASK_YES_NO( QUESTION ) \ CERR( "" ); \ CERR( QUESTION << "[y/n]" ); \ cin >> reply; \ if( reply != "y" && reply != "n" ){ \ CERR( "y/nのいずれかで答えてください。" ); \ CERR( "終了します。" ); \ CERR( "" ); \ return; \ } \ CERR( "" ); \ #define SLS( CLASS ) void CLASS ## LibrarySearch( int& num , int& num_temp , string& reply , vector<string>& problems , int& problems_size ) #define CALL_SLS( CLASS ) CLASS ## LibrarySearch( num , num_temp , reply , problems , problems_size ) SLS( ExplicitExpression ); SLS( ExplicitExpressionUnary ); SLS( ExplicitExpressionMultiary ); SLS( ExplicitExpressionIteration ); SLS( ExplicitExpressionArraySum ); SLS( ExplicitExpressionFunctionOnPermutation ); SLS( ExplicitExpressionFunctionOnTree ); SLS( FunctionOnTree ); SLS( ExplicitExpressionFunctionOnNonTreeGraph ); SLS( ExplicitExpressionOrder ); SLS( ExplicitExpressionProbability ); SLS( ExplicitExpressionOther ); SLS( Maximisation ); SLS( MaximisationExplicitExpression ); SLS( MaximisationFunctionOnArray ); SLS( MaximisationSubArraySum ); SLS( MaximisationArrayFunction ); SLS( MaximisationArrayLength ); SLS( MaximisationFunctionOnTree ); SLS( MinimisationMovingCost ); SLS( MaximisationStringMatching ); SLS( MaximisationBipartiteMatching ); SLS( Counting ); SLS( CountingExplicitExpression ); SLS( CountingArray ); SLS( CountingSubArray ); SLS( CountingSumFixedSubArray ); SLS( CountingRestrctedSubArray ); SLS( CountingRestrctedContinuousSubArray ); SLS( CountingRestrctedDiscontinuousSubArray ); SLS( CountingRestrctedSubPermutation ); SLS( CountingArbitraryArray ); SLS( CountingSubArrayImage ); SLS( CountingPartitionOfTree ); SLS( CountingString ); SLS( Solving ); SLS( Query ); SLS( QueryArray ); SLS( QueryGraph ); SLS( Decision ); SLS( DecisionConnectedness ); SLS( DecisionHigherConnectedness ); SLS( DecisionGame ); SLS( DecisionAccessibility ); SLS( DecisionSatisfiability ); SLS( Construction ); void LibrarySearch( bool& searched_library ) { int num = 0; vector<string> problems{}; int problems_size = 13; int num_temp = 0; string reply{}; ASK_YES_NO( "ライブラリーを探索しますか?" ); if( reply != "y" ){ CERR( "ライブラリーを探索せずに続行します。" ); CERR( "" ); return; } searched_library = true; ASK_NUMBER( "明示式の計算問題" , "最大/最小化問題" , "数え上げ問題" , "求解問題" , "クエリ処理問題" , "真偽判定問題" , "構築問題" ); if( num == num_temp++ ){ CALL_SLS( ExplicitExpression ); } else if( num == num_temp++ ){ CALL_SLS( Maximisation ); } else if( num == num_temp++ ){ CALL_SLS( Counting ); } else if( num == num_temp++ ){ CALL_SLS( Solving ); } else if( num == num_temp++ ){ CALL_SLS( Query ); } else if( num == num_temp++ ){ CALL_SLS( Decision ); } else if( num == num_temp++ ){ CALL_SLS( Construction ); } ASK_YES_NO( "マルチテストケースですか?" ); if( reply == "y" ){ CERR( "テストケースを跨ぐ前計算が可能か否かを優先的に考察しましょう。" ); CERR( "" ); CERR( "テストケース全体でのNの総和に直接上限が与えられている問題では、" ); CERR( "ライブラリーの使用時は配列の初期化が各テストケースに必要となる場合に" ); CERR( "TLEとなる可能性が高いです。" ); CERR( "- 動的配列への置き換え" ); CERR( "- 座標圧縮" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\CoordinateCompress" ); CERR( "を検討しましょう。" ); CERR( "" ); CERR( "配列を手元の環境でデバッグしやすくするためにstaticをつけている場合は" ); CERR( "テストケースを跨いで値が残ってしまわないように注意しましょう。" ); CERR( "" ); } CERR( "ライブラリー探索は以上です。終了します。" ); CERR( "" ); } SLS( ExplicitExpression ) { ASK_YES_NO( "入力は1つの数か、1つの数と法を表す数ですか?" ); if( reply == "y" ){ CALL_SLS( ExplicitExpressionUnary ); } else { CALL_SLS( ExplicitExpressionMultiary ); } } SLS( ExplicitExpressionUnary ) { CERR( "まずは小さい入力の場合を愚直に計算し、OEISで検索しましょう。" ); CERR( "https://oeis.org/?language=japanese" ); CERR( "" ); CERR( "次に出力の定義と等価な式を考察しましょう。" ); CERR( "- 単調ならば、冪乗や階乗" ); CERR( "- 定義にp進法が使われていれば、各種探索アルゴリズム" ); CERR( "- 入力が素数に近い場合に規則性があれば、p進付値、p進法、" ); CERR( " オイラー関数、約数の個数など" ); CERR( "を検討しましょう。" ); CERR( "" ); CERR( "前計算の候補としては" ); CERR( "- 素数列挙" ); CERR( "- 1つまたは複数の整数の約数列挙" ); CERR( "- オイラー関数の値の列挙" ); CERR( "- サブゴールとなる関係式を満たす解の列挙" ); CERR( "を検討しましょう。" ); } SLS( ExplicitExpressionMultiary ) { ASK_YES_NO( "関数の反復合成の計算問題ですか?" ); if( reply == "y" ){ CALL_SLS( ExplicitExpressionIteration ); } else { ASK_NUMBER( "配列上の関数の総和の計算問題" , "順列上の関数の計算問題" , "木上の関数の総和の計算問題" , "木以外のグラフ上の関数の総和の計算問題" , "序数の計算問題" , "確率/期待値の計算問題" , "その他の明示式の計算問題" ); if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionArraySum ); } else if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionFunctionOnPermutation ); } else if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionFunctionOnTree ); } else if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionFunctionOnNonTreeGraph ); } else if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionOrder ); } else if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionProbability ); } else if( num == num_temp++ ){ CALL_SLS( ExplicitExpressionOther ); } } } SLS( ExplicitExpressionIteration ) { CERR( "定義域の要素数N、テストケース数T、反復回数の上限Kとします。" ); CERR( "- O((N + T)log_2 K)が通りそうならばダブリング" ); CERR( " \\Mathematics\\Function\\Iteration\\Doubling" ); CERR( "- O(TN)が通りそうならばループ検出" ); CERR( " \\Mathematics\\Function\\Iteration\\LoopDetection" ); CERR( "- O(N)すら通らなさそうならば関数の規則性を見付けるための実験" ); CERR( "を検討しましょう。" ); } SLS( ExplicitExpressionArraySum ) { ASK_NUMBER( "成分を受け取る関数の総和の計算問題" , "部分列を受け取る関数の総和の計算問題" ); if( num == num_temp++ ){ CERR( "成分を受け取る関数fが与えられているとします。" ); CERR( "fが一次式の場合、実質内積と定数の和となります。" ); CERR( "内積は片方の添え字を反転させることで畳み込みに帰着させることができます。" ); CERR( "配列への操作がシフトである場合は繰り返し内積を求めることになるので、" ); CERR( "適当な法での高速フーリエ変換" ); CERR( "\\Mathematics\\Arithmetic\\Mod" ); CERR( "\\Mathematics\\Polynoial" ); CERR( "を検討しましょう。" ); } else if( num == num_temp++ ){ ASK_NUMBER( "連続部分列への分割に関する関数の総和の計算問題" , "連続とは限らない部分列への分割に関する関数の総和の計算問題" ); if( num == num_temp++ ){ CERR( "配列の添字集合は全順序集合なので、木の分割の問題に一般化されます。" ); CALL_SLS( ExplicitExpressionFunctionOnTree ); CERR( "" ); CERR( "更にfが部分列の長さに関する再帰的な構造を持つ場合、全ての連続部分列に" ); CERR( "対しfの値を前計算することを検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "配列の並び換えによって答えが変わらないので、適切にソートしてから" ); CERR( "計算することを検討しましょう。" ); } } CERR( "" ); CERR( "入力が大きい場合と小さい場合で解法を変える考察を忘れないようにしましょう。" ); } SLS( ExplicitExpressionFunctionOnPermutation ) { CERR( "- 符号そのものの計算問題は" ); CERR( " - O(N log_2 N)やO(N^2)が間に合いそうなら転倒数の計算" ); CERR( " - O(N log_2 N)が間に合わなさそうなら互換表示(O(N))" ); CERR( "- 符号と何かの積の和は行列式に帰着させ、" ); CERR( " - 行列式そのものなら行基本変形(O(N^3))" ); CERR( " - 余因子展開の途中の値が必要ならメモ化再帰(O(N 2^N))" ); CERR( "を検討しましょう。" ); CERR( "" ); CERR( "1つの順列の転倒数は、" ); CERR( "- O(N^2)が通りそうならば愚直な二重ループ" ); CERR( "- O(N log_2 N)が通りそうならば可換群BIT" ); CERR( " \\Mathematics\\Combinatorial\\Permutation" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\BIT" ); CERR( "で計算しましょう。" ); CERR( "" ); CERR( "条件を満たす順列全体をわたる転倒数の総和/期待値は、" ); CERR( "各i<jごとにそこで転倒が生じる順列の個数を計算し、その総和/期待値を" ); CERR( "取りましょう。条件が良ければ、転倒が生じる順列の個数は転倒が生じるとは" ); CERR( "限らない順列の個数の半分となります。" ); } SLS( ExplicitExpressionFunctionOnTree ) { CALL_SLS( FunctionOnTree ); CERR( "" ); CERR( "部分木に関する良い遷移関係を探し、(全方位)木DP" ); CERR( "\\Mathematics\\Geometry\\Graph\\DepthFirstSearch\\Tree" ); CERR( "を検討しましょう。" ); ASK_YES_NO( "fがbit演算である問題ですか?" ); if( reply == "y" ){ CERR( "「Tの各ノードvを根とする部分木でのj桁目のbit状態sの個数dp[v][s][j]」" ); CERR( "を管理するv,s,jに関する動的計画法を検討しましょう。" ); CERR( "これはTが全順序集合でbit演算が1種類なら" ); CERR( "「第i成分で切った部分列でのj桁目のbitがs(=0,1)である個数dp[i][s][j]」" ); CERR( "を管理することに他なりません。" ); CERR( "" ); CERR( "適宜、bitsetや累積和などのデータ構造で高速化しましょう。" ); } } SLS( FunctionOnTree ) { CERR( "木を受け取る関数fが与えられているとします。" ); CERR( "" ); CERR( "木Tの分割Pに対し、Pの各成分pを渡るf(p)の総和をF(P)と置きます。" ); CERR( "Tに根を固定し、深さ優先探索でTの頂点にラベルづけをします。" ); CERR( "" ); CERR( "Pの各成分pに対しpの各頂点のラベルの最小値をD(p)と置き、" ); CERR( "Dについて昇順にPを並べます。" ); CERR( "" ); CERR( "Pの末尾成分pを削除した分割P'が元の木からpを削除した木の分割であり" ); CERR( "F(P)=F(P')+f(p)と表せることに注意しましょう。" ); } SLS( ExplicitExpressionFunctionOnNonTreeGraph ) { CERR( "ゼータ変換/メビウス変換" ); CERR( "\\Mathematics\\Combinatorial\\ZetaTransform" ); CERR( "を検討しましょう。" ); } SLS( ExplicitExpressionOrder ) { CERR( "集合Sを何らかの順序でソートした配列aに関する問題で、" ); CERR( "- 与えられた要素sが下から何番目かを答える場合は、非負整数iごとに" ); CERR( " 不等式a[i]<=sを判定する方法を考察し、iに関する二分探索" ); CERR( "- 与えられた非負整数iに対するa[i]を答える場合は、Sの要素sごとに" ); CERR( " 不等式a[i]<=sを判定する方法を考察し、sに関する二分探索" ); CERR( "を検討しましょう。" ); } SLS( ExplicitExpressionProbability ) { CERR( "- 確率計算は" ); CERR( " - 余事象や包除原理(高速ゼータ変換/メビウス変換)" );; CERR( " \\Mathematics\\Combinatorial\\ZetaTransform" ); CERR( " - 同様に確からしい事象の特定" ); CERR( " - ベイズの定理" ); CERR( "- 期待値計算は" ); CERR( " - 上記方法での確率計算" ); CERR( " - 対象を独立な和で表して線形性" ); CERR( "を検討しましょう。" ); } SLS( ExplicitExpressionOther ) { CERR( "- 出力の定義と等価な式への変形" ); CERR( " - 和の順序交換" ); CERR( " - 同じ値になる項の纏め上げ" ); CERR( " - 二項展開や積の和典型などの組み合わせ論的解釈" ); CERR( " https://ei1333.hateblo.jp/entry/2021/07/30/144201" ); CERR( "- 和の動く範囲の差分に注目した動的計画法" ); CERR( "を検討しましょう。" ); } SLS( Maximisation ) { ASK_NUMBER( "明示式の最大/最小化問題" , "配列上の関数の最大/最小化問題" , "配列の隣接成分間関係式を満たす部分列の最長化問題" , "木上の関数の最大/最小化問題" , "移動コスト最小化問題" , "文字列のマッチングに関する最大/最長化問題" , "最大二部マッチング問題" ); if( num == num_temp++ ){ CALL_SLS( MaximisationExplicitExpression); } else if( num == num_temp++ ){ CALL_SLS( MaximisationFunctionOnArray ); } else if( num == num_temp++ ){ CALL_SLS( MaximisationArrayLength ); } else if( num == num_temp++ ){ CALL_SLS( MaximisationFunctionOnTree ); } else if( num == num_temp++ ){ CALL_SLS( MinimisationMovingCost ); } else if( num == num_temp++ ){ CALL_SLS( MaximisationStringMatching ); } else if( num == num_temp++ ){ CALL_SLS( MaximisationBipartiteMatching ); } } SLS( MaximisationExplicitExpression ) { ASK_NUMBER( "凸関数の最小/最大化問題" , "可微分関数の最小/最大化問題" , "絶対値の最小/最大化問題" ); if( num == num_temp++ ){ CERR( "三分探索を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "ニュートン法を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "符号を用いて絶対値を外しましょう。" ); CERR( "- 単調な式に帰着できる場合、二分探索" ); CERR( "- 最大化問題の場合、符号パターンの全探策" ); CERR( "- マンハッタン距離などは一次変換" ); CERR( "を検討しましょう。" ); } CERR( "" ); CERR( "複数のパラメータを決定すべき場合は、サブゴールの式の値を決め打ちましょう。" ); } SLS( MaximisationFunctionOnArray ) { ASK_NUMBER( "問題文または入力で与えられる1つの配列に関する問題" , "条件を満たす任意の配列に関する問題" ); if( num == num_temp++ ){ ASK_NUMBER( "成分を受け取る関数の部分和の最大化問題" , "配列の変更と配列を受け取る関数の合成の最大化問題" ); if( num == num_temp++ ){ CALL_SLS( MaximisationSubArraySum ); } else if( num == num_temp++ ){ CALL_SLS( MaximisationArrayFunction ); } } else if( num == num_temp++ ){ CERR( "- 取り得る値が少なく関数が長さに関して再帰的構造を持つ場合は、" ); CERR( " 「長さiの時に可能な値全体または一部の集合dp[i]」" ); CERR( " を管理するiに関する動的計画法" ); CERR( "- 「v以上の値を取り得るか否か」が判定可能である時は" ); CERR( " vに関する二分探索" ); CERR( "を検討しましょう。" ); } } SLS( MaximisationSubArraySum ) { CERR( "項数N、重みの総和の上限W(多変数も可)、価値(和を取る値)の上限Vとします。" ); CERR( "- B=∞ならば、通常のナップサック問題と同様の動的計画法" ); CERR( "- B<∞でO(2^N)が通りそうならば愚直に全探策" ); CERR( "- B<∞でO(N 2^{N/2})が通りそうならば半分全列挙" ); CERR( "- B<∞で重みと価値が等しくO(NV)が通りそうならば[B-V,B+V]での実現可能性を" ); CERR( " 遷移する動的計画法" ); CERR( " https://stackoverflow.com/a/18949218" ); CERR( "- Wが10^5オーダーで重みと価値が等しくO((N+W)log_2 W)が通りそうならば" ); CERR( " 適当な法での畳み込み(確率的解法)" ); CERR( " \\Mathematics\\Polynomial" ); CERR( "- 選択に順序と制約がある場合、O(N 2^N)が通りそうならばbit全探策" ); CERR( "を検討しましょう。" ); } SLS( MaximisationArrayFunction ) { CERR( "配列を受け取る関数Fが与えられているとします。与えられた配列Aに" ); CERR( "何らかの処理をして得られる配列Bに対するF(B)の最大化問題は、" ); CERR( "最大化すべき式のサブゴールfに表れる項xのうち決め打ちやすいものを探しましょう。" ); CERR( "- 配列の長さをiで打ち切った時のxの候補数をX(i)" ); CERR( "- 配列の長さをiで打ち切ってxを決め打った時の配列の長さi+1でのxの候補数をdX(i)" ); CERR( "と置きます。" ); CERR( "- O(sum_i X(i) dX(i))が通りそうでfがxからO(1)で計算できるならば、" ); CERR( " iとxに関する動的計画法" ); CERR( "- O(N log_2 X(N))が通りそうで" ); CERR( " - fがxからO(N)で計算できxに関して単調ならば、xの二分探索" ); CERR( " - fがxからO(N/x)で計算できるならば、xの全探索" ); CERR( "- O(N log_2 N)が通りそうでxを並び替えるとfがxからO(log_2 N)で計算できるならば、" ); CERR( " 優先度つきキューなどでのxの管理" ); CERR( "を検討しましょう。" ); } SLS( MaximisationArrayLength ) { CERR( "全順序か疎な半順序かで効率的な実装が違います。" ); CERR( "- 全順序ならば、条件を満たす部分列の長さの最大値をインデックスに持つ" ); CERR( " 配列を用いて、それらの部分列の末尾である項を記録すること" ); CERR( " \\Mathematics\\Combinatorial\\Counting\\IncreasingSubsequence" ); CERR( "- 疎な半順序ならば、条件を満たす部分列の末尾をインデックスに持つ" ); CERR( " 連想配列を用いて、それら部分列の長さの最大値を記録すること" ); CERR( " \\Mathematics\\Combinatorial\\Counting\\IncreasingSubsequence\\Subwalk" ); CERR( "を検討しましょう。" ); } SLS( MaximisationFunctionOnTree ) { CALL_SLS( FunctionOnTree ); CERR( "「第i頂点までで切った時のF(P)たちの最大値dp[i]」" ); CERR( "を管理するiに関する動的計画法(O(N^2×fの計算量))" ); CERR( "を検討しましょう。" ); } SLS( MinimisationMovingCost ) { ASK_NUMBER( "2点最小コスト移動(迷路)問題" , "多点最小コスト移動(スタンプラリー)問題" ); if( num == num_temp++ ){ CERR( "特定の経路を進むと思い込んで考察漏れをする可能性があります。" ); CERR( "なるべく全ての経路を許した探索アルゴリズムを適用した方が無難です。" ); CERR( "- 特定の2点のみを考える場合、BFSやDijkstra" ); CERR( " \\Mathematics\\Geometry\\Graph\\BreadthFirst" ); CERR( " \\Mathematics\\Geometry\\Graph\\Dijkstra" ); CERR( "- 全ての2点の組み合わせを考える場合、" ); CERR( " - 一般のモノイド演算を考えておりO(V^3)が通りそうならば、FloydWarshall" ); CERR( " \\Mathematics\\Geometry\\Graph\\FloydWarshall" ); CERR( " - max演算を考えておりO(E(log_2 E + α(V)))が通りそうならば、UnionFind" ); CERR( " \\Utility\\VLTree\\UnionFindForest" ); CERR( "を検討しましょう。" ); CERR( "" ); CERR( "点の座標と最小化すべきコスト以外の数値変化がある場合、最小コスト移動における" ); CERR( "その数値の動く範囲を絞って点の座標との組を頂点とするグラフを検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "HeldKarpや、移動方法を分類するパラメータの全探策などを検討しましょう。" ); } } SLS( MaximisationStringMatching ) { CERR( "基本的には丁寧にループを回して解きましょう。" ); CERR( "- 比較対象が少ない場合、前または後ろから順に探索(貪欲法/動的計画法)" ); CERR( "- ワイルドカードを含む場合、" ); CERR( " - 前または後ろから順に場合分けをしてO(N)で処理できるか" ); CERR( " - 可能な代入方法を絞り込んでO(N)種類に落せるか" ); CERR( "- 比較回数が多い場合、ローリングハッシュ" ); CERR( " \\Utility\\String\\RollingHash" ); CERR( "- マッチングする文字列の最長化をする場合、Zアルゴリズム" ); CERR( " https://qiita.com/Pro_ktmr/items/16904c9570aa0953bf05" ); CERR( "- マッチングする文字数の最大化をする場合、文字の種類分の{0,1}値配列に" ); CERR( " 分けて内積の最大化(添え字を反転させて適当な法での畳み込み)" ); CERR( " \\Mathematics\\Arithmetic\\Mod" ); CERR( " \\Mathematics\\Polynomial" ); CERR( "を検討しましょう。" ); } SLS( MaximisationBipartiteMatching ) { CERR( "HopcroftKarpや最大流" ); CERR( "\\Mathematics\\Geometry\\Graph\\HopcroftKarp" ); CERR( "を検討しましょう。" ); } SLS( Counting ) { ASK_NUMBER( "固定長変数関数で与えられる明示式の数え上げ問題" , "配列に関する数え上げ問題" , "木の分割の数え上げ問題" , "文字列の数え上げ問題" ); if( num == num_temp++ ){ CALL_SLS( CountingExplicitExpression ); } else if( num == num_temp++ ){ CALL_SLS( CountingArray ); } else if( num == num_temp++ ){ CALL_SLS( CountingPartitionOfTree ); } else if( num == num_temp++ ){ CALL_SLS( CountingString ); } } SLS( CountingExplicitExpression ) { CERR( "- 変数の対称性があれば大小関係を制限した全探策" ); CERR( "- 何らかの約数となるなど動く範囲が狭い変数があればそれらを決め打った全探策" ); CERR( "- 多変数の合成関数で表せる場合は半分全列挙" ); CERR( "を検討しましょう。" ); } SLS( CountingArray ) { ASK_NUMBER( "問題文または入力で与えられる1つの配列に関する問題" , "条件を満たす任意の配列に関する問題" ); if( num == num_temp++ ){ CALL_SLS( CountingSubArray ); } else if( num == num_temp++ ){ CALL_SLS( CountingArbitraryArray ); } } SLS( CountingSubArray ) { ASK_NUMBER( "配列の成分を受け取る関数の部分和を固定した部分列の数え上げ問題" , "配列の隣接成分間関係式を満たす部分列の数え上げ問題" "配列の部分列から取得位置情報を落とした配列の数え上げ問題" ); if( num == num_temp++ ){ CALL_SLS( CountingSumFixedSubArray ); } else if( num == num_temp++ ){ CALL_SLS( CountingRestrctedSubArray ); } else if( num == num_temp++ ){ CALL_SLS( CountingSubArrayImage ); } } SLS( CountingSumFixedSubArray ) { CERR( "項数N、重みの総和の上限Wとします。" ); CERR( "- 重みと価値が異なりO(2^N)が通りそうならば愚直な2変数多項式乗算" ); CERR( "- 重みと価値が等しくO(2^N)が通りそうならば愚直な1変数多項式乗算" ); CERR( "- 重みと価値が等しくO(2^{N/2}N)が通りそうならば半分ずつ多項式乗算を" ); CERR( " して最後にそれらの積の1係数のみの計算" ); CERR( "- 重みと価値が等しくWが10^5オーダーで法が与えられていてO((N+W)log_2 W)が" ); CERR( " 通りそうならば適当な法での高速フーリエ変換" ); CERR( " \\Mathematics\\Arithmetic\\Mod" ); CERR( " \\Mathematics\\Polynomial" ); CERR( "を検討しましょう。" ); } SLS( CountingRestrctedSubArray ) { ASK_NUMBER( "連続部分列の数え上げ問題" , "連続とは限らない部分列の数え上げ問題" , "部分順列(部分列の並び換え)の数え上げ問題" ); CERR( "長さNの配列Aと、L+n<=N+1を満たす整数n,Lと" ); CERR( "- 整数iに関する条件Q(i)" ); CERR( "- 各0<=l<Lに対するn-tuple(b_0,...,b_{n-1})に関する条件R_l(b_0,...,b_{n-1})" ); CERR( "が与えられているとし、配列Bの条件P(B)を" ); CERR( "「Bの長さ|B|がQ(|B|)かつ|B|<L+nを満たし、かつ任意の0<=l<=|B|-nに対し" ); CERR( " R_l(B_l,...,B_{l+n-1})が成り立つ。」" ); CERR( "の論理積として定めます。" ); CERR( "" ); CERR( "例えば山型の部分列の数え上げ問題ではN>=3かつ(n,L)=(2,2)で" ); CERR( "- Q(i)=「i=3」" ); CERR( "- R_0(b_0,b_1)=「b_0<b_1」" ); CERR( "- R_1(b_0,b_1)=「b_0>b_1」" ); CERR( "と表されます。" ); CERR( "" ); if( num == num_temp++ ){ CALL_SLS( CountingRestrctedContinuousSubArray ); } else if( num == num_temp++ ){ CALL_SLS( CountingRestrctedDiscontinuousSubArray ); } else if( num == num_temp++ ){ CALL_SLS( CountingRestrctedSubPermutation ); } CERR( "を検討しましょう。" ); CERR( "" ); CERR( "特にR_l(B)たちがgcdやmaxなどの羃等演算に関する等式で与えられる場合は、" ); CERR( "不等式の方が扱いやすいのでゼータ変換/メビウス変換" ); CERR( "\\Mathematics\\Combinatorial\\ZetaTransform" ); CERR( "を検討しましょう。" ); } SLS( CountingRestrctedContinuousSubArray ) { CERR( "P(B)を満たすAの連続部分列Bの数え上げは、" ); CERR( "- R_lたちがlに依存しないならば尺取り法O(N)" ); CERR( "- R_lたちがlに依存する場合、" ); CERR( " - O(N^2)が通りそうなら左端を固定した愚直探索" ); CERR( " - O(N^2)が通らなさそうならR_lたちの読み替え" ); CERR( "を検討しましょう。" ); } SLS( CountingRestrctedDiscontinuousSubArray ) { CERR( "P(B)を満たすAの連続とは限らない部分列Bの数え上げは、" ); CERR( "- n-1<=i<=max{j<=N|Q(j)}を満たす各i" ); CERR( "- (0,1,...,N-1)の長さn-1の各部分列s" ); CERR( "に対する" ); CERR( "「長さiで、任意の0<=l<=i-nに対しR_l(B)を満たし、" ); CERR( " 末尾n-1項がsに対応するAの部分列Bの個数dp[i][s]」" ); CERR( "を管理するi,sに関する動的計画法" ); CERR( "\\Mathematics\\Combinatorial\\Counting\\IncreasingSubsequence" ); CERR( "\\Mathematics\\Combinatorial\\Counting\\IncreasingSubsequence\\Subwalk" ); CERR( "を検討しましょう。" ); } SLS( CountingRestrctedSubPermutation ) { CERR( "P(B)を満たすAの部分順列Bの数え上げは、" ); CERR( "- n-1<=|S|<=max{j<=N|Q(j)}を満たす(0,1,...,N-1)の部分集合S" ); CERR( "- Sの長さn-1の各部分順列s" ); CERR( "に対する" ); CERR( "「任意の0<=l<=|S|-nに対しR_l(B)を満たし、末尾n-1項がsに対応し、」" ); CERR( " 全体がSに対応するAの部分順列Bの個数dp[S][s]」" ); CERR( "を管理するS,sに関する動的計画法を検討しましょう。" ); CERR( "" ); CERR( "sの網羅は[0,N)^{n-1}の全探策でもS内の順列探索と定数倍しか変わらないので" ); CERR( "実装の速さを優先しましょう。" ); CERR( "" ); CERR( "Nが小さい場合は概算値" ); CERR( "7! ≒ 5×10^3" ); CERR( "8! ≒ 4×10^4" ); CERR( "9! ≒ 4×10^5" ); CERR( "10! ≒ 4×10^6" ); CERR( "11! ≒ 4×10^7" ); CERR( "12! ≒ 5×10^8" ); CERR( "を参考に順列の全列挙" ); CERR( "\\Mathematics\\Combinatorial\\Permutation" ); CERR( "を検討しましょう。" ); } SLS( CountingArbitraryArray ) { ASK_NUMBER( "配列を受け取る関数の値の数え上げ問題" , "隣接成分間関係式を満たす配列の数え上げ問題" ); if( num == num_temp++ ){ CERR( "- 配列の種類が少ない場合は、全ての配列に対する関数の値の前計算" ); CERR( "- 取り得る値が少なく関数が長さに関して再帰的構造を持つ場合は、" ); CERR( " 「長さiの時に値vである配列の総数dp[i][v]」" ); CERR( " を管理するi,vに関する動的計画法" ); } else if( num == num_temp++ ){ CERR( "- いくつかの条件の重ね合わせの時は包除原理" ); CERR( "- 全順序の場合は数の分割方法などへの翻訳" ); CERR( "- 疎な半順序の場合はグラフの前計算" ); } CERR( "を検討しましょう。" ); } SLS( CountingSubArrayImage ) { CERR( "入力で与えられる配列をAと置きます。" ); CERR( "配列として等しいAの部分列のうち辞書式順序最小のものを数え上げる" ); CERR( "部分列DPを検討しましょう。具体的には" ); CERR( "「末尾が第i成分由来で辞書順最小なAの部分列の個数dp[i]」" ); CERR( "「s=A[j]を満たすj<iの最大値last[i][s]」" ); CERR( "の2つを管理するiに関する動的計画法を検討しましょう。" ); CERR( "" ); CERR( "適宜、累積和などのデータ構造で高速化しましょう。" ); } SLS( CountingPartitionOfTree ) { CALL_SLS( FunctionOnTree ); CERR( "F(P)が固定された時のPの数え上げ問題は" ); CERR( "「第i成分までで切った時のF(P)=vを満たすPの個数dp[i][v]」" ); CERR( "を管理するi,vに関する動的計画法(O(N^2 v_max×fの計算量))" ); CERR( "を検討しましょう。" ); } SLS( CountingString ) { ASK_NUMBER( "部分文字列から取得位置情報を落とした文字列の数え上げ問題" , "回文である部分文字列の数え上げ問題" ); if( num == num_temp++ ){ CERR( "文字列を文字の配列とみなすことで、配列の問題に帰着させることができます。" ); CALL_SLS( CountingSubArrayImage ); } else if( num == num_temp++ ){ CERR( "回文判定は長さに関して再帰的に計算できます。" ); CERR( "- O(N^2)が通る場合、愚直な再帰により前計算で全ての部分列の回文判定" ); CERR( "- O(N^2)が通らない場合、Manacherのアルゴリズムやローリングハッシュで前計算" ); CERR( " https://snuke.hatenablog.com/entry/2014/12/02/235837" ); CERR( "を検討しましょう。" ); } } SLS( Solving ) { CERR( "- 単調関数は二分探索" ); CERR( "- 可微分関数はニュートン法" ); CERR( "- 一次関数は掃き出し法" ); CERR( "を検討しましょう。" ); } SLS( Query ) { ASK_NUMBER( "配列の問題" , "グラフの問題" ); if( num == num_temp++ ){ CALL_SLS( QueryArray ); } else if( num == num_temp++ ){ CALL_SLS( QueryGraph ); } } SLS( QueryArray ) { ASK_NUMBER( "可換群構造+を使う問題" , "可換羃等モノイド構造∨を使う問題" , "モノイド構造*を使う問題" , "非結合的マグマ構造*を使う問題" , "集合へのマグマ作用(*,\\cdot)を使う問題" , "モノイドへのマグマ作用(+,\\cdot)を使う問題" , "定数とのmaxを取った値の区間和取得を使う問題" ); if( num == num_temp++ ){ CERR( "- 区間加算/区間取得が必要ならば可換群BIT" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\BIT\\Template" ); CERR( "- 一点代入/一点加算/区間取得が必要ならば可換群平方分割" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\SqrtDecomposition\\Template" ); CERR( "- 区間以外の領域で加算/全更新後の一点取得が必要ならば階差数列" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\Tree\\DifferenceSeqeuence" ); CERR( "を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "- 一点代入/区間加算/一点取得/区間取得が必要ならば可換羃等モノイドBIT" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\BIT\\IntervalMax\\Template" ); CERR( "を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "- 一点代入/区間取得が必要ならばモノイドBIT" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\BIT\\Template\\Monoid" ); CERR( "- 一点加算/区間加算/一点取得/区間取得が必要ならばモノイド平方分割" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\SqrtDecomposition\\Template\\Monoid" ); CERR( "- 一点代入/一点取得/区間取得が必要ならばモノイドセグメント木" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\SegmentTree" ); CERR( "を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "- 写像のコード化" ); CERR( " \\Mathematics\\Function\\Encoder" ); CERR( "によりモノイドに帰着させることを検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "- 一点作用/区間作用/一点取得が必要ならば双対平方分割" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\SqrtDecomposition\\Template\\Dual" ); CERR( "を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "- 区間代入/区間作用/区間加算/一点取得/区間取得が必要な場合は遅延評価平方分割" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\AffineSpace\\SqrtDecomposition\\Template\\LazyEvaluation" ); CERR( "を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "maxで全体更新でなく区間更新をする場合の汎用的な解法は分かりません。" ); CERR( "例えば区間が包含について単調でmaxを取る値も単調であれば全体更新と" ); CERR( "同様の処理ができます。状況に応じた考察をしましょう。" ); CERR( "" ); CERR( "maxで全体更新をする場合、maxを取る値は単調である場合に帰着できます。" ); CERR( "maxで全体更新をしない場合、つまりただmaxの区間和を処理するだけの場合、" ); CERR( "クエリの順番を入れ替えることができるので、単調な全体更新に帰着できます。" ); CERR( "従って以下では単調な全体更新の問題を考えます。" ); CERR( "" ); CERR( "maxを取る定数を変数化し、元の値との大小を表す{0,1}値の係数を考えます。" ); CERR( "すると区間作用前後の値は統一的にその係数と変数を使って表せます。" ); CERR( "配列の各成分の係数の値が変化するイベントとクエリをソートして管理し、" ); CERR( "クエリがイベントを跨ぐたびに係数を更新することを検討しましょう。" ); CERR( "" ); CERR( "例えばクエリB_qに対するmax(A_i,B_q)の区間和は、" ); CERR( "- 優先度つきキューA'={(A_i,i)|i}(構築O(N log N))" ); CERR( "- (B_q,q)_qをソートしたB'(構築O(Q log Q))" ); CERR( "- 長さNの数列C=(0,...,0)(構築O(N))" ); CERR( "を用意し、B'を前から探索して順に各クエリ(B_q,q)を処理します。" ); CERR( "具体的にはA'を前から探索して順にA_i<B_qとなる各iに対し" ); CERR( "- A'から(A_i,i)を削除(クエリ合計O(N))" ); CERR( "- A_iを0に更新(クエリ合計O(N log N))" ); CERR( "- C_iを1に更新(各クエリO(log N))" ); CERR( "- A+C×B_qの区間和取得(各クエリO(log N))" ); CERR( "とすれば合計O((N + Q)log N + Q log Q)で処理できます。" ); } } SLS( QueryGraph ) { ASK_YES_NO( "代数構造を扱う問題ですか?" ); if( reply == "yes" ){ CERR( "- 可換群構造に関する加算/全更新後の一点取得が必要ならば階差数列" ); CERR( " \\Mathematics\\SetTheory\\DirectProduct\\Tree\\DifferenceSeqeuence" ); } else { CERR( "深さ優先探索や動的木やUnionFind" ); CERR( "\\Mathematics\\Geometry\\Graph\\DepthFirst\\Tree" ); CERR( "\\Utility\\VLTree" ); CERR( "\\Utility\\VLTree\\UnionFndForest" ); } CERR( "を検討しましょう。" ); } SLS( Decision ) { ASK_NUMBER( "0次連結性判定問題" , "高次連結性判定問題" , "勝敗判定問題" , "到達可能性問題" , "充足可能性問題" ); if( num == num_temp++ ){ CALL_SLS( DecisionConnectedness ); } else if( num == num_temp++ ){ CALL_SLS( DecisionHigherConnectedness ); } else if( num == num_temp++ ){ CALL_SLS( DecisionGame ); } else if( num == num_temp++ ){ CALL_SLS( DecisionAccessibility ); } else if( num == num_temp++ ){ CALL_SLS( DecisionSatisfiability ); } } SLS( DecisionConnectedness ) { CERR( "幅優先探索やUnionFind" ); CERR( "\\Mathematics\\Geometry\\Graph\\BreadthFirst" ); CERR( "\\Utility\\VLTree\\UnionFindForest" ); CERR( "を検討しましょう。" ); CERR( "" ); } SLS( DecisionHigherConnectedness ) { CERR( "- 強連結性判定は深さ優先探索" ); CERR( " \\Mathematics\\Geometry\\Graph\\DepthFirst" ); CERR( "- 高次ホモロジー計算は適当な法での掃き出し法" ); CERR( "を検討しましょう。" ); } SLS( DecisionGame ) { CERR( "ゲームの和に分解できる場合は最小単位で考察をし、グランディ数を実装しましょう。" ); CERR( "これ以上分解できないゲームには整礎構造を探し、順序数の小さい順に実験をしましょう。" ); } SLS( DecisionAccessibility ) { ASK_YES_NO( "矩形領域の到達可能性問題ですか?" ); if( reply == "yes" ){ CERR( "迷路の攻略可能性は" ); CERR( "- スタートとゴールが同一の弧状連結成分に属すこと" ); CERR( "- スタートとゴールを分断する壁のパスの非存在性" ); CERR( "などに翻訳しグラフ上の最小コスト移動問題に帰着させましょう。" ); } else { CERR( "適切なグラフ上の最小コスト移動問題に帰着させましょう。" ); CALL_SLS( MinimisationMovingCost ); } CERR( "" ); CALL_SLS( MinimisationMovingCost ); } SLS( DecisionSatisfiability ) { ASK_NUMBER( "命題論理の充足性判定問題" , "二部グラフの充足可能性問題" ); if( num == num_temp++ ){ CERR( "各命題変数の真理値を頂点とするグラフを考え連結性判定に帰着させて" ); CALL_SLS( DecisionConnectedness ); CERR( "" ); CERR( "区間処理による真理値更新は各種代数的データ構造を用いましょう。" ); SLS( QueryArray ); } else if( num == num_temp++ ){ CERR( "完全二部マッチング判定は最大二部マッチングに帰着させて" ); SLS( MaximisationBipartiteMatching ); } } SLS( Construction ) { ASK_NUMBER( "数や配列や文字列の構築" , "経路の構築" , "戦略の構築" , "ソースコードの構築" ); if( num == num_temp++ ){ CERR( "p進法を検討しましょう。" ); } else if( num == num_temp++ ){ CERR( "可能な経路の定めるグラフの問題に帰着させましょう。" ); CALL_SLS( DecisionAccessibility ); } else if( num == num_temp++ ){ CERR( "ゲームの問題に帰着させましょう。" ); CALL_SLS( DecisionGame ); } else if( num == num_temp++ ){ CERR( "正解を出力をするソースコードを提出しましょう。" ); } } // N_max > 0の場合のみサポート。 template <int N_max> class MahlerTransform { private: const int m_p; ll m_fact[N_max]; ll m_fact_inv[N_max]; ll m_inv[N_max]; public: // pがN_max以上の素数である場合のみサポート。 // O(N_max)で前計算値を格納する。 inline MahlerTransform( const int& p ); // 0 <= i_start <= i_final < N_maxの場合のみサポート。 // O((i_final-i_start)^2)で(f[i_start],...,f[i_final])のマーラー変換を法pで計算し // (a[i_start],...,a[i_final])に格納する。 // 要素数のみに依存する包除原理を法pで解くことに対応。 void Convert( const ll ( &f )[N_max] , ll ( &a )[N_max] , const int& i_start , const int& i_final ); // O((i_final-i_start)^2)で(a[i_start],...,a[i_final])の逆マーラー変換を法pで計算し // (f[i_start],...,f[i_final])に格納する。 // 要素数0,1,...,nのデータ群から要素数n以下のデータを得ることに対応。 void InverseConvert( const ll ( &a )[N_max] , ll ( &f )[N_max] , const int& i_start , const int& i_final ); // 0 <= n < N_maxである場合のみサポート。 // 前計算された値をO(1)で参照する。 inline ll Combination( const int& n , const int& i ) const; inline const ll& Factorial( const int& n ) const; inline const ll& FactorialInverse( const int& n ) const; // 0 < n < N_maxである場合のみサポート。 inline const ll& Inverse( const int& n ) const; }; template <int N_max> inline MahlerTransform<N_max>::MahlerTransform( const int& p ) : m_p( p ) , m_fact() , m_fact_inv() , m_inv() { static_assert( 0 < N_max ); assert( N_max <= m_p ); m_fact[0] = m_fact_inv[0] = m_fact[1] = m_fact_inv[1] = m_inv[1] = 1; ll fact_temp = 1; ll fact_inv_temp = 1; for( int i = 2 ; i < N_max ; i++ ){ m_fact[i] = ( fact_temp *= i ) %= m_p; m_fact_inv[i] = ( fact_inv_temp *= m_inv[i] = m_p - m_inv[m_p % i] * ( m_p / i ) % m_p ) %= m_p; } } template <int N_max> void MahlerTransform<N_max>::Convert( const ll ( &f )[N_max] , ll ( &a )[N_max] , const int& i_start , const int& i_final ) { assert( 0 <= i_start && i_start <= i_final && i_final < N_max ); for( int i = i_start ; i <= i_final ; i++ ){ a[i] = f[i]; } for( int j = i_start + 1; j <= i_final ; j++ ){ for( int i = i_final ; i >= j ; i-- ){ ll& a_i = a[i] -= a[i - 1]; a_i < 0 ? a_i += m_p : a_i; } } return; } template <int N_max> void MahlerTransform<N_max>::InverseConvert( const ll ( &a )[N_max] , ll ( &f )[N_max] , const int& i_start , const int& i_final ) { assert( 0 <= i_start && i_start <= i_final && i_final < N_max ); for( int i = i_start ; i <= i_final ; i++ ){ ll& f_i = f[i] = 0; for( int j = i_start ; j <= i ; j++ ){ ( f_i += m_fact[i - i_start] * m_fact_inv[j - i_start] % m_p * m_fact_inv[i - j] % m_p * a[j] % m_p ) < m_p ? f_i : f_i -= m_p; } } return; } template <int N_max> inline ll MahlerTransform<N_max>::Combination( const int& n , const int& i ) const { assert( 0 <= n && n < N_max && 0 <= i ); return n < i ? 0 : m_fact[n] * m_fact_inv[i] % m_p * m_fact_inv[n-i] % m_p; } template <int N_max> inline const ll& MahlerTransform<N_max>::Factorial( const int& n ) const { assert( 0 <= n && n < N_max ); return m_fact[n]; } template <int N_max> inline const ll& MahlerTransform<N_max>::FactorialInverse( const int& n ) const { assert( 0 <= n && n < N_max ); return m_fact_inv[n]; } template <int N_max> inline const ll& MahlerTransform<N_max>::Inverse( const int& n ) const { assert( 0 < n && n < N_max ); return m_inv[n]; } int main() { UNTIE; LIBRARY_SEARCH; START_MAIN; DEXPR( int , bound_N , 500000 , 100 ); // 0が5個 CIN_ASSERT( N , 1 , bound_N ); DEXPR( int , bound_M , 500000 , 100 ); // 0が5個 CIN_ASSERT( M , N , bound_M ); CEXPR( int , bound_K , 26 ); CIN_ASSERT( K , 1 , bound_K ); CIN( string , S_str ); vector<char> S{}; FOR( i , 0 , N ){ S.push_back( S_str[i] ); } CEXPR( ll , P , 998244353 ); ll power[bound_K+1][bound_M+1]; FOREQ( k , 0 , bound_K ){ ll ( &power_k )[bound_M+1] = power[k]; FOREQ( m , 0 , M ){ POWER_MOD( k_power , k , m , P ); power_k[m] = k_power; } } set<char> C{}; MahlerTransform<bound_K+2> mt{ P }; ll answer = 0; FOR( i , 0 , N ){ char S_i = S[i]; int M_i = M - i; ll count[2][bound_K+2] = {}; int size = C.size(); FOREQ( k , size , K ){ count[0][k] = count[1][k] = k == 0 ? 0 : k == 1 ? M_i : ( power[k][M_i] - 1 ) * mt.Inverse( k - 1 ) % P; } ll total = ( power[26][M_i] - 1 ) * mt.Inverse( 25 ) % P; ll a[2]; FOR( j , 0 , 2 ){ ll ( &count_j )[bound_K+2] = count[j]; int n = size + j; ll mahler_count[bound_K+2]; mt.Convert( count_j , mahler_count , n , bound_K + 1 ); ll& a_j = a[j] = total; FOR( k , n , K ){ a_j -= mahler_count[k] * mt.Combination( 26 - n , k - n ) % P; } a_j %= P; } ll num[2] = {}; FOR( c , 'a' , S_i ){ num[1 - C.count( c )]++; } FOR( j , 0 , 2 ){ answer += a[j] * num[j]; } C.insert( S_i ); ( i < N - 1 ? int( C.size() ) >= K : false ) ? ++answer : answer; } ( answer %= P ) < 0 ? answer += P : answer; COUT( ( answer ) ); FINISH_MAIN; QUIT; }