绝命沙虫 精度,double,模拟 牛客白月赛44

链接:https://ac.nowcoder.com/acm/contest/11221/C
来源:牛客网

时间限制:C/C++ 1秒,其他语言2秒
空间限制:C/C++ 262144K,其他语言524288K
64bit IO Format: %lld
题目描述
风暴来,黄沙起,绝命沙虫剧毒见。

初始时你的手上有 NN 元 RMB。

交易规则如下:

你通过充值 aa 元可以获得 a\times100a×100 红点和 \min{10000,a\times100\times(M-1)}min{10000,a×100×(M−1)} 绿点,其中 MM 是充值返点倍率。

你每次充值的数量必须是正整数,也就是说不能为零或负数。

你通过出售 bb 红点可以获得 \left\lfloor\dfrac{b}{200}\right\rfloor⌊
200
b

⌋ 元 RMB 和 \left\lfloor\dfrac{b}{10}\right\rfloor⌊
10
b

⌋ 点消费经验。

你通过出售 cc 绿点可以获得 \left\lfloor\dfrac{c}{10}\right\rfloor⌊
10
c

⌋ 点消费经验。

给定 N,MN,M,请回答你最后获得的消费经验是多少。

注意:任何时刻只要你手上有 RMB 或者有红点/绿点你都依次使用并且用光,你不会手上有而不用。
输入描述:
全文第一行输入一个整数 T(1\le T\le10^5)T(1≤T≤10
5
),表示数据组数。

第一行输入两个数 N(1\le N\le10^5),M(1.1\le M\le2.0)N(1≤N≤10
5
),M(1.1≤M≤2.0),分别表示初始拥有的 RMB 数量和充值返点倍率。

数据保证 NN 是正整数,MM 一定是一位小数。
输出描述:
每行输出一个整数,你最后获得的消费经验是多少。
示例1
输入
复制
2
10 1.5
160 2.0
输出
复制
270
5760
说明
对于样例 #1,充入 1010 RMB,获得 10\times100=100010×100=1000 红点和 10\times100\times(1.5-1)=50010×100×(1.5−1)=500 绿点。

花掉它们,获得 \dfrac{1000}{10}+\dfrac{500}{10}=150
10
1000

+
10
500

=150 消费经验和 \dfrac{1000}{200}=5
200
1000

=5 RMB。

充入 55 RMB,获得 500500 红点和 250250 绿点;花掉它们,获得 50+25=7550+25=75 消费经验和 22 RMB。

充入 22 RMB,获得 200200 红点和 100100 绿点;花掉它们,获得 20+10=3020+10=30 消费经验和 11 RMB。

充入 11 RMB,获得 100100 红点和 5050 绿点;花掉它们,获得 10+5=1510+5=15 消费经验。

结算(最终获得):150+75+30+15=270150+75+30+15=270 消费经验。

对于样例 #2,充入 160160 RMB,获得 1600016000 红点和 1000010000 绿点;转换成 8080 RMB 和 26002600 消费经验;

充入 8080 RMB,获得 80008000 红点和 80008000 绿点;转换成 4040 RMB 和 16001600 消费经验;

充入 4040 RMB,获得 40004000 红点和 40004000 绿点;转换成 2020 RMB 和 800800 消费经验;

\cdots⋯

充入 11 RMB,获得 100100 红点和 100100 绿点;转换成 2020 消费经验。

结算(最终获得):2600+1600+800+400+200+100+40+20=57602600+1600+800+400+200+100+40+20=5760 消费经验。

思路 :

  • 直接模拟即可
  • 关于double的精度问题 :double类型在计算时转化为二进制故有损失,例如 double a = 3.0 - 2.6,结果是0.39999999,这就是精度损失
  • 这里模拟的公式中由于m是double类型,因此m - 1也是double类型,会有精度损失
  • 解决方案有两种 :1.依靠乘10(这里保证m只有一位小数)化为整型;2.加一个特别小的数然后向下取整(强转int),注意强转int是在最后整个公式的结果处,而不是double运算处
#include <iostream>
using namespace std;

typedef long long ll;

int main()
{
    int _; cin >> _;
    while (_ -- )
    {
        int n;
        double m;
        cin >> n >> m;
        
        ll r = 0, g = 0;
        ll res = 0;
        
        while (n)
        {
            r = n * 100, g = min(10000, (int)(n * 100 * (m + 0.00000000000001 - 1)));
            res += r / 10 + g / 10;
            n = r / 200;
        }
        cout << res << endl;
    }
}

#include <iostream>
using namespace std;

typedef long long ll;

int main()
{
    int _; cin >> _;
    while (_ -- )
    {
        int n;
        double m;
        cin >> n >> m;
        
        ll r = 0, g = 0;
        ll res = 0;
        
        while (n)
        {
            r = n * 100, g = min(10000, (int)(n * 10 * (10 * m - 10)));
            res += r / 10 + g / 10;
            n = r / 200;
        }
        cout << res << endl;
    }
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值