【题目链接】
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;
}