Submission #14497112
Source Code Expand
#include<bits/stdc++.h>
template <int MOD_> struct modnum {
static constexpr int MOD = MOD_;
static_assert(MOD_ > 0, "MOD must be positive");
private:
using ll = long long;
int v;
static int minv(int a, int m) {
a %= m;
assert(a);
return a == 1 ? 1 : int(m - ll(minv(m, a)) * ll(m) / a);
}
public:
modnum() : v(0) {}
modnum(ll v_) : v(int(v_ % MOD)) { if (v < 0) v += MOD; }
explicit operator int() const { return v; }
friend std::ostream& operator << (std::ostream& out, const modnum& n) { return out << int(n); }
friend std::istream& operator >> (std::istream& in, modnum& n) { ll v_; in >> v_; n = modnum(v_); return in; }
friend bool operator == (const modnum& a, const modnum& b) { return a.v == b.v; }
friend bool operator != (const modnum& a, const modnum& b) { return a.v != b.v; }
modnum inv() const {
modnum res;
res.v = minv(v, MOD);
return res;
}
friend modnum inv(const modnum& m) { return m.inv(); }
modnum neg() const {
modnum res;
res.v = v ? MOD-v : 0;
return res;
}
friend modnum neg(const modnum& m) { return m.neg(); }
modnum operator- () const {
return neg();
}
modnum operator+ () const {
return modnum(*this);
}
modnum& operator ++ () {
v ++;
if (v == MOD) v = 0;
return *this;
}
modnum& operator -- () {
if (v == 0) v = MOD;
v --;
return *this;
}
modnum& operator += (const modnum& o) {
v += o.v;
if (v >= MOD) v -= MOD;
return *this;
}
modnum& operator -= (const modnum& o) {
v -= o.v;
if (v < 0) v += MOD;
return *this;
}
modnum& operator *= (const modnum& o) {
v = int(ll(v) * ll(o.v) % MOD);
return *this;
}
modnum& operator /= (const modnum& o) {
return *this *= o.inv();
}
friend modnum operator ++ (modnum& a, int) { modnum r = a; ++a; return r; }
friend modnum operator -- (modnum& a, int) { modnum r = a; --a; return r; }
friend modnum operator + (const modnum& a, const modnum& b) { return modnum(a) += b; }
friend modnum operator - (const modnum& a, const modnum& b) { return modnum(a) -= b; }
friend modnum operator * (const modnum& a, const modnum& b) { return modnum(a) *= b; }
friend modnum operator / (const modnum& a, const modnum& b) { return modnum(a) /= b; }
};
template <typename T, int NDIMS> struct tensor_view {
static_assert(NDIMS >= 0);
protected:
std::array<int, NDIMS> shape;
std::array<int, NDIMS> strides;
T* data;
tensor_view(std::array<int, NDIMS> shape_, std::array<int, NDIMS> strides_, T* data_) : shape(shape_), strides(strides_), data(data_) {}
public:
tensor_view() : shape{0}, strides{0}, data(nullptr) {}
protected:
int flatten_index(std::array<int, NDIMS> idx) const {
int res = 0;
for (int i = 0; i < NDIMS; i++) { res += idx[i] * strides[i]; }
return res;
}
int flatten_index_checked(std::array<int, NDIMS> idx) const {
int res = 0;
for (int i = 0; i < NDIMS; i++) {
assert(0 <= idx[i] && idx[i] < shape[i]);
res += idx[i] * strides[i];
}
return res;
}
public:
T& operator[] (std::array<int, NDIMS> idx) const {
return data[flatten_index(idx)];
}
T& at(std::array<int, NDIMS> idx) const {
return data[flatten_index_checked(idx)];
}
template <int D = NDIMS>
std::enable_if_t<(0 < D), tensor_view<T, NDIMS-1>> operator[] (int idx) const {
std::array<int, NDIMS-1> nshape; std::copy(shape.begin()+1, shape.end(), nshape.begin());
std::array<int, NDIMS-1> nstrides; std::copy(strides.begin()+1, strides.end(), nstrides.begin());
T* ndata = data + (strides[0] * idx);
return tensor_view<T, NDIMS-1>(nshape, nstrides, ndata);
}
template <int D = NDIMS>
std::enable_if_t<(0 < D), tensor_view<T, NDIMS-1>> at(int idx) const {
assert(0 <= idx && idx < shape[0]);
return operator[](idx);
}
template <int D = NDIMS>
std::enable_if_t<(0 == D), T&> operator * () const {
return *data;
}
template <typename U, int D> friend struct tensor_view;
template <typename U, int D> friend struct tensor;
};
template <typename T, int NDIMS> struct tensor {
protected:
std::array<int, NDIMS> shape;
std::array<int, NDIMS> strides;
int len;
T* data;
public:
tensor() : shape{0}, strides{0}, len(0), data(nullptr) {}
explicit tensor(std::array<int, NDIMS> shape_, const T& t = T()) {
shape = shape_;
strides[NDIMS-1] = 1;
for (int i = NDIMS-1; i > 0; i--) {
strides[i-1] = strides[i] * shape[i];
}
len = strides[0] * shape[0];
data = new T[len];
std::fill(data, data + len, t);
}
tensor(const tensor& o) : shape(o.shape), strides(o.strides), len(o.len), data(new T[len]) {
for (int i = 0; i < len; i++) {
data[i] = o.data[i];
}
}
tensor& operator=(tensor&& o) noexcept {
using std::swap;
swap(shape, o.shape);
swap(strides, o.strides);
swap(len, o.len);
swap(data, o.data);
}
tensor(tensor&& o) : tensor() {
*this = std::move(o);
}
tensor& operator=(const tensor& o) {
return *this = tensor(o);
}
~tensor() { delete[] data; }
using view_t = tensor_view<T, NDIMS>;
view_t view() {
return tensor_view<T, NDIMS>(shape, strides, data);
}
operator view_t() {
return view();
}
using const_view_t = tensor_view<const T, NDIMS>;
const_view_t view() const {
return tensor_view<const T, NDIMS>(shape, strides, data);
}
operator const_view_t() const {
return view();
}
T& operator[] (std::array<int, NDIMS> idx) { return view()[idx]; }
T& at(std::array<int, NDIMS> idx) { return view().at(idx); }
const T& operator[] (std::array<int, NDIMS> idx) const { return view()[idx]; }
const T& at(std::array<int, NDIMS> idx) const { return view().at(idx); }
template <int D = NDIMS>
std::enable_if_t<(0 < D), tensor_view<T, NDIMS-1>> operator[] (int idx) {
return view()[idx];
}
template <int D = NDIMS>
std::enable_if_t<(0 < D), tensor_view<T, NDIMS-1>> at(int idx) {
return view().at(idx);
}
template <int D = NDIMS>
std::enable_if_t<(0 < D), tensor_view<const T, NDIMS-1>> operator[] (int idx) const {
return view()[idx];
}
template <int D = NDIMS>
std::enable_if_t<(0 < D), tensor_view<const T, NDIMS-1>> at(int idx) const {
return view().at(idx);
}
template <int D = NDIMS>
std::enable_if_t<(0 == D), T&> operator * () {
return *view();
}
template <int D = NDIMS>
std::enable_if_t<(0 == D), const T&> operator * () const {
return *view();
}
};
int main() {
using namespace std;
ios_base::sync_with_stdio(false), cin.tie(nullptr);
using num = modnum<998244353>;
int A, B, C, D; cin >> A >> B >> C >> D;
assert(A <= C && B <= D);
tensor<array<num, 2>, 2> dp({C+1, D+1});
dp[{A,B}][1] ++;
for (int i = A; i <= C; i++) {
for (int j = B; j <= D; j++) {
if (i < C) {
dp[{i+1, j}][0] += dp[{i,j}][0] * j;
dp[{i+1, j}][0] += dp[{i,j}][1] * j;
}
if (j < D) {
dp[{i, j+1}][1] += dp[{i,j}][0];
dp[{i, j+1}][1] += dp[{i,j}][1] * i;
}
}
}
cout << dp[{C,D}][0] + dp[{C,D}][1] << '\n';
return 0;
}
Submission Info
| Submission Time |
|
| Task |
B - Extension |
| User |
ecnerwala |
| Language |
C++ (GCC 9.2.1) |
| Score |
600 |
| Code Size |
7107 Byte |
| Status |
AC |
| Exec Time |
141 ms |
| Memory |
73532 KiB |
Judge Result
| Set Name |
Sample |
All |
| Score / Max Score |
0 / 0 |
600 / 600 |
| Status |
|
|
| Set Name |
Test Cases |
| Sample |
s1.txt, s2.txt, s3.txt |
| All |
01.txt, 02.txt, 03.txt, 04.txt, 05.txt, 06.txt, 07.txt, 08.txt, 09.txt, 10.txt, 11.txt, 12.txt, 13.txt, 14.txt, s1.txt, s2.txt, s3.txt |
| Case Name |
Status |
Exec Time |
Memory |
| 01.txt |
AC |
141 ms |
73532 KiB |
| 02.txt |
AC |
120 ms |
73348 KiB |
| 03.txt |
AC |
7 ms |
3580 KiB |
| 04.txt |
AC |
2 ms |
3652 KiB |
| 05.txt |
AC |
60 ms |
68716 KiB |
| 06.txt |
AC |
60 ms |
73516 KiB |
| 07.txt |
AC |
49 ms |
26016 KiB |
| 08.txt |
AC |
64 ms |
52284 KiB |
| 09.txt |
AC |
6 ms |
3512 KiB |
| 10.txt |
AC |
63 ms |
46316 KiB |
| 11.txt |
AC |
10 ms |
6292 KiB |
| 12.txt |
AC |
65 ms |
72744 KiB |
| 13.txt |
AC |
85 ms |
65764 KiB |
| 14.txt |
AC |
4 ms |
3540 KiB |
| s1.txt |
AC |
3 ms |
3600 KiB |
| s2.txt |
AC |
3 ms |
3560 KiB |
| s3.txt |
AC |
2 ms |
3616 KiB |