結果
問題 | No.1810 RGB Biscuits |
ユーザー |
|
提出日時 | 2023-06-15 00:45:02 |
言語 | PyPy3 (7.3.15) |
結果 |
AC
|
実行時間 | 254 ms / 2,000 ms |
コード長 | 7,575 bytes |
コンパイル時間 | 207 ms |
コンパイル使用メモリ | 82,440 KB |
実行使用メモリ | 78,232 KB |
最終ジャッジ日時 | 2024-06-23 05:21:09 |
合計ジャッジ時間 | 4,366 ms |
ジャッジサーバーID (参考情報) |
judge3 / judge1 |
(要ログイン)
ファイルパターン | 結果 |
---|---|
sample | AC * 2 |
other | AC * 20 |
ソースコード
import copyclass matrix():def __init__(self):self.mod = 10**9+7def multiplication(self,arr1,arr2):'''例arr12 3 4 56 7 8 9arr21 23 45 67 8'''H = len(arr1)W = len(arr2[0])arr3 = [[0]*W for i in range(H)]for i in range(H):for j in range(W):val = 0for k in range(len(arr1[0])):val += arr1[i][k]*arr2[k][j]arr3[i][j] = valreturn arr3def determinant(self,arr):'''正方行列N*Nの行列式計算量O(N**3)'''arr_calc = copy.deepcopy(arr)N = len(arr_calc)for i in range(N-1):d = arr_calc[i][i]for j in range(i+1,N):e = arr_calc[j][i]/dfor k in range(i,N):arr_calc[j][k] -= e*arr_calc[i][k]#arr_calc 上△行列det = 1for i in range(N):det *= arr_calc[i][i]return detdef invarr(self,arr):'''正方行列N*Nの逆行列det == 0ならreturn False計算量O(N**3)'''arr_calc = copy.deepcopy(arr)if self.determinant(arr_calc) == 0:return FalseN = len(arr_calc)for i in range(N):v = [0]*(N)v[i] = 1arr_calc[i].extend(v)for i in range(N-1):d = arr_calc[i][i]for j in range(i+1,N):e = arr_calc[j][i]/dfor k in range(i,2*N):arr_calc[j][k] -= e*arr_calc[i][k]for i in range(N-1,-1,-1):d = arr_calc[i][i]for k in range(i,2*N):arr_calc[i][k] /= dfor j in range(i-1,-1,-1):c = arr_calc[j][i]for k in range(i,2*N):arr_calc[j][k] -= c*arr_calc[i][k]inv = [[0]*(N) for i in range(N)]for i in range(N):for j in range(N):inv[i][j] = arr_calc[i][j+N]return invdef SimultaneousE(self,arr):'''3x+2y+z = 44x+5y+6z = 37x+8y+9z = 2->3 2 1 44 5 6 37 8 9 2'''N = len(arr)arr1 = [[0]*(N) for i in range(N)]for i in range(N):for j in range(N):arr1[i][j] = arr[i][j]v = [[0] for i in range(N)]for i in range(N):v[i][0] = arr[i][-1]if self.determinant(arr1) == 0:return Falseinva = self.invarr(arr1)return self.multiplication(inva,v)def invmod(self,a):#mod逆元if a == 0:return 0if a == 1:return 1return (-self.invmod(self.mod % a) * (self.mod // a)) % self.moddef multiplication_mod(self,arr1,arr2):H = len(arr1)W = len(arr2[0])arr3 = [[0]*W for i in range(H)]for i in range(H):for j in range(W):val = 0for k in range(len(arr1[0])):val += arr1[i][k]*arr2[k][j]arr3[i][j] = val%self.modreturn arr3def determinant_mod(self,arr):'''正方行列N*Nの行列式計算量O(N**3)'''arr_calc = copy.deepcopy(arr)N = len(arr_calc)for i in range(N-1):d = arr_calc[i][i]for j in range(i+1,N):e = arr_calc[j][i]*self.invmod(d)e %= self.modfor k in range(i,N):arr_calc[j][k] -= e*arr_calc[i][k]arr_calc[j][k] %= self.mod#arr_calc 上△行列det = 1for i in range(N):det *= arr_calc[i][i]det %= self.modreturn detdef invarr_mod(self,arr):'''正方行列N*Nの逆行列det == 0ならreturn False計算量O(N**3)'''arr_calc = copy.deepcopy(arr)det = self.determinant_mod(arr_calc)if det == 0:return FalseN = len(arr_calc)for i in range(N):v = [0]*(N)v[i] = detarr_calc[i].extend(v)for i in range(N-1):d = arr_calc[i][i]for j in range(i+1,N):e = arr_calc[j][i]*self.invmod(d)for k in range(i,2*N):arr_calc[j][k] -= e*arr_calc[i][k]arr_calc[j][k] %= self.modfor i in range(N-1,-1,-1):d = arr_calc[i][i]for k in range(i,2*N):arr_calc[i][k] *= self.invmod(d)for j in range(i-1,-1,-1):c = arr_calc[j][i]for k in range(i,2*N):arr_calc[j][k] -= c*arr_calc[i][k]arr_calc[j][k] %= self.modinv = [[0]*(N) for i in range(N)]for i in range(N):for j in range(N):inv[i][j] = arr_calc[i][j+N]*self.invmod(det)%self.modreturn invdef SimultaneousE_mod(self,arr):'''3x+2y+z = 44x+5y+6z = 37x+8y+9z = 2->3 2 1 44 5 6 37 8 9 2'''N = len(arr)arr1 = [[0]*(N) for i in range(N)]for i in range(N):for j in range(N):arr1[i][j] = arr[i][j]v = [[0] for i in range(N)]for i in range(N):v[i][0] = arr[i][-1]det = self.determinant_mod(arr1)if det == 0:return Falseinva = self.invarr_mod(arr1)v2 = self.multiplication_mod(inva,v)for i in range(N):v2[i][0] %= self.modreturn v2def modPow_matrix(self,arr,n):'''N*Nの正方行列arrをn乗する。'''N = len(arr)if n==0:arr1 = [[0]*(N) for i in range(N)]for i in range(N):arr1[i][i] = 1return arr1if n==1:for i in range(N):for j in range(N):arr[i][j] %= self.modreturn arrif n % 2 == 1:arr2 = self.multiplication_mod(arr,self.modPow_matrix(arr,n-1))return arr2arr3 = self.modPow_matrix(arr,n//2)return self.multiplication_mod(arr3,arr3)def Pow_matrix(self,arr,n):'''N*Nの正方行列arrをn乗する。'''N = len(arr)if n==0:arr1 = [[0]*(N) for i in range(N)]for i in range(N):arr1[i][i] = 1return arr1if n==1:return arrif n % 2 == 1:arr2 = self.multiplication(arr,self.Pow_matrix(arr,n-1))return arr2arr3 = self.Pow_matrix(arr,n//2)return self.multiplication(arr3,arr3)A,B = map(int,input().split())N = int(input())mod = 10**9+7MX = matrix()ansls = []arr = [[A,B,1],[1,0,0],[0,0,0]]arr2 = [[1,0,0],[0,1,0],[A,B,1]]ini = [[1],[1],[0]]for i in range(N):T = int(input())T2 = T//2Tr = T%2md = MX.modPow_matrix(arr, T2)if Tr==1:md = MX.multiplication_mod(arr2,md)vec = MX.multiplication_mod(md,ini)ans = sum([vec[i][0] for i in range(3)])%modansls.append(ans)print(*ansls,sep='\n')