【蓝桥杯】产值调整

产值调整

题目描述

偏远的小镇上,三兄弟共同经营着一家小型矿业公司"兄弟矿业"。公司旗下有三座矿山:金矿、银矿和铜矿,它们的初始产值分别用非负整数 A A A、 B B B 和 C C C 表示。这些矿山的产出是小镇经济的核心,支撑着三兄弟和许多矿工家庭的生计。

然而,各矿山的产值波动剧烈,有时金矿收益高而银矿、铜矿低迷,有时则相反。这种不稳定性让公司收入难以预测,也常引发兄弟间的争执。为了稳定经营,三兄弟设计了一个公平的产值调整策略,每年执行一次,每次调整时,将根据当前的产值 A A A、 B B B、 C C C,计算新产值:

  1. 金矿新产值: A ′ = ⌊ B + C 2 ⌋ A'=\lfloor \dfrac{B+C}{2} \rfloor A′=⌊2B+C⌋;
  2. 银矿新产值: B ′ = ⌊ A + C 2 ⌋ B'=\lfloor \dfrac{A+C}{2} \rfloor B′=⌊2A+C⌋;
  3. 铜矿新产值: C ′ = ⌊ A + B 2 ⌋ C'=\lfloor \dfrac{A+B}{2} \rfloor C′=⌊2A+B⌋;

其中, ⌊ ⌋ \lfloor \rfloor ⌊⌋ 表示向下取整。例如, ⌊ 3.7 ⌋ = 3 \lfloor 3.7\rfloor = 3 ⌊3.7⌋=3, ⌊ 5.2 ⌋ = 5 \lfloor 5.2\rfloor = 5 ⌊5.2⌋=5。

计算出 A ′ A' A′、 B ′ B' B′、 C ′ C' C′ 后,同时更新: A A A 变为 A ′ A' A′, B B B 变为 B ′ B' B′, C C C 变为 C ′ C' C′,作为下一年调整的基础。

三兄弟认为这个方法能平衡产值波动,于是计划连续执行 K K K 次调整。现在,请你帮他们计算,经过 K K K 次调整后,金矿、银矿和铜矿的产值分别是多少。

输入格式

输入的第一行包含一个整数 T T T,表示测试用例的数量。

接下来的 T T T 行,每行包含四个整数 A , B , C , K A,B,C,K A,B,C,K,分别表示金矿、银矿和铜矿的初始产值,以及需要执行的调整次数。

输出格式

对于每个测试用例,输出一行,包含三个整数,表示经过 K K K 次调整后金矿、银矿和铜矿的产值,用空格分隔。

输入输出样例 #1
输入 #1
复制代码
2
10 20 30 1
5 5 5 3
输出 #1
复制代码
25 20 15
5 5 5
说明/提示
评测用例规模与约定
  • 对于 30 % 30\% 30% 的评测用例, 1 ≤ T ≤ 100 1 \leq T \leq 100 1≤T≤100, 1 ≤ A , B , C , K ≤ 1 0 5 1 \leq A, B, C, K \leq 10^5 1≤A,B,C,K≤105。
  • 对于 100 % 100\% 100% 的评测用例, 1 ≤ T ≤ 1 0 5 1 \leq T \leq 10^5 1≤T≤105, 1 ≤ A , B , C , K ≤ 1 0 9 1 \leq A, B, C, K \leq 10^9 1≤A,B,C,K≤109。

P12133 [蓝桥杯 2025 省 B] 产值调整

【思路分析】

做这个题之前观察一下数据范围,K是 1 0 9 10^9 109,T是 1 0 5 10^5 105 如何暴力迭代的话,会是 1 0 14 10^{14} 1014,一定会超时

暴力做法

java 复制代码
import java.io.*;
import java.util.*;
public class Main {
    public static void main(String[] agrs) throws Exception {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        PrintWriter pw = new PrintWriter(System.out);
        int n = Integer.parseInt(br.readLine());
        while(n-- > 0) {
            String[] data = br.readLine().split(" ");
            int a = Integer.parseInt(data[0]);
            int b = Integer.parseInt(data[1]);
            int c = Integer.parseInt(data[2]);
            int k = Integer.parseInt(data[3]);
            int aa, bb, cc;
            for(int i = 0; i < k; i++) {
                aa = (b + c) / 2;
                bb = (a + c) / 2;
                cc = (a + b) / 2;
                a = aa;
                b = bb;
                c = cc;
            }
            pw.print(a + " " + b + " " + c);
            pw.println();
        }
        pw.flush();
        pw.close();
        br.close();
    }
}

这种做法顶多过50%的样例,所以我们需要做出优化

一开始我做了一通数学推导,推导完之后我将时间复杂度降低到了O(1),本来还在沾沾自喜,但是一看运行结果,直接WA了。

我突然想到,我按照实数进行推导了,忘记向下取整这回事了。推导过程大概就是推导出a、b、c的迭代变化最总会是等差数列,而a,b,c的总和不变,因此可以直接通过公式求得最后的值。

最后我只能尝试打表,看一下每次a、b、c的变化,这一打表了不得,我发现经过有限次迭代后,最终都会收敛到a == b == c

输入

复制代码
1
150 140 30 50

输出

复制代码
85 90 145
117 115 87
101 102 116
109 108 101
104 105 108
106 106 104
105 105 106
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105
105 105 105

因此,我们只需要加上判断提前结束循环,这样的优化最终得以AC

java 复制代码
import java.io.*;
import java.util.*;
public class Main {
    public static void main(String[] agrs) throws Exception {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        PrintWriter pw = new PrintWriter(System.out);
        int n = Integer.parseInt(br.readLine());
        while(n-- > 0) {
            String[] data = br.readLine().split(" ");
            int a = Integer.parseInt(data[0]);
            int b = Integer.parseInt(data[1]);
            int c = Integer.parseInt(data[2]);
            int k = Integer.parseInt(data[3]);
            int aa, bb, cc;
            for(int i = 0; i < k; i++) {
                aa = (b + c) / 2;
                bb = (a + c) / 2;
                cc = (a + b) / 2;
                a = aa;
                b = bb;
                c = cc;
                if(a == b && a == c && b == c) {
                    break;
                }
            }
            pw.print(a + " " + b + " " + c);
            pw.println();
        }
        pw.flush();
        pw.close();
        br.close();
    }
}
相关推荐
@小码农2 天前
2025年全国青少年信息素养大赛 Gandi编程 小低组初赛真题
数据结构·人工智能·算法·蓝桥杯
@小码农2 天前
LMCC大模型认证 青少年组 第一轮模拟样题
数据结构·人工智能·算法·蓝桥杯
_OP_CHEN2 天前
【算法基础篇】(三十三)动态规划之区间 DP:从回文串到石子合并,吃透区间类问题的万能解法
c++·算法·蓝桥杯·动态规划·算法竞赛·acm/icpc·区间动态规划
_OP_CHEN3 天前
【算法基础篇】(三十二)动态规划之背包问题扩展:从多重到多维,解锁背包问题全场景
c++·算法·蓝桥杯·动态规划·背包问题·算法竞赛·acm/icpc
闻缺陷则喜何志丹3 天前
【图论 组合数学】P10912 [蓝桥杯 2024 国 B] 数星星|普及+
c++·数学·蓝桥杯·图论
_Voosk3 天前
C指针存储字符串为何不能修改内容
c语言·开发语言·汇编·c++·蓝桥杯·操作系统
闻缺陷则喜何志丹4 天前
【计算几何 二分查找】P12261 [蓝桥杯 2024 国 Java B] 激光炮|普及+
c++·数学·蓝桥杯·计算几何·洛谷
_OP_CHEN4 天前
【算法基础篇】(三十一)动态规划之基础背包问题:从 01背包到完全背包,带你吃透背包问题的核心逻辑
算法·蓝桥杯·动态规划·背包问题·01背包·完全背包·acm/icpc
闻缺陷则喜何志丹4 天前
【计算几何】P12144 [蓝桥杯 2025 省 A] 地雷阵|普及+
c++·数学·蓝桥杯·计算几何
迈巴赫车主4 天前
蓝桥杯 20531黑客java
java·开发语言·数据结构·算法·职场和发展·蓝桥杯