Typesetting math: 100%

【JZOJ6405】【NOIP2019模拟11.04】c

题目大意

给出一个矩阵的定义:
在这里插入图片描述

求它的逆矩阵的各项平方和。

n1000000,m109+6

Solution

手玩m=0的情况可以发现逆矩阵的定义是类似的:

  • ji,(Pn1)(i,j)=(1)i+j(ji)
  • j>i,(Pn1)(i,j)=0

模拟矩阵乘法就能证明这个结论。

当多了一个jm时,矩阵应该是这样的:

  • ji,(Pn1)(i,j)=(1)i+j(ji)im
  • j>i,(Pn1)(i,j)=0

证明与上面的类似。

于是问题变成求:

i=1ni2mj=1i(ji)2

组合数的平方和即:

i=0nC(n,i)2

转化为:

i=0nC(n,i)C(n,ni)

考虑其组合意义,相当于把一个长度为2n的序列分为两部分,枚举一部分选i个,另一部分选ni个。其实就是在2n个数里选n个,于是上面的式子就变成:

i=1ni2m((i2i)1)

线性求一下逆元就能O(n)解决了。

Code

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;

typedef long long ll;
const int N = 2000007;
const ll P = 1e9 + 7;

int n, m;
ll ans, fac[N], inv[N];

ll pow(ll a, ll b) {
	ll ret = 1;
	for (; b; a = a * a % P, b >>= 1) if (b & 1) ret = ret * a % P;
	return ret;
}
ll C(int n, int m) { return fac[n] * inv[m] % P * inv[n - m] % P; }

int main() {
	freopen("c.in", "r", stdin);
	//freopen("c.out", "w", stdout);
	fac[0] = 1;
	for (int i = 1; i <= 2000000; ++i) fac[i] = fac[i - 1] * i % P;
	inv[0] = inv[1] = 1;
	for (int i = 2; i <= 2000000; ++i) inv[i] = inv[P % i] * (P - P / i) % P;
	for (int i = 2; i <= 2000000; ++i) inv[i] = inv[i] * inv[i - 1] % P;
	scanf("%d%d", &n, &m);
	for (int i = 1; i <= n; ++i) ans = (ans + pow(i, 2 * m) * (C(2 * i, i) - 1 + P) % P) % P;
	printf("%lld\n", ans);
	return 0;
}

作者:zjlcnblogs

出处:https://www.cnblogs.com/zjlcnblogs/p/11795278.html

版权:本作品采用「署名-非商业性使用-相同方式共享 4.0 国际」许可协议进行许可。

posted @   gz-gary  阅读(97)  评论(0编辑  收藏  举报
编辑推荐:
· 一次Java后端服务间歇性响应慢的问题排查记录
· dotnet 源代码生成器分析器入门
· ASP.NET Core 模型验证消息的本地化新姿势
· 对象命名为何需要避免'-er'和'-or'后缀
· SQL Server如何跟踪自动统计信息更新?
阅读排行:
· “你见过凌晨四点的洛杉矶吗?”--《我们为什么要睡觉》
· 编程神器Trae:当我用上后,才知道自己的创造力被低估了多少
· C# 从零开始使用Layui.Wpf库开发WPF客户端
· C#/.NET/.NET Core技术前沿周刊 | 第 31 期(2025年3.17-3.23)
· 接口重试的7种常用方案!
more_horiz
keyboard_arrow_up dark_mode palette
选择主题
点击右上角即可分享
微信分享提示