泛型编程与OI——modint

博客 动态
0 280
羽尘
羽尘 2022-02-01 16:54:48
悬赏:0 积分 收藏

泛型编程与 OI——modint

博客链接。

在 OI 中,有大量的题目要求对一些数字取模,这便是本文写作的背景。

背景介绍

这些题目要么是因为答案太大,不方便输出结果,例如许多计数 dp;要么是因为答案是浮点数,出题人不愿意写一个确定精度的 Special Judge,例如很多期望概率题;要么是因为这道题目直接考察了模的性质和运用,比如大量的 998244353 类的多项式题目。

过去的做法

在这种要求之下,取模运算就成为了编程中不可缺少的一部分。下面以式子 \(\texttt{ans}=(x+y+z)\times u\) 为例介绍几种写法。

第一种 直接取模

这种方法是直接取模,简单直接,清晰明了。

constexpr int p=998244353;int ans=1ll*(((x+y)%p+z)%p)*u%p;

但是这种方法有着严重的缺陷,一是容易忘记大括号,二是容易中间运算时搞错运算顺序、忘记取模,三是式子太长、括号太多、不易检验。

因此,不推荐运用这种方法。

第二种 函数取模

这种方法有效地解决了直接取模的忘记取模的漏洞。

constexpr int p=998244353;int add(int a,int b){    return a+b>=p?a+b-p:a+b;}int sub(int a,int b){    return a<b?a-b+p:a-b;}int mul(int a,int b){    return 1ll*a*b%p;}int ans=mul(add(add(x,y),z),u);

但是,这种写法的式子依旧太长,不易检验,并且如果编译器没有任何优化(现在不存在这种情况了)的话,大量的函数调用将会耗费不少的时间。并且如果要对多个模数取模,则需要写多个函数,显得代码冗长。

泛型编程

考虑到函数取模的优点,我们不妨通过类的运算符重载来进一步优化 add 等函数。

同时为了解决多个模数的问题,我们考虑泛型编程,将模数直接包含在类型中。

template<typename T,const T p>class modint{	private:		T v;	public:		modint(){}		modint(const T& x){assert(0<=x&&x<p);v=x;}		modint operator+(const modint& a)const{			return v+a.v>=p?v+a.v-p:v+a.v;		}		modint operator-(const modint& a)const{			return v<a.v?v-a.v+p:v-a.v;		}		modint operator*(const modint& a)const{			return 1ll*v*a.v%p;		}		T operator()(void)const{			return v;		}};modint<int,998244353> x(),y(),z(),u();modint<int,998244353> ans=(x+y+z)*u;

这样使用的时候,一方面减少了心智负担,不用操心运算时忘记取模;另一方面采取了常数更小的加减法操作,运算更快。

唯一的缺点就是类型名难写,但是模数个数少的时候可以缩写,即写成:

typedef modint<int,998244353> modInt1;

这样就解决了类型名长的缺点。

posted @ 2022-02-01 15:51 diaearth 阅读(0) 评论(0) 编辑 收藏 举报
回帖
    羽尘

    羽尘 (王者 段位)

    2335 积分 (2)粉丝 (11)源码

     

    温馨提示

    亦奇源码

    最新会员