This documentation is automatically generated by online-judge-tools/verification-helper
View the Project on GitHub rainbou-kpr/library
#define PROBLEM "https://yukicoder.me/problems/no/565" #include "../cpp/matrix.hpp" #include <iostream> #include <string> std::string extend(std::string s, int k) { std::string ret = ""; for(char c : s) ret += std::string(k, c); return ret; } int main() { int r, k, h, w; std::cin >> r >> k >> h >> w; std::vector<std::string> c(h); for(int i = 0; i < h; i ++) std::cin >> c[i]; Matrix<char> mat(c); for(int i = 0; i < r; i += 90) mat = mat.transpose().rev_lr(); for(auto s : mat.vstr()) { std::string t = extend(s, k); for(int i = 0; i < k; i ++) std::cout << t << std::endl; } return 0; }
#line 1 "test/yukicoder-rotate-enlarge_1.test.cpp" #define PROBLEM "https://yukicoder.me/problems/no/565" #line 2 "cpp/matrix.hpp" #include <vector> #include <string> #include <algorithm> #include <valarray> #include <cassert> #include <type_traits> namespace matrix { template <typename T> struct OperatorPropertyDefault { constexpr static T zero() { return T(0); } constexpr static T add(const T &a, const T &b) { return a + b; } constexpr static T neg(const T &a) { return -a; } constexpr static T one() { return T(1); } constexpr static T mul(const T &a, const T &b) { return a * b; } constexpr static T inv(const T &a) { return T(1) / a; } }; } /** * @brief 行列 * @tparam T 型 行列(グリッド)の要素となるintやchar * @tparam OperatorProperty 行列の要素の演算を定義する構造体 0,1と+,*なら省略可 * * @note * OperatorPropertyの例(整数のxorとandによる環) * @code * struct XorOperatorProperty { * constexpr static int zero() { return 0; } * constexpr static int add(const int &a, const int &b) { return a ^ b; } * constexpr static int neg(const int &a) { return a; } * constexpr static int one() { return 1; } * constexpr static int mul(const int &a, const int &b) { return a & b; } * }; * @endcode * constexprである必要はなく、またinvなど使わないものは定義しなくてもよい(必要なものがなかったらコンパイルエラーが発生する) */ template<class T, class OperatorProperty = matrix::OperatorPropertyDefault<T>> struct Matrix { int n, m; std::vector<std::vector<T>> v; /** * @brief コンストラクタ * @param v 行列(グリッド)の元となる vector<string> や vector<vector<T>> * @return Matrix */ template <typename Iterable> Matrix(const std::vector<Iterable>& v_) noexcept : n(v_.size()), m(v_.size() == 0 ? 0 : v_[0].size()) { v.resize(n); for(int i = 0; i < n; i++) { v[i].assign(v_[i].begin(), v_[i].end()); } } /** * @brief コンストラクタ * @param _n 行列(グリッド)の行数 * @param _m 行列(グリッド)の列数 * @param _val 行列(グリッド)の要素の初期値 * @return Matrix */ Matrix(int _n, int _m, T _val = OperatorProperty::zero()) : n(_n), m(_m), v(n, std::vector<T>(m, _val)) {} auto begin() noexcept {return v.begin();} auto end() noexcept {return v.end();} /** * @brief 行列(グリッド)の行数 * @return size_t */ [[nodiscard]] size_t size() const {return v.size();} std::vector<T>& operator [] (int i) {return v[i];} const std::vector<T>& operator [] (int i) const {return v[i];} Matrix<T>& operator = (const std::vector<std::vector<T>> &A) noexcept { n = A.size(); m = (n == 0 ? 0 : A[0].size()); v = A; return *this; } bool operator == (const Matrix<T> &A) noexcept { return this->v == A.v; } /** * @brief 転置 * @return Matrix */ [[nodiscard]] Matrix transpose() noexcept { if(n == 0) return Matrix(v); std::vector<std::vector<T>> ret(m); for(int i = 0; i < m; i ++) { ret[i].resize(n); for(int j = 0; j < n; j ++) ret[i][j] = v[j][i]; } return Matrix(ret); } /** * @brief 左右反転 * @return Matrix */ [[nodiscard]] Matrix rev_lr() noexcept { std::vector<std::vector<T>> ret = v; for(int i = 0; i < n; i ++) std::reverse(ret[i].begin(), ret[i].end()); return Matrix(ret); } /** * @brief 上下反転 * @return Matrix */ [[nodiscard]] Matrix rev_ud() noexcept { std::vector<std::vector<T>> ret = v; reverse(ret.begin(), ret.end()); return Matrix(ret); } /** * @brief 時計周りに90度回転 * @param k 回転する回数 * @return Matrix */ [[nodiscard]] Matrix rotate(int k) noexcept { k %= 4; if(k == 0) return *this; if(k < 0) k += 4; if(k == 2) return this->rev_lr().rev_ud(); std::vector<std::vector<T>> ret(m); if(k == 1) { for(int i = 0; i < m; i ++) { ret[i].resize(n); for(int j = 0; j < n; j ++) ret[i][j] = v[n - j - 1][i]; } } else { for(int i = 0; i < m; i ++) { ret[i].resize(n); for(int j = 0; j < n; j ++) ret[i][j] = v[j][m - i - 1]; } } return Matrix(ret); } /** * @brief (i, j)を((i + dy) mod n, (j + dx) mod m)に移動 * @return Matrix */ [[nodiscard]] Matrix shift(int dy, int dx) noexcept { std::vector<std::vector<T>> ret = v; for(int i = 0, ni = dy; i < n; i ++, ni ++) { if(ni >= n) ni = 0; for(int j = 0, nj = dx; j < m; j ++, nj ++) { if(nj >= m) nj = 0; ret[ni][nj] = v[i][j]; } } return Matrix(ret); } /** * @brief 左にk回シフト * @return Matrix */ [[nodiscard]] Matrix shift_l(int k) noexcept { return this->shift(0, -k); } /** * @brief 右にk回シフト * @return Matrix */ [[nodiscard]] Matrix shift_r(int k) noexcept { return this->shift(0, k); } /** * @brief 上にk回シフト * @return Matrix */ [[nodiscard]] Matrix shift_u(int k) noexcept { return this->shift(-k, 0); } /** * @brief 下にk回シフト * @return Matrix */ [[nodiscard]] Matrix shift_d(int k) noexcept { return this->shift(k, 0); } /** * @brief グリッドをvector<string>で返す * @return std::vector<std::string> */ [[nodiscard]] std::vector<std::string> vstr() noexcept { std::vector<std::string> ret(n); for(int i = 0; i < n; i ++) { ret[i].assign(v[i].begin(), v[i].end()); } return ret; } /** * @brief グリッドのj列目を返す * @param j 返す列番号(0-indexed) * @return std::vector<T> */ [[nodiscard]] std::vector<T> col(int j) noexcept { std::vector<T> ret(n); for(int i = 0; i < n; i ++) { ret[i] = v[i][j]; } return ret; } /** * @brief グリッドのi行目をstringで返す * @param i 返す行番号(0-indexed) * @return std::string */ [[nodiscard]] std::string str(int i) noexcept { std::string ret; ret.assign(v[i].begin(), v[i].end()); return ret; } /** * @brief 保持している行列に行列Bを足す * @param B 足す行列 * @return @c *this */ Matrix &operator+=(const Matrix &B) { if(n == 0) return (*this); assert(n == B.size() && m == B[0].size()); for(int i = 0; i < n; i++) for(int j = 0; j < m; j++) (*this)[i][j] = OperatorProperty::add((*this)[i][j], B[i][j]); return (*this); } /** * @brief 保持している行列から行列Bを引く * @param B 引く行列 * @return @c *this */ Matrix &operator-=(const Matrix &B) { if(n == 0) return (*this); assert(n == B.size() && m == B[0].size()); for(int i = 0; i < n; i++) for(int j = 0; j < m; j++) (*this)[i][j] = OperatorProperty::add((*this)[i][j], OperatorProperty::neg(B[i][j])); return (*this); } /** * @brief 保持している行列に行列Bを掛ける * @param B 掛ける行列 * @return @c *this */ Matrix &operator*=(const Matrix &B) { int p = B[0].size(); Matrix<T> C(n, p); for(int i = 0; i < n; i ++) { for(int k = 0; k < m; k ++) { for(int j = 0; j < p; j ++) { C[i][j] = OperatorProperty::add(C[i][j], OperatorProperty::mul((*this)[i][k], B[k][j])); } } } v.swap(C.v); m = p; return (*this); } /** * @brief 保持している行列のk乗を求める * @param k 指数 * @return Matrix */ [[nodiscard]] Matrix pow(long long k) const { Matrix<T> A = *this, B(n, n); for(int i = 0; i < n; i ++) B[i][i] = OperatorProperty::one(); while(k > 0) { if(k & 1) B *= A; A *= A; k >>= 1; } return B; } [[nodiscard]] Matrix operator+(const Matrix &B) const { return (Matrix(*this) += B); } [[nodiscard]] Matrix operator-(const Matrix &B) const { return (Matrix(*this) -= B); } [[nodiscard]] Matrix operator*(const Matrix &B) const { return (Matrix(*this) *= B); } /** * @brief 行列Aと列ベクトルBの積 * @param A Matrix<T> * @param B vector<T> * @return vector<T> 列ベクトル */ [[nodiscard]] friend std::vector<T> operator*(const Matrix &A, const std::vector<T> &B) { std::vector<T> ret(A.n, 0); for(int i = 0; i < A.n; i ++) { for(int j = 0; j < A.m; j ++) { ret[i] = OperatorProperty::add(ret[i], OperatorProperty::mul(A[i][j], B[j])); } } return ret; } /** * @brief 行ベクトルAと行列Bの積 * @param A vector<T> * @param B Matrix<T> * @return vector<T> 行ベクトル */ [[nodiscard]] friend std::vector<T> operator*(const std::vector<T> &A, const Matrix &B) { std::vector<T> ret(B.m, 0); for(int i = 0; i < B.n; i ++) { for(int j = 0; j < B.m; j ++) { ret[j] = OperatorProperty::add(ret[j], OperatorProperty::mul(A[i], B[i][j])); } } return ret; } /** * @brief 行列式 * @return 行列式 */ [[nodiscard]] T det() const { assert(n == m); if(n == 0) return T(1); if constexpr(std::is_same_v<OperatorProperty, matrix::OperatorPropertyDefault<T>>) { std::vector A(n, std::valarray(T(0), n)); for(int i = 0; i < n; i ++) for(int j = 0; j < n; j ++) A[i][j] = this->v[i][j]; return forward_elimination(A); } else { auto A = this->v; return forward_elimination(A); } } /** * @brief 逆行列 * @return 逆行列 */ [[nodiscard]] Matrix inv() const { assert(n == m); if(n == 0) return Matrix(n, n); if constexpr(std::is_same_v<OperatorProperty, matrix::OperatorPropertyDefault<T>>) { std::vector A(n, std::valarray(T(0), n+n)); for(int i = 0; i < n; i ++) { for(int j = 0; j < n; j ++) A[i][j] = this->v[i][j]; A[i][n+i] = T(1); } assert(forward_elimination(A) != T(0)); for(int i = n - 1; i >= 0; i --) { for(int j = i - 1; j >= 0; j --) { A[j] -= A[i] * A[j][i]; } } Matrix<T> ret(n, n); for(int i = 0; i < n; i ++) for(int j = 0; j < n; j ++) ret[i][j] = A[i][n+j]; return ret; } else { std::vector A(n, std::vector<T>(n+n, OperatorProperty::zero())); for(int i = 0; i < n; i ++) { for(int j = 0; j < n; j ++) A[i][j] = this->v[i][j]; A[i][n+i] = OperatorProperty::one(); } assert(forward_elimination(A) != T(0)); for(int i = n - 1; i >= 0; i --) { for(int j = i - 1; j >= 0; j --) { for(int k = n; k < n+n; k ++) { A[j][k] = OperatorProperty::add(A[j][k], OperatorProperty::neg(OperatorProperty::mul(A[i][k], A[j][i]))); } } } Matrix<T> ret(n, n); for(int i = 0; i < n; i ++) for(int j = 0; j < n; j ++) ret[i][j] = A[i][n+j]; return ret; } } private: /** * @brief ガウスの消去法の前進消去を行って上三角行列を作り、行列式を返す * @param A 行列 * @return 行列式 * @note * rank(A) < nの場合は0を返す * 正方行列ではない場合、0以外が残る左からn個の列を使って行列式を計算する */ T forward_elimination(std::vector<std::valarray<T>>& A) const { std::vector<int> pivot_col(n, 0); T d(1); for(int i = 0; i < n; i ++) { if(i - 1 >= 0) pivot_col[i] = pivot_col[i - 1] + 1; while(pivot_col[i] < m) { int pivot = i; if constexpr(std::is_floating_point_v<T>) { for(int j = i + 1; j < n; j ++) if(std::abs(A[j][pivot_col[i]]) > std::abs(A[pivot][pivot_col[i]])) pivot = j; if(std::abs(A[pivot][pivot_col[i]]) < 1e-9) { pivot_col[i] ++; continue; } } else { while(pivot < n && A[pivot][pivot_col[i]] == T(0)) pivot ++; if(A[pivot][pivot_col[i]] == T(0)) { pivot_col[i] ++; continue; } } if(pivot != i) { std::swap(A[i], A[pivot]); d *= -T(1); } break; } if(pivot_col[i] == m) return T(0); T scale = A[i][pivot_col[i]]; d *= scale; A[i] /= scale; for(int j = i + 1; j < n; j ++) A[j] -= A[i] * A[j][pivot_col[i]]; } return d; } /** * @brief ガウスの消去法の前進消去を行って上三角行列を作り、行列式を返す * @param A 行列 * @return 行列式 * @note * rank(A) < nの場合は0を返す * 正方行列ではない場合、0以外が残る左からn個の列を使って行列式を計算する */ T forward_elimination(std::vector<std::vector<T>>& A) const { std::vector<int> pivot_col(n, 0); T d = OperatorProperty::one(); for(int i = 0; i < n; i ++) { if(i - 1 >= 0) pivot_col[i] = pivot_col[i - 1] + 1; while(pivot_col[i] < m) { int pivot = i; while(pivot < n && A[pivot][pivot_col[i]] == OperatorProperty::zero()) pivot ++; if(A[pivot][pivot_col[i]] == OperatorProperty::zero()) { pivot_col[i] ++; continue; } if(pivot != i) { std::swap(A[i], A[pivot]); d = OperatorProperty::neg(d); } break; } if(pivot_col[i] == m) return T(0); T scale = A[i][pivot_col[i]]; d = OperatorProperty::mul(d, scale); for(int j = pivot_col[i]; j < m; j ++) A[i][j] = OperatorProperty::mul(A[i][j], OperatorProperty::inv(scale)); for(int j = i + 1; j < n; j ++) { T scale = A[j][pivot_col[i]]; for(int k = pivot_col[i]; k < m; k ++) { A[j][k] = OperatorProperty::add(A[j][k], OperatorProperty::neg(OperatorProperty::mul(A[i][k], scale))); } } } return d; } }; #line 4 "test/yukicoder-rotate-enlarge_1.test.cpp" #include <iostream> #line 6 "test/yukicoder-rotate-enlarge_1.test.cpp" std::string extend(std::string s, int k) { std::string ret = ""; for(char c : s) ret += std::string(k, c); return ret; } int main() { int r, k, h, w; std::cin >> r >> k >> h >> w; std::vector<std::string> c(h); for(int i = 0; i < h; i ++) std::cin >> c[i]; Matrix<char> mat(c); for(int i = 0; i < r; i += 90) mat = mat.transpose().rev_lr(); for(auto s : mat.vstr()) { std::string t = extend(s, k); for(int i = 0; i < k; i ++) std::cout << t << std::endl; } return 0; }