結果
問題 | No.1815 K色問題 |
ユーザー | 沙耶花 |
提出日時 | 2022-01-20 20:31:30 |
言語 | C++17 (gcc 12.3.0 + boost 1.83.0) |
結果 |
AC
|
実行時間 | 578 ms / 2,000 ms |
コード長 | 4,678 bytes |
コンパイル時間 | 5,602 ms |
コンパイル使用メモリ | 288,384 KB |
実行使用メモリ | 6,912 KB |
最終ジャッジ日時 | 2024-09-29 22:38:12 |
合計ジャッジ時間 | 7,339 ms |
ジャッジサーバーID (参考情報) |
judge3 / judge1 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 8 ms
6,820 KB |
testcase_01 | AC | 9 ms
6,820 KB |
testcase_02 | AC | 8 ms
6,820 KB |
testcase_03 | AC | 7 ms
6,816 KB |
testcase_04 | AC | 8 ms
6,912 KB |
testcase_05 | AC | 8 ms
6,912 KB |
testcase_06 | AC | 201 ms
6,912 KB |
testcase_07 | AC | 53 ms
6,816 KB |
testcase_08 | AC | 381 ms
6,820 KB |
testcase_09 | AC | 37 ms
6,820 KB |
testcase_10 | AC | 50 ms
6,816 KB |
testcase_11 | AC | 125 ms
6,912 KB |
testcase_12 | AC | 9 ms
6,820 KB |
testcase_13 | AC | 8 ms
6,816 KB |
testcase_14 | AC | 519 ms
6,820 KB |
testcase_15 | AC | 578 ms
6,820 KB |
ソースコード
#include <stdio.h> #include <bits/stdc++.h> #include <atcoder/all> using namespace atcoder; using mint = modint1000000007; using namespace std; #define rep(i,n) for(int i=0;i<(n);i++) #define Inf 1000000001 template <class S, S (*op0)(S, S), S (*e0)(), S (*op1)(S, S), S (*e1)() > struct matrix{ array<array<mint,2>,2> v; int _h,_w; matrix(vector<vector<S>> X){ v = X; _h = X.size(); _w = 0; if(X.size()>0)_w = X[0].size(); } matrix(int h,int w){ //v.resize(h,vector<S>(w,e0())); _h = h; _w = w; } void add_element(int from,int to,S x){ v[to][from] = op0(v[to][from],x); } matrix e(){ assert(_h==_w); matrix<S,op0,e0,op1,e1> ret(_h,_w); for(int i=0;i<_h;i++){ for(int j=0;j<_w;j++){ if(i==j)ret.v[i][j] = e1(); else ret.v[i][j] = e0(); } } return ret; } matrix &operator*=(const matrix &another){ matrix<S,op0,e0,op1,e1> ret(_h,another._w); for(int i=0;i<_h;i++){ for(int j=0;j<another._w;j++){ ret.v[i][j] = e0(); for(int k=0;k<_w;k++){ ret.v[i][j] = op0(ret.v[i][j],op1(v[i][k],another.v[k][j])); } } } v = ret.v; return (*this); } matrix operator*(const matrix &another)const{ return (matrix(*this)*=another); } matrix pow(long long cnt){ matrix<S,op0,e0,op1,e1> ret = e(); auto temp = *this; while(cnt!=0LL){ if((cnt&1)==1){ ret *= temp; } temp *= temp; cnt>>=1; } return ret; } }; mint op0(mint a,mint b){ return a+b; } mint e0(){ return 0; } mint op1(mint a,mint b){ return a*b; } mint e1(){ return 1; } long long N,M,K; vector<vector<int>> tt; void check(vector<int> t){ rep(i,2){ rep(j,N-1){ if(t[i*N+j] == t[i*N+j+1])return; } } rep(i,N){ if(t[i]==t[i+N])return; } tt.push_back(t); } void dfs(vector<int> &t,int cur){ if(t.size()==N*2){ check(t); return; } rep(i,cur){ t.push_back(i); dfs(t,cur); t.pop_back(); } t.push_back(cur); cur++; dfs(t,cur); t.pop_back(); } vector<int> trans(vector<int> t){ map<int,int> used; rep(i,t.size()){ if(used.count(t[i])){ t[i] = used[t[i]]; } else{ int tt = used.size(); used[t[i]] = tt; t[i] = tt; } } return t; } struct combi{ deque<mint> kaijou; deque<mint> kaijou_; combi(int n){ kaijou.push_back(1); for(int i=1;i<=n;i++){ kaijou.push_back(kaijou[i-1]*i); } mint b=kaijou[n].inv(); kaijou_.push_front(b); for(int i=1;i<=n;i++){ int k=n+1-i; kaijou_.push_front(kaijou_[0]*k); } } mint combination(int n,int r){ if(r>n)return 0; mint a = kaijou[n]*kaijou_[r]; a *= kaijou_[n-r]; return a; } mint junretsu(int a,int b){ mint x = kaijou_[a]*kaijou_[b]; x *= kaijou[a+b]; return x; } mint catalan(int n){ return combination(2*n,n)/(n+1); } }; int main(){ cin>>N>>M>>K; { vector<int> t; dfs(t,0); } sort(tt.begin(),tt.end()); vector<vector<int>> t; rep(i,tt.size()){ vector<int> temp; rep(j,N){ temp.push_back(tt[i][j]); } t.push_back(temp); } sort(t.begin(),t.end()); t.erase(unique(t.begin(),t.end()),t.end()); //cout<<tt.size()<<endl; mint ans= 0; combi C(400000); vector<int> A(tt.size()),B(tt.size()); vector<vector<int>> Minus(tt.size()); rep(j,tt.size()){ int cnt = 0; vector<bool> f(N*2,false); vector<int> x,y; rep(k,tt[j].size()){ if(k<N){ if(f[tt[j][k]]){ continue; } else{ cnt++; f[tt[j][k]] = true; } x.push_back(tt[j][k]); } else{ if(f[tt[j][k]]){ } else{ Minus[j].push_back(cnt); cnt++; f[tt[j][k]] = true; } y.push_back(tt[j][k]); } } y = trans(y); int d0 = distance(t.begin(),lower_bound(t.begin(),t.end(),x)); int d1 = distance(t.begin(),lower_bound(t.begin(),t.end(),y)); A[j]= d0; B[j] = d1; } vector<vector<int>> Minus2(t.size()); rep(j,t.size()){ int cnt = 0; vector<bool> f(N,false); rep(k,t[j].size()){ if(f[t[j][k]]){ continue; } else{ Minus2[j].push_back(cnt); cnt++;; f[t[j][k]] = true; } } } rep(i,K){ matrix<mint,op0,e0,op1,e1> mat(t.size(),t.size()); rep(j,tt.size()){ mint v = 1; rep(k,Minus[j].size()){ v *= K-i-Minus[j][k]; } mat.add_element(A[j],B[j],v); } mat = mat.pow(M-1); matrix<mint,op0,e0,op1,e1> mat2(t.size(),1); rep(j,t.size()){ mint v = 1; rep(k,Minus2[j].size()){ v *= K-i-Minus2[j][k]; } mat2.v[j][0] = v; } mat2 = mat * mat2; mint sum = 0; rep(j,t.size()){ sum += mat2.v[j][0]; } sum *= C.combination(K,i); if(i%2==1)sum *= -1; ans += sum; } cout<<ans.val()<<endl; return 0; }