当前位置:首页 > 牛客 > 【牛客14777题解法】动态规划+递归优化:详解比赛得分调整问题

【牛客14777题解法】动态规划+递归优化:详解比赛得分调整问题

9个月前 (07-04)

【牛客14777题解法】动态规划+递归优化:详解比赛得分调整问题  动态规划 递归优化 状态压缩 第1张

一、题目解读

牛客14777题要求判断在给定比赛场次n、当前得分k及两队得分差d1、d2的情况下,是否可以通过调整剩余比赛得分使最终比分满足特定条件。题目核心在于处理动态变化的得分组合,并验证其可行性,需结合数学推导算法优化技巧。

二、解题思路

1. 边界条件处理:

    当剩余比赛次数k为0时,需检查总得分n是否为3的倍数(题目约束)。

    当n=k时,判断两队得分差是否均为0(比赛结束条件)。

2. 动态规划核心:

    计算总得分上限total=n*3,剩余需分配得分remaining=n-k。

    通过双重循环枚举4种比分组合(s1,s2∈{-1,1}),模拟两队得分变化。

3. 优化策略:

    利用数学性质筛除非法组合(如总分余数不为3)。

    计算调整后最高得分max_score,推导所需调整次数needed,结合剩余比赛次数验证可行性。

三、解题步骤

1. 输入与初始化:

    读取测试用例数t,循环处理每个案例。

2. 核心函数check(n,k,d1,d2):

    执行边界条件判断(见上文)。

    计算总分与剩余得分,枚举s1、s2组合。

    推导调整后的得分(a,b,c),验证非负性。

    计算max_score与调整需求needed,检查剩余次数是否满足条件。

3. 输出结果:根据check返回值输出"yes"或"no"。

四、代码与注释

#include <iostream>  
#include <cmath>  
using namespace std;  

// 核心判断函数  
bool check(long long n, long long k, long long d1, long long d2) {  
    // 特殊情况处理:比赛已结束  
    if (k == 0) return (n % 3 == 0);  
    if (n == k) return (d1 == 0 && d2 == 0);  
    
    long long total = n * 3; // 总分上限  
    long long remaining = n - k; // 剩余比赛次数  
    // 枚举4种可能的比分组合  
    for (int s1 : {-1, 1}) {  
        for (int s2 : {-1, 1}) {  
            long long sum = 2 * s1 * d1 + s2 * d2; // 当前组合得分  
            if ((k - sum) % 3!= 0) continue; // 跳过非法组合(总分非3的倍数)  
            
            long long a = (k - sum) / 3; // 调整后得分A  
            long long b = a + s1 * d1; // 得分B  
            long long c = b + s2 * d2; // 得分C  
            // 检查非负性  
            if (a < 0 || b < 0 || c < 0) continue;  
            
            // 计算需要调整的分数(使三者相等)  
            long long max_score = max(a, max(b, c));  
            long long needed = (max_score - a) + (max_score - b) + (max_score - c);  
            // 检查剩余比赛是否足够且得分合法  
            if (needed <= remaining && (remaining - needed) % 3 == 0) {  
                return true;  
            }  
        }  
    }  
    return false;  
}  

int main() {  
    ios::sync_with_stdio(false);  
    cin.tie(nullptr); // 加速IO操作  
    int t;  
    cin >> t;  
    while (t--) {  
        long long n, k, d1, d2;  
        cin >> n >> k >> d1 >> d2;  
        cout << (check(n, k, d1, d2)? "yes" : "no") << endl;  
    }  
    return 0;  
}

五、总结

本解法通过动态规划思想结合递归枚举,有效处理得分调整问题。关键在于:

1. 边界条件优化:减少无效计算;

2. 数学推导:利用模3性质筛除非法状态;

3. 状态压缩:通过s1、s2组合简化得分计算。

时间复杂度为O(n^2),适用于中等规模数据。建议在实际应用中结合记忆化搜索进一步优化。

原创内容 转载请注明出处

分享给朋友:

相关文章

牛客12576题解题全解析:动态规划+质因数分解实现跳跃问题最优解

牛客12576题解题全解析:动态规划+质因数分解实现跳跃问题最优解

一、题目解读牛客12576题是一道经典的算法题,要求给定起点N和终点M,求解从N到M的最少跳跃次数。题目考察的核心在于路径优化与动态规划思想,需结合数论中的质因数分解技巧,通过合理设计算法降低时间复杂...

NOIP 2008火柴棒等式题解(C++代码实现)  动态规划与枚举算法详解

NOIP 2008火柴棒等式题解(C++代码实现) 动态规划与枚举算法详解

一、题目解读火柴棒等式问题(NOIP 2008,洛谷P1149)要求使用给定数量的火柴棒,构造形如 A + B = C 的等式,其中A、B、C均为整数,且火柴棒总数恰好等于输入值。需统计符合条件的等式...

2024蓝桥杯省赛B组“传送阵”题解(C++代码+图论算法优化)

2024蓝桥杯省赛B组“传送阵”题解(C++代码+图论算法优化)

一、题目解读2024年蓝桥杯省B组“传送阵”题目要求处理一个包含n个节点的图,节点间存在单向传输关系。每个节点i可传送至a[i]指定的节点,形成可能存在的环结构。题目需求解从任意节点出发能到达的最长路...

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

一、题目解读LeetCode 120题“三角形最小路径和”要求给定一个由数字组成的三角形,从顶部开始向下移动,每次可向左或向右移动一格,计算从顶至底的最小路径和。三角形以二维向量形式给出,每层元素数量...

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

一、题目解读洛谷P4551题要求在一个无向图中,寻找任意两点路径权值异或后的最大值。题目输入为图的边信息(点数n和n-1条边),每条边包含起点、终点及权值。需输出所有路径中权值异或的最大值。问题核心在...

【蓝桥杯国赛A组】冰山体积计算:动态规划与map统计的解题方案(洛谷P8767)

【蓝桥杯国赛A组】冰山体积计算:动态规划与map统计的解题方案(洛谷P8767)

一、题目解读本题为2021年蓝桥杯国赛A组题目“冰山”(洛谷P8767),要求处理冰山在融化与新生成过程中的体积变化。每日存在两种操作:冰山体积按固定值x融化(体积不足x的部分视为完全融化),以及新增...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。