普洱高端网站建设价格企业网站需要注意什么
本文涉及知识点
C++动态规划
 【矩阵快速幂】封装类及测试用例及样例
P8624 [蓝桥杯 2015 省 AB] 垒骰子
题目描述
赌圣 atm 晚年迷恋上了垒骰子,就是把骰子一个垒在另一个上边,不能歪歪扭扭,要垒成方柱体。
经过长期观察,atm 发现了稳定骰子的奥秘:有些数字的面贴着会互相排斥!
我们先来规范一下骰子: 1 1 1 的对面是 4 4 4, 2 2 2 的对面是 5 5 5, 3 3 3 的对面是 6 6 6。
假设有 m m m 组互斥现象,每组中的那两个数字的面紧贴在一起,骰子就不能稳定的垒起来。
atm 想计算一下有多少种不同的可能的垒骰子方式。
两种垒骰子方式相同,当且仅当这两种方式中对应高度的骰子的对应数字的朝向都相同。
由于方案数可能过多,请输出模 1 0 9 + 7 10^9+7 109+7 的结果。
不要小看了 atm 的骰子数量哦~。
输入格式
第一行两个整数 n n n, m m m。
n n n 表示骰子数目。
接下来 m m m 行,每行两个整数 a a a, b b b,表示 a a a 和 b b b 数字不能紧贴在一起。
输出格式
一行一个数,表示答案模 1 0 9 + 7 10^9+7 109+7 的结果。
输入输出样例 #1
输入 #1
2 1
1 2
输出 #1
544
说明/提示
对于 30 % 30\% 30% 的数据: n ≤ 5 n \le 5 n≤5。
对于 60 % 60\% 60% 的数据: n ≤ 100 n \le 100 n≤100。
对于 100 % 100\% 100% 的数据: 0 < n ≤ 1 0 9 , m ≤ 36 0<n \le 10^9,m \le 36 0<n≤109,m≤36。
时限 2 秒, 256M
蓝桥杯 2015 年省赛 AB 组 I 题。
P8624 状态机动态规划 矩阵快速幂
1到6,改成0到5。
 初始pre 6个4。pre[i]表示最顶端的骰子为i的方案数。
 当骰子i面向上时,水平方向可以旋转,故有4种状态。
 6*6的mat初始全为4。
 如果(i,j)不能沾在一起,则 k1 = (j+3)%6 k2=(i+3)%6 mat[i][k1]=0,mat[j][k2]=0
 ans = pre * matn-1
 答案是ans之和。
单元测试
#include <iostream>
#include <sstream>
#include <vector>
#include<map>
#include<unordered_map>
#include<set>
#include<unordered_set>
#include<string>
#include<algorithm>
#include<functional>
#include<queue>
#include <stack>
#include<iomanip>
#include<numeric>
#include <math.h>
#include <climits>
#include<assert.h>
#include<cstring>
#include<list>#include <bitset>
using namespace std;template<class T1, class T2>
std::istream& operator >> (std::istream& in, pair<T1, T2>& pr) {in >> pr.first >> pr.second;return in;
}template<class T1, class T2, class T3 >
std::istream& operator >> (std::istream& in, tuple<T1, T2, T3>& t) {in >> get<0>(t) >> get<1>(t) >> get<2>(t);return in;
}template<class T1, class T2, class T3, class T4 >
std::istream& operator >> (std::istream& in, tuple<T1, T2, T3, T4>& t) {in >> get<0>(t) >> get<1>(t) >> get<2>(t) >> get<3>(t);return in;
}template<class T = int>
vector<T> Read() {int n;scanf("%d", &n);vector<T> ret(n);for (int i = 0; i < n; i++) {cin >> ret[i];}return ret;
}template<class T = int>
vector<T> Read(int n) {vector<T> ret(n);for (int i = 0; i < n; i++) {cin >> ret[i];}return ret;
}template<int N = 1'000'000>
class COutBuff
{
public:COutBuff() {m_p = puffer;}template<class T>void write(T x) {int num[28], sp = 0;if (x < 0)*m_p++ = '-', x = -x;if (!x)*m_p++ = 48;while (x)num[++sp] = x % 10, x /= 10;while (sp)*m_p++ = num[sp--] + 48;AuotToFile();}void writestr(const char* sz) {strcpy(m_p, sz);m_p += strlen(sz);AuotToFile();}inline void write(char ch){*m_p++ = ch;AuotToFile();}inline void ToFile() {fwrite(puffer, 1, m_p - puffer, stdout);m_p = puffer;}~COutBuff() {ToFile();}
private:inline void AuotToFile() {if (m_p - puffer > N - 100) {ToFile();}}char  puffer[N], * m_p;
};template<int N = 1'000'000>
class CInBuff
{
public:inline CInBuff() {}inline CInBuff<N>& operator>>(char& ch) {FileToBuf();ch = *S++;return *this;}inline CInBuff<N>& operator>>(int& val) {FileToBuf();int x(0), f(0);while (!isdigit(*S))f |= (*S++ == '-');while (isdigit(*S))x = (x << 1) + (x << 3) + (*S++ ^ 48);val = f ? -x : x; S++;//忽略空格换行		return *this;}inline CInBuff& operator>>(long long& val) {FileToBuf();long long x(0); int f(0);while (!isdigit(*S))f |= (*S++ == '-');while (isdigit(*S))x = (x << 1) + (x << 3) + (*S++ ^ 48);val = f ? -x : x; S++;//忽略空格换行return *this;}template<class T1, class T2>inline CInBuff& operator>>(pair<T1, T2>& val) {*this >> val.first >> val.second;return *this;}template<class T1, class T2, class T3>inline CInBuff& operator>>(tuple<T1, T2, T3>& val) {*this >> get<0>(val) >> get<1>(val) >> get<2>(val);return *this;}template<class T1, class T2, class T3, class T4>inline CInBuff& operator>>(tuple<T1, T2, T3, T4>& val) {*this >> get<0>(val) >> get<1>(val) >> get<2>(val) >> get<3>(val);return *this;}template<class T = int>inline CInBuff& operator>>(vector<T>& val) {int n;*this >> n;val.resize(n);for (int i = 0; i < n; i++) {*this >> val[i];}return *this;}template<class T = int>vector<T> Read(int n) {vector<T> ret(n);for (int i = 0; i < n; i++) {*this >> ret[i];}return ret;}template<class T = int>vector<T> Read() {vector<T> ret;*this >> ret;return ret;}
private:inline void FileToBuf() {const int canRead = m_iWritePos - (S - buffer);if (canRead >= 100) { return; }if (m_bFinish) { return; }for (int i = 0; i < canRead; i++){buffer[i] = S[i];//memcpy出错			}m_iWritePos = canRead;buffer[m_iWritePos] = 0;S = buffer;int readCnt = fread(buffer + m_iWritePos, 1, N - m_iWritePos, stdin);if (readCnt <= 0) { m_bFinish = true; return; }m_iWritePos += readCnt;buffer[m_iWritePos] = 0;S = buffer;}int m_iWritePos = 0; bool m_bFinish = false;char buffer[N + 10], * S = buffer;
};class CMatMul
{
public:CMatMul(long long llMod = 1e9 + 7) :m_llMod(llMod) {}// 矩阵乘法vector<vector<long long>> multiply(const vector<vector<long long>>& a, const vector<vector<long long>>& b) {const int r = a.size(), c = b.front().size(), iK = a.front().size();assert(iK == b.size());vector<vector<long long>> ret(r, vector<long long>(c));for (int i = 0; i < r; i++){for (int j = 0; j < c; j++){for (int k = 0; k < iK; k++){ret[i][j] = (ret[i][j] + a[i][k] * b[k][j]) % m_llMod;}}}return ret;}// 矩阵快速幂vector<vector<long long>> pow(const vector<vector<long long>>& a, vector<vector<long long>> b, long long n) {vector<vector<long long>> res = a;for (; n; n /= 2) {if (n % 2) {res = multiply(res, b);}b = multiply(b, b);}return res;}vector<vector<long long>> TotalRow(const vector<vector<long long>>& a){vector<vector<long long>> b(a.front().size(), vector<long long>(1, 1));return multiply(a, b);}
protected:const  long long m_llMod;
};class Solution {
public:int Ans(const long long N, vector<pair<int, int>>& not1) {vector<vector<long long>> pre(1), mat(6, vector<long long>(6, 4));pre[0].assign(6, 4);for (auto [x, y] : not1) {x--, y--;mat[x][(y + 3) % 6] = 0;mat[y][(x + 3) % 6] = 0;}CMatMul matMul;auto ans = matMul.pow(pre, mat, N - 1);auto iAns = accumulate(ans[0].begin(), ans[0].end(), 0LL);return iAns % ((int)1e9 + 7);}
};int main() {
#ifdef _DEBUGfreopen("a.in", "r", stdin);
#endif // DEBUG	long long N;int m;cin >> N >> m;auto not1 = Read<pair<int, int>>(m);auto res = Solution().Ans(N, not1);cout << res;
#ifdef _DEBUG		//printf("N=%lld,", N);//Out(a, "a=");//Out(b, ",b=");/*Out(edge, "edge=");Out(que, "que=");*/
#endif // DEBUG		return 0;
}
单元测试
long long N;vector<pair<int, int>> not;TEST_METHOD(TestMethod1){auto res = Solution().Ans(2, not);AssertEx(6*4*6*4, res);}TEST_METHOD(TestMethod2){not = { {1,2} };auto res = Solution().Ans(2,not );AssertEx(544, res);}TEST_METHOD(TestMethod3){not = { {1,2} };auto res = Solution().Ans(1, not );AssertEx(4*6, res);}
扩展阅读
| 我想对大家说的话 | 
|---|
| 工作中遇到的问题,可以按类别查阅鄙人的算法文章,请点击《算法与数据汇总》。 | 
| 学习算法:按章节学习《喜缺全书算法册》,大量的题目和测试用例,打包下载。重视操作 | 
| 有效学习:明确的目标 及时的反馈 拉伸区(难度合适) 专注 | 
| 闻缺陷则喜(喜缺)是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。 | 
| 子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。 | 
| 如果程序是一条龙,那算法就是他的是睛 | 
| 失败+反思=成功 成功+反思=成功 | 
视频课程
先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
 https://edu.csdn.net/course/detail/38771
 如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
 https://edu.csdn.net/lecturer/6176
测试环境
操作系统:win7 开发环境: VS2019 C++17
 或者 操作系统:win10 开发环境: VS2022 C++17
 如无特殊说明,本算法用**C++**实现。
