mirror of
https://gitlab.com/mfocko/Codeforces.git
synced 2024-12-22 11:51:24 +01:00
chore(rs): add ‹Z›
Signed-off-by: Matej Focko <me@mfocko.xyz>
This commit is contained in:
parent
ee4703623b
commit
c376b11d32
1 changed files with 106 additions and 19 deletions
|
@ -40,31 +40,118 @@ fn main() {
|
||||||
|
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
mod math {
|
mod math {
|
||||||
const MOD: i64 = 1_000_000_007;
|
#[derive(Copy, Clone, Default)]
|
||||||
|
pub struct Z(i64);
|
||||||
|
|
||||||
pub fn add(a: i64, b: i64) -> i64 {
|
impl Z {
|
||||||
(a + b) % MOD
|
const MOD: i64 = 1_000_000_007;
|
||||||
}
|
|
||||||
|
|
||||||
pub fn sub(a: i64, b: i64) -> i64 {
|
pub fn new(x: i64) -> Z {
|
||||||
((a - b) % MOD + MOD) % MOD
|
Z(x.rem_euclid(Z::MOD))
|
||||||
}
|
|
||||||
|
|
||||||
pub fn mul(a: i64, b: i64) -> i64 {
|
|
||||||
(a * b) % MOD
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn exp(b: i64, e: i64) -> i64 {
|
|
||||||
if e == 0 {
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let half = exp(b, e / 2);
|
pub fn pow(self, mut exp: u32) -> Z {
|
||||||
if e % 2 == 0 {
|
let mut ans = Z::new(1);
|
||||||
return mul(half, half);
|
let mut base = self;
|
||||||
|
while exp > 0 {
|
||||||
|
if exp % 2 == 1 {
|
||||||
|
ans *= base;
|
||||||
|
}
|
||||||
|
base *= base;
|
||||||
|
exp >>= 1;
|
||||||
|
}
|
||||||
|
ans
|
||||||
}
|
}
|
||||||
|
|
||||||
mul(half, mul(half, b))
|
pub fn inv(self) -> Z {
|
||||||
|
assert_ne!(self.0, 0);
|
||||||
|
self.pow((Z::MOD - 2) as u32)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::Neg for Z {
|
||||||
|
type Output = Z;
|
||||||
|
|
||||||
|
fn neg(self) -> Z {
|
||||||
|
Z::new(-self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::Add<Z> for Z {
|
||||||
|
type Output = Z;
|
||||||
|
|
||||||
|
fn add(self, rhs: Z) -> Z {
|
||||||
|
Z::new(self.0 + rhs.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::Sub<Z> for Z {
|
||||||
|
type Output = Z;
|
||||||
|
|
||||||
|
fn sub(self, rhs: Z) -> Z {
|
||||||
|
Z::new(self.0 - rhs.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::Mul<Z> for Z {
|
||||||
|
type Output = Z;
|
||||||
|
|
||||||
|
fn mul(self, rhs: Z) -> Z {
|
||||||
|
Z::new(self.0 * rhs.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::Div<Z> for Z {
|
||||||
|
type Output = Z;
|
||||||
|
|
||||||
|
fn div(self, rhs: Z) -> Z {
|
||||||
|
#![allow(clippy::suspicious_arithmetic_impl)]
|
||||||
|
self * rhs.inv()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::AddAssign<Z> for Z {
|
||||||
|
fn add_assign(&mut self, rhs: Z) {
|
||||||
|
*self = *self + rhs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::SubAssign<Z> for Z {
|
||||||
|
fn sub_assign(&mut self, rhs: Z) {
|
||||||
|
*self = *self - rhs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::MulAssign<Z> for Z {
|
||||||
|
fn mul_assign(&mut self, rhs: Z) {
|
||||||
|
*self = *self * rhs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::ops::DivAssign<Z> for Z {
|
||||||
|
fn div_assign(&mut self, rhs: Z) {
|
||||||
|
*self = *self / rhs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Display for Z {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for Z {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::str::FromStr for Z {
|
||||||
|
type Err = std::num::ParseIntError;
|
||||||
|
|
||||||
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
|
Ok(Z::new(s.parse()?))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A trait implementing the unsigned bit shifts.
|
/// A trait implementing the unsigned bit shifts.
|
||||||
|
|
Loading…
Reference in a new issue