G. 小红的数轴移动(二)

小红的数轴移动(二)

https://ac.nowcoder.com/acm/contest/91177/G

Description

小红站在数轴的x点上,她准备进行n次操作,每次操作如下: 若小红站在原点,则原地不动。否则向原点的方向,移动 距离。 小红可以自己选择操作的先后次序。她希望给出一个方案,使得最终移动的总距离最小,你能帮帮她吗?

Solution

排列问题并不好做,我们先做如下观察:

observation1:该问题等价于对序列分配权值 求移动总距离最小。

Brief Proof

证明二者等价性,我们尝试用转化后的问题构造出原问题的一个解。

将操作分为正向和反向两种移动方式,。 如果此时位置,我们使用操作直到,反之亦然。

因为两种操作位移和一定是。若想到达原点至少将两个集合操作用完,同时又只有在原点才会出现无法用尽的情况,所以一定能构造一个解。

由observation1就变成了一个简单的组合优化问题,可以很轻松的用DP解决。

tip1: 写的过程中很容易把问题歪到小红一定会走到原点的情况,如果获得200分考虑是否漏考虑了小红用完所有操作也走不到原点的情况。

Code

#include <bits/stdc++.h>
using namespace std;

int main() {
    ios::sync_with_stdio(false);
    cin.tie(0);

    int n, x;
    cin >> n >> x;
    vector<int> a(n);
    for (auto& it : a) cin >> it;
    int sa = accumulate(a.begin(), a.end(), 0);
    vector<vector<int>> f(n + 1, vector<int>(sa * 2 + 1, INT_MAX / 2));
    f[0][x + sa] = 0;
    for (int i = 0; i < n; i++) {
        for (int j = -sa; j <= sa; j++) {
            f[i + 1][j + sa] = min(f[i + 1][j + sa], f[i][j + sa]);
            if (j - a[i] + sa >= 0)
                f[i + 1][j + sa] = min(f[i + 1][j + sa], a[i] + f[i][j - a[i] + sa]);
            if (j + a[i] + sa <= 2 * sa)
                f[i + 1][j + sa] = min(f[i + 1][j + sa], a[i] + f[i][j + a[i] + sa]);
        }
    }
    if (f[n][sa] > 1e9) {
        cout << sa << "\n";
        for (int i = 0; i < n; i++) cout << i + 1 << " ";
        return 0;
    }
    cout << f[n][sa] << "\n";
    vector<int> pos, positive, negtive;
    pos.emplace_back(0);
    for (int i = n - 1; i >= 0; i--) {
        int now = pos.back();
        if (now - a[i] + sa >= 0 && f[i + 1][now + sa] == a[i] + f[i][now - a[i] + sa]) {
            pos.emplace_back(now - a[i]);
            positive.emplace_back(i);
        } else if (now + a[i] + sa <= 2 * sa && f[i + 1][now + sa] == a[i] + f[i][now + a[i] + sa]) {
            pos.emplace_back(now + a[i]);
            negtive.emplace_back(i);
        }
    }
    // assert(pos.back() == x);
    vector<int> opseq, usd(n);
    int now = x;
    while (now != 0) {
        // cout<<"from "<<now<<" to ";
        if (now > 0) {
            int t = negtive.back();
            negtive.pop_back();
            usd[t] = 1;
            opseq.emplace_back(t);
            now -= a[t];
        } else if (now < 0) {
            int t = positive.back();
            positive.pop_back();
            usd[t] = 1;
            opseq.emplace_back(t);
            now += a[t];
        }
        // cout << now << "\n";
    }
    // reverse(opseq.begin(), opseq.end());
    for (int i = 0; i < n; i++)
        if (!usd[i]) opseq.emplace_back(i);
    for (auto it : opseq) cout << it + 1 << " ";   
    return 0;
}

Appendix

00

全部评论

相关推荐

2025-12-02 02:15
门头沟学院
最近菊厂陆续开了,极力劝退那些拿13级的985硕士,就13级那么点儿薪资,一线城市每个月到手1.8/7/6w,租房2k还是破烂,吃饭2k还是预制菜,买个1k衣服都是聚酯纤维破塑料,稍微出去浪一浪,能留1w就是万岁,要是再有个啥都想买的对象,一线工作一年难存10w。隔壁工地混泥土,钳工,焊工一天800+,还包吃包住。读书18年到985硕士出来就为了进厂螺丝工?还不如从8岁童工开始干活,别人读书完了你工龄18+,混不上领导也是个小头头了。当然专科进来正式工,od都行,一般本科进来13级也OK,毕竟22岁年纪摆在那个地方还不需要太花钱。读硕博的基本26岁,工作两年就要结婚的,兜里没几个崽,连彩礼都要信用贷。菊厂离职的不少,毕竟正常没人受得了9116(梗:再来一次911刷6)。为啥这时候劝?因为刚下班,因为国考刚完,省考下周,就是可惜选调只有当年应届能报。现在回想能拍断大腿。应届生真实好身份,错过这一次,选调,考公,考编,当老师,进医院,研究所,高校,央国企,基本都无缘了,就连报名资格都被剥夺了,可谓是被党和国家遗弃的废材,统称“社会上的”,扔到社会去流浪,被用坏了就扔医院,长期超负载使用,零件修不好基本可以扔火里回炉重造了。体制内奉行找体制内的,都是党和国家选的人才,智力不差,样貌不丑,身材端正,收入稳定,安居房政策福利待遇也OK。因公出行都是报销,周末顺带“游山玩水“,这种体制内单身资源但凡想找对象,去社会上随便吆喝一声都排队。观察一下,基本没什么公务员在相亲,因为早就被邻里邻居抢光了。
哈哈哈,你是老六:就这不去的人大把人干呢,现在不缺人干活,你不干大把干呢,还有那个说农民工赚钱的,那个800+我估计肯定也就那一段时间,哪有这么赚钱,还是一句话,要想存下钱必须花销极低,能省的就不花钱,工资要高点
点赞 评论 收藏
分享
2025-11-23 15:14
中原工学院 Java
程序员花海_:实习和校招简历正确格式应该是教育背景+实习+项目经历+个人评价 其中项目经历注意要体现业务 实习经历里面的业务更是要自圆其说 简历模板尽可能保持干净整洁 不要太花哨的
点赞 评论 收藏
分享
评论
5
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务