解题思路

这是一道贪心题目。
关键思路:

  1. 由于每次操作只能除以2(向下取整),所以最终的相等值一定是数组中最小值经过若干次除以2得到的数
  2. 对于每个数,我们需要计算将其除以2多少次才能达到目标值
  3. 为了使操作次数最少,我们需要:
    • 枚举所有可能的目标值(从最小值开始,每次除以2)
    • 计算将所有数变成该目标值需要的操作次数
    • 取所有方案中的最小值

代码

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

// 计算将num变成target需要的操作次数
int countOps(long long num, long long target) {
    if(num < target) return 1e9;  // 不可能的情况
    int ops = 0;
    while(num > target) {
        num /= 2;
        ops++;
    }
    return num == target ? ops : 1e9;
}

int main() {
    int n;
    cin >> n;
    vector<long long> a(n);
    
    // 输入数组并找到最小值
    long long minVal = 1e18;
    for(int i = 0; i < n; i++) {
        cin >> a[i];
        minVal = min(minVal, a[i]);
    }
    
    long long ans = 1e18;
    // 枚举所有可能的目标值
    long long target = minVal;
    while(target > 0) {
        long long totalOps = 0;
        bool possible = true;
        
        // 计算将所有数变成target需要的操作次数
        for(int i = 0; i < n && possible; i++) {
            int ops = countOps(a[i], target);
            if(ops >= 1e9) possible = false;
            totalOps += ops;
        }
        
        if(possible) {
            ans = min(ans, totalOps);
        }
        target /= 2;
    }
    
    cout << ans << endl;
    return 0;
}
import java.util.*;

public class Main {
    // 计算将num变成target需要的操作次数
    static long countOps(long num, long target) {
        if(num < target) return (long)1e9;  // 不可能的情况
        int ops = 0;
        while(num > target) {
            num /= 2;
            ops++;
        }
        return num == target ? ops : (long)1e9;
    }
    
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        long[] a = new long[n];
        
        // 输入数组并找到最小值
        long minVal = Long.MAX_VALUE;
        for(int i = 0; i < n; i++) {
            a[i] = sc.nextLong();
            minVal = Math.min(minVal, a[i]);
        }
        
        long ans = Long.MAX_VALUE;
        // 枚举所有可能的目标值
        long target = minVal;
        while(target > 0) {
            long totalOps = 0;
            boolean possible = true;
            
            // 计算将所有数变成target需要的操作次数
            for(int i = 0; i < n && possible; i++) {
                long ops = countOps(a[i], target);
                if(ops >= 1e9) possible = false;
                totalOps += ops;
            }
            
            if(possible) {
                ans = Math.min(ans, totalOps);
            }
            target /= 2;
        }
        
        System.out.println(ans);
    }
}
def count_ops(num, target):
    # 计算将num变成target需要的操作次数
    if num < target:
        return float('inf')  # 不可能的情况
    ops = 0
    while num > target:
        num //= 2
        ops += 1
    return ops if num == target else float('inf')

n = int(input())
a = list(map(int, input().split()))

# 找到最小值
min_val = min(a)

ans = float('inf')
# 枚举所有可能的目标值
target = min_val
while target > 0:
    total_ops = 0
    possible = True
    
    # 计算将所有数变成target需要的操作次数
    for num in a:
        ops = count_ops(num, target)
        if ops == float('inf'):
            possible = False
            break
        total_ops += ops
    
    if possible:
        ans = min(ans, total_ops)
    target //= 2

print(ans)

算法及复杂度

  • 算法:贪心。枚举所有可能的目标值,计算最小操作次数。
  • 时间复杂度: ,其中n是数组长度,M是最小值。
  • 空间复杂度: ,需要存储输入数组。