信息学奥赛一本通 1910:【00NOIP普及组】计算器的改良 | 洛谷 P1022 [NOIP2000 普及组] 计算器的改良

【题目链接】

ybt 1910:【00NOIP普及组】计算器的改良
洛谷 P1022 [NOIP2000 普及组] 计算器的改良

【题目考点】

1. 字符数组
2. 一元一次方程
3. 模拟

【解题思路】

例:2x + 1 = 3这个一元一次方程中,2是系数,x是未知数,1与3都是常数。

  • 由于题目指定这是一元一次方程,而且除了未知数和系数间是乘法,其余只有加减法。考虑的情况比较简单
  • 将未知数与系数的乘积都移动到等号左边,将常数都移动到等号右边,经过加和运算后,一定可以形成ax = b的形式,其中a是系数,b是常数,x是未知数。下一步运算:x = b/a,即可得到方程的解。
  • 在代码中模拟上述求方程的过程即可。遍历并解析公式字符串,遇到一个未知数项(一次项),就将其移动到等号左边,系数加和。遇到一个常数项,就将其移动到等号右边,常数加和。最后得到ax=b中的a和b,进而求出解。

特殊地,如果rn为0,ln为-1,输出rn/ln会得到-0.000,与要求的0.000不同,因此对rn为0时的情况做特殊处理。

【题解代码】

解法1:模拟
#include<bits/stdc++.h>
using namespace std;
#define N 1005
int main()
{
    char s[N], v;//s:公式字符串 v:未知数字母 
    cin >> s;
    int len = strlen(s), num = 0, sign = 1;//num:构造出的数字 sign:构造出的数字的符号 
    int ln = 0, rn = 0;//ln:等号左侧未知数的系数和 rn:等号右侧常数和 
    bool isLeft = true;//是否在遍历等号左边 
    for(int i = 0; i <= len; ++i)//遍历包括最后的'\0' 
    {
        if(s[i] >= '0' && s[i] <= '9')//如果是数字,则构造数字 
            num = num * 10 + s[i] - '0';
        else
        {
            if(s[i] >= 'a' && s[i] <= 'z')//如果是未知数 
            {
                v = s[i];//确定未知数字母 
                if(isLeft == false)//如果是右侧的项,则移动到左侧,系数符号改变 
                    sign = -sign;
                if(num == 0)
                	num = 1;
                ln += sign * num;//未知数系数增加sign*num 
            }
            else//如果是常数项 
            {//如果s[i]是'\0'且最后一个字符是字母,那么会运行到这一句,此时num为0,不影响rn的值 
                if(isLeft)//如果是左侧的项,移动到右侧后系数符号改变 
                    sign = -sign;
                rn += sign * num;//右侧常数增加sign*num 
            }
            sign = 1;//符号还原为正 
            num = 0;//构造数字变量归0 
            if(s[i] == '-')//下一个数字符号位负 
                sign = -1;
            else if(s[i] == '=')//经过等号,后面就是等号右边的式子了 
                isLeft = false;
        }
    }
    cout << v << '=';
	if(rn == 0)
		cout << fixed << setprecision(3) << 0.0;
	else 
    	cout << fixed << setprecision(3) << (double)rn / ln;
	return 0;
}
  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值