제출 #1034695

#제출 시각아이디문제언어결과실행 시간메모리
1034695xnqs회문 (APIO14_palindrome)C++17
73 / 100
559 ms131072 KiB
#include <iostream> #include <fstream> #include <vector> #include <queue> #include <utility> #include <algorithm> #include <unordered_map> #include <map> struct TrieNode { int lazy = 0; std::unordered_map<int, TrieNode*> next; ~TrieNode() { for (auto& i : next) { delete i.second; } } }; template<const int PRIME = 53, const int MOD = 1000000007, const bool REV = 0> class StringHasher { private: int len; std::vector<int> pPRIME; std::vector<int> pfx_hash; public: StringHasher(std::string str) { len = str.size(); pPRIME.assign(str.size(),0); pPRIME[0] = 1; pfx_hash.assign(str.size(),0); pfx_hash[0] = str[((REV) ? str.size()-1 : 0)] - 'a' + 1; for (int i = 1; i < str.size(); i++) { pPRIME[i] = 1LL*pPRIME[i-1]*PRIME%MOD; pfx_hash[i] = 1LL*pfx_hash[i-1]*PRIME%MOD + (str[((REV) ? str.size()-i-1 : i)] - 'a' + 1); if (pfx_hash[i]>=MOD) { pfx_hash[i] -= MOD; } } } inline int hash_query(int l, int r) { if (REV) { l = len-l-1; r = len-r-1; std::swap(l,r); } int ret = pfx_hash[r] - ((l-1>=0) ? 1LL*pPRIME[r-l+1]*pfx_hash[l-1]%MOD : 0); if (ret<0) { ret += MOD; } return ret; } int operator() (int l, int r) { return hash_query(l,r); } }; class PairHasher { public: std::size_t operator() (const std::pair<int,int>& p) const { return 1000000069LL*p.first + p.second; } }; std::string str; TrieNode* root; std::unordered_map<std::pair<int,int>, TrieNode*, PairHasher> map; void insert(TrieNode* node, const std::string& str, int l, int r, int h1, int h2) { if (l>r+1) { return; } for (int i = l; i <= r; i++) { if (!node->next[str[i]-'a']) { node->next[str[i]-'a'] = new TrieNode; } node = node->next[str[i]-'a']; h1 = 1LL*h1*53%1000000007 + (str[i]-'a'+1); if (h1>=1000000007) { h1 -= 1000000007; } h2 = 1LL*h2*53%1000000009 + (str[i]-'a'+1); if (h2>=1000000009) { h2 -= 1000000009; } map[std::pair<int,int>(h1,h2)] = node; } node->lazy += 1; } void insert_naive(TrieNode* node, const std::string& str, int l, int r) { for (int i = l; i <= r; i++) { if (!node->next[str[i]-'a']) { node->next[str[i]-'a'] = new TrieNode; } node = node->next[str[i]-'a']; } node->lazy += 1; } std::pair<int,int64_t> dfs(TrieNode* node, bool odd = 1, int depth = 0) { int freq = node->lazy; int64_t max = 0; for (int i = 0; i < 26; i++) { if (node->next[i]!=nullptr) { auto [a, b] = dfs(node->next[i], odd, depth+1); freq += a; max = std::max(max, b); } } max = std::max(max, static_cast<int64_t>(2LL*depth-odd)*freq); return std::pair<int,int64_t>(freq,max); } int main() { std::ios_base::sync_with_stdio(false); std::cin.tie(nullptr); std::cout.tie(nullptr); std::cin >> str; StringHasher<53, 1000000007, 0> hash1(str); StringHasher<53, 1000000009, 0> hash2(str); StringHasher<53, 1000000007, 1> hash1_r(str); StringHasher<53, 1000000009, 1> hash2_r(str); int64_t ans = 0; // odd length #if 1 root = new TrieNode; for (int i = 0; i < str.size(); i++) { // bs length of palindrome centered in i int len = 1; int l = 1, r = str.size(); while (l<=r) { int m = (l+r)>>1; if (i-m+1>=0&&i+m-1<str.size()&& hash1(i-m+1,i)==hash1_r(i,i+m-1)&& hash2(i-m+1,i)==hash2_r(i,i+m-1)) { len = m; l = m+1; } else { r = m-1; } } // bs length of longest prefix in trie so we get amortized build time TrieNode* longest_pfx = root; int longest_pfx_len = 0; int longest_pfx_h1 = 0; int longest_pfx_h2 = 0; l = 1, r = len; while (l<=r) { int m = (l+r)>>1; int h1 = hash1(i,i+m-1); int h2 = hash2(i,i+m-1); if (!map.count(std::pair<int,int>(h1,h2))) { r = m-1; } else { longest_pfx = map[std::pair<int,int>(h1,h2)]; longest_pfx_len = m; longest_pfx_h1 = h1; longest_pfx_h2 = h2; l = m+1; } } insert(longest_pfx,str,i+longest_pfx_len,i+len-1,longest_pfx_h1,longest_pfx_h2); } ans = std::max(ans, dfs(root).second); delete root; map.clear(); #endif // even length #if 1 root = new TrieNode; for (int i = 1; i < str.size(); i++) { if (str[i-1]!=str[i]) { continue; } // bs length of palindrome centered in i int len = 1; int l = 1, r = str.size(); while (l<=r) { int m = (l+r)>>1; if (i-m>=0&&i+m-1<str.size()&& hash1(i-m,i-1)==hash1_r(i,i+m-1)&& hash2(i-m,i-1)==hash2_r(i,i+m-1)) { len = m; l = m+1; } else { r = m-1; } } // bs length of longest prefix in trie so we get amortized build time TrieNode* longest_pfx = root; int longest_pfx_len = 0; int longest_pfx_h1 = 0; int longest_pfx_h2 = 0; l = 1, r = len; while (l<=r) { int m = (l+r)>>1; int h1 = hash1(i,i+m-1); int h2 = hash2(i,i+m-1); if (!map.count(std::pair<int,int>(h1,h2))) { r = m-1; } else { longest_pfx = map[std::pair<int,int>(h1,h2)]; longest_pfx_len = m; longest_pfx_h1 = h1; longest_pfx_h2 = h2; l = m+1; } } insert(longest_pfx,str,i+longest_pfx_len,i+len-1,longest_pfx_h1,longest_pfx_h2); } ans = std::max(ans, dfs(root, /*odd=*/0).second); delete root; #endif std::cout << ans << "\n"; }

컴파일 시 표준 에러 (stderr) 메시지

palindrome.cpp: In function 'int main()':
palindrome.cpp:136:20: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  136 |  for (int i = 0; i < str.size(); i++) {
      |                  ~~^~~~~~~~~~~~
palindrome.cpp:142:23: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  142 |    if (i-m+1>=0&&i+m-1<str.size()&&
      |                  ~~~~~^~~~~~~~~~~
palindrome.cpp:186:20: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  186 |  for (int i = 1; i < str.size(); i++) {
      |                  ~~^~~~~~~~~~~~
palindrome.cpp:196:21: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  196 |    if (i-m>=0&&i+m-1<str.size()&&
      |                ~~~~~^~~~~~~~~~~
palindrome.cpp: In instantiation of 'StringHasher<PRIME, MOD, REV>::StringHasher(std::string) [with int PRIME = 53; int MOD = 1000000007; bool REV = false; std::string = std::__cxx11::basic_string<char>]':
palindrome.cpp:126:43:   required from here
palindrome.cpp:33:21: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
   33 |   for (int i = 1; i < str.size(); i++) {
      |                   ~~^~~~~~~~~~~~
palindrome.cpp: In instantiation of 'StringHasher<PRIME, MOD, REV>::StringHasher(std::string) [with int PRIME = 53; int MOD = 1000000009; bool REV = false; std::string = std::__cxx11::basic_string<char>]':
palindrome.cpp:127:43:   required from here
palindrome.cpp:33:21: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
palindrome.cpp: In instantiation of 'StringHasher<PRIME, MOD, REV>::StringHasher(std::string) [with int PRIME = 53; int MOD = 1000000007; bool REV = true; std::string = std::__cxx11::basic_string<char>]':
palindrome.cpp:128:45:   required from here
palindrome.cpp:33:21: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
palindrome.cpp: In instantiation of 'StringHasher<PRIME, MOD, REV>::StringHasher(std::string) [with int PRIME = 53; int MOD = 1000000009; bool REV = true; std::string = std::__cxx11::basic_string<char>]':
palindrome.cpp:129:45:   required from here
palindrome.cpp:33:21: warning: comparison of integer expressions of different signedness: 'int' and 'std::__cxx11::basic_string<char>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...