SRM 450 DIV 1 总结

今天的250p还是个比较简单的博弈题,500p居然是个坑爹的模拟题,细节太多啦,需要注意各种东西。


250p:

题意:给你n堆石头,编号0~n-1,Alice先手,每次每个人可以从最左边的堆里取任意个石头,要取第i堆的石头必须要保证0~i-1的石头全部已经取完,最后一个取石头的人胜。


解题思路:由于取石头要从左到右取,那就可以分为两种情况,第一种,某一堆石头只有一个,那就必须取一个,

第二种,某堆石头大于一个,当前决策的人就处于必胜态了,因为如果取完这堆石头的人赢的话,那当前决策的人就可以直接取完这堆石头,反之当前决策的人可以取一些石头使得石头只有一个,那第二个人就必须取完这堆石头了。

所以一旦某个人可以取超过一个的石头,那他就赢了。


500p:

题意: 给你n个工厂,k个专家,每工作一次你可以获得n*k的金钱,设定建一个工厂或者雇一个专家都是需要price的金钱,你可以在任意时候弄任意多的工厂和专家只要你有足够的钱,问你最少需要工作多少次才能赚取target的金钱。


解题思路:这题一开始看还以为有一定的数学关系,其实就是一个恶心的模拟题。每一次你的选择都有两种,一种是不买工厂和专家了,直接工作足够的次数使得你的金钱数达到target,另一种就是你要建一个工厂或者雇一个专家,而你应该是得工厂和专家个数尽量平均这样n*k才最大,所以写一个dfs递归搜一下就行了,要注意的是刚开始n*k可能会超过long long,所以要特判一下,这里用的方法是把k转换成二进制的数,n*k看做tot,然后tot来加上n*(k所有位的二进制数),一旦tot超过10^12,直接返回一次就可以了。具体见代码


250p:

class OrderedNim {
    public:
    string winner(vector <int> layout) ;


};

string ans;
string OrderedNim::winner(vector <int> lay) {
    int n = lay.size();
    if(n == 1) {
        ans = "Alice";
        return ans;
    }
    int cur = 0;
    for(int i = 0;i < n; i++) {
        if(lay[i] > 1) {
            if(cur==0) ans = "Alice";
            else    ans = "Bob";
            return ans;
        }
        cur = !cur;
    }
    cur = !cur;
    if(cur == 0)    ans = "Alice";
    else    ans = "Bob";
    return ans;
}

500p: 

class StrongEconomy {
    public:
    LL earn(LL n, LL k, LL price, LL target) ;

};

LL ans, t, p;

void swap(LL &a, LL &b) {
    LL t = a; a = b; b = t;
}

LL mul(LL a, LL b) {
    LL ans = 0;
    bool flag = 0;
    for(int i = 0;i < 45; i++) {
        if(b&(1LL<<i)) {
            if(flag)    return -1;
            ans = ans + a;
        }
        if(flag)    continue;
        a = a + a;
        if(a > 1000000000000LL) flag = 1;
    }
    return ans;
}

void dfs(LL step, LL n, LL k, LL money) {
    LL now = (t-money)/(n*k);
    if((t-money)%(n*k) != 0)    now++;
    if(step+now < ans)  ans = step+now;
   // printf("%I64d %I64d %I64d\n", step, n, k);
    if(n*k > t) return ;
    LL add = (p-money)/(n*k);
    if((p-money)%(n*k) != 0)    add++;
    step += add;
    money += add*(n*k);
    LL temp = money/p;
    money %= p;
    if(n > k)   swap(n, k);
    n += temp;
    if(n > k) {
        LL tot = n+k;
        n = k = tot/2;
        n += tot%2;
    }
    if(step < ans)
        dfs(step, n, k, money);

}

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值