P1005 [NOIP2007 提高组] 矩阵取数游戏

使用__int128类型实现矩阵动态规划算法
本文详细介绍了如何使用C++中的`__int128`类型处理大整数,并展示了如何运用动态规划方法解决一个矩阵问题,涉及函数如`qpow`、`read`和`write`。
#include<bits/stdc++.h>
using namespace std;
const int maxn=105;
__int128 dp[maxn][maxn][maxn],a[maxn][maxn];
int n,m;
__int128 qpow(__int128 x,__int128 y)
{
	__int128 res=1;
	while(y)
	{
		if(y&1) res=res*x;
		x=x*x;
		y>>=1;
	}
	return res;
}
__int128 read()
{
    __int128 x=0,f=1;
    char ch=getchar();
    while(ch<'0'||ch>'9')
    {
        if(ch=='-')
            f=-1;
        ch=getchar();
    }
    while(ch>='0' && ch<='9')
        x=x*10+ch-'0',ch=getchar();
    return x*f;
}
void write(__int128 x)
{
    if(x<0) putchar('-'),x=-x;
    if(x>9) write(x/10);
    putchar(x%10+'0');
    return;
}
int main()
{
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;++i)
		for(int j=1;j<=m;++j)
			a[i][j]=read();
	for(int i=1;i<=n;++i)
	{
		dp[i][0][m]=a[i][m]*2;
		dp[i][1][m+1]=a[i][1]*2;
	}
	for(int g=1;g<=n;++g)
	{
		for(int i=0;i<=m;++i)
		{
			for(int j=m+1;j>i;--j)
			{
				if(i>=1) dp[g][i][j]=max(dp[g][i][j],dp[g][i-1][j]+qpow(2,i+m-j+1)*a[g][i]);
				if(j<=m) dp[g][i][j]=max(dp[g][i][j],dp[g][i][j+1]+qpow(2,i+m-j+1)*a[g][j]);
			}
		}
	}
	__int128 ans=0;
	for(int i=1;i<=n;++i)
	{
		__int128 sum=0;
		for(int j=0;j<=m;++j)
		{
			sum=max(sum,dp[i][j][j+1]);
		}
		ans+=sum;
	}
	write(ans);
	return 0;	
}
# P1005 [NOIP 2007 提高] 矩阵游戏 ## 题目描述 帅帅经常跟同学玩一个矩阵游戏:对于一个给定的 $n \times m$ 的矩阵矩阵中的每个元素 $a_{i,j}$ 均为非负整游戏规则如下: 1. 每次时须从每行各走一个元素,共 $n$ 个。经过 $m$ 次后矩阵内所有元素; 2. 每次走的各个元素只能是该元素所在行的行首或行尾; 3. 每次都有一个得分值,为每行的得分之和,每行的得分 = 被走的元素值 $\times 2^i$,其中 $i$ 表示第 $i$ 次(从 $1$ 开始编号); 4. 游戏结束总得分为 $m$ 次得分之和。 帅帅想请你帮忙写一个程序,对于任意矩阵,可以求出后的最大得分。 ## 输入格式 输入文件包括 $n+1$ 行: 第一行为两个用空格隔开的整 $n$ 和 $m$。 第 $2\sim n+1$ 行为 $n \times m$ 矩阵,其中每行有 $m$ 个用单个空格隔开的非负整。 ## 输出格式 输出文件仅包含 $1$ 行,为一个整,即输入矩阵后的最大得分。 ## 输入输出样例 #1 ### 输入 #1 ``` 2 3 1 2 3 3 4 2 ``` ### 输出 #1 ``` 82 ``` ## 说明/提示 **【据范围】** 对于 $60\%$ 的据,满足 $1\le n,m\le 30$,答案不超过 $10^{16}$。 对于 $100\%$ 的据,满足 $1\le n,m\le 80$,$0\le a_{i,j}\le1000$。 **【题目来源】** NOIP 2007 提高第三题。 #include <bits/stdc++.h> using namespace std; int n , m; int num[85][85]; long long sc = 0; int main(){ cin >> n >> m; for(int i = 1;i <= n;i++){ for(int j = 1;j <= m;j++){ cin >> num[i][j]; if(num[i][j] <= num[i][j - 1]){ int k = 1; while(k < j){ if(num[i][j - k + 1] <= num[i][j - k]){ swap(num[i][j - k + 1] , num[i][j - k]); }else{ break; } k++; } } } int k = 1; for(int j = 1;j <= m;j++){ k *= 2; sc += k * num[i][j]; } } cout << sc; return 0; } 我的代码哪错了
07-10
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值