首页 > 其他分享 >P1957 口算练习题

P1957 口算练习题

时间:2024-01-23 13:56:04浏览次数:24  
标签:练习题 P1957 运算 cin sprintf 读入 input line

1.题目介绍

口算练习题

题目描述

王老师正在教简单算术运算。细心的王老师收集了 \(i\) 道学生经常做错的口算题,并且想整理编写成一份练习。 编排这些题目是一件繁琐的事情,为此他想用计算机程序来提高工作效率。王老师希望尽量减少输入的工作量,比如 \(\texttt{5+8}\) 的算式最好只要输入 \(\texttt 5\) 和 \(\texttt 8\),输出的结果要尽量详细以方便后期排版的使用,比如对于上述输入进行处理后输出 \(\texttt{5+8=13}\) 以及该算式的总长度 \(6\)。王老师把这个光荣的任务交给你,请你帮他编程实现以上功能。

输入格式

第一行一个整数 \(i\)。

接着的 \(i\) 行为需要输入的算式,每行可能有三个数据或两个数据。

若该行为三个数据则第一个数据表示运算类型,\(\texttt a\) 表示加法运算,\(\texttt b\) 表示减法运算,\(\texttt c\) 表示乘法运算,接着的两个数据表示参加运算的运算数。

若该行为两个数据,则表示本题的运算类型与上一题的运算类型相同,而这两个数据为运算数。

输出格式

输出 \(2\times i\) 行。对于每个输入的算式,输出完整的运算式及结果,第二行输出该运算式的总长度。

样例 #1

样例输入 #1

4
a 64 46
275 125
c 11 99
b 46 64

样例输出 #1

64+46=110
9
275+125=400
11
11*99=1089
10
46-64=-18
9

提示

【数据规模与约定】

对于 \(50\%\) 的数据,输入的算式都有三个数据,第一个算式一定有三个数据。

对于所有数据,\(0<i\leq 50\),运算数为非负整数且小于 \(10000\)。

2.题解

2.1 使用字符数组

思路

1.这里开始读入的时候要注意,因为后面我们需要读入一整行,在i后面还有一个换行符,无论是用cin还是scanf这里读入时都需要清除换行符!!!
2.这里我们选择一次读入一行(因为未知到底是两个变量还是三个变量,无法用循环来表示),这里读入string是用getline,读入字符数组可用fgets和cin.getline均可
3.我们判断第一个是否为字符,若是则覆盖存储,并清空该位置上的字符,便于后续读入两个操作数;若不是,直接读入两个操作数即可。
4.进行具体的运算判断,用sprintf将数据写入字符数组中,便于后续输出和计算长度。

代码

#include <bits/stdc++.h>
using namespace std;
int main()
{
	int i;
	char ch;
//	cin >> i; //cin >> i 读取整数后,可能会留下输入缓冲区中的换行符
//	cin.ignore();  // 清除换行符

	scanf("%d\n", &i); 
	
	for(int t = 0; t < i; t++){
//		string input_line; 
		char input_line[20];
		
//		getline(cin, input_line); //1.这是对于string类型读一行的 

//		fgets(input_line, sizeof(input_line), stdin);
		cin.getline(input_line, 20);  //2.这两种方式都可以将一行数据读入到一个字符数组中 
		
//		cin >> input_line; //3.cin读入直到遇到第一个空白字符为止,不适合用来读入一行数据。
 
		int a, b, ans;
		/*这里比较巧妙,因为确定字符只占一个位子,要是有,我就覆盖存,并将原来他的位置变为空(这样后面使用sscanf都是统一读两个操作数);要是没有,无需操作*/ 
		if(input_line[0] == 'a' || input_line[0] == 'b' || input_line[0] == 'c')
		{
			ch = input_line[0];		 
			input_line[0] = ' ';
		} 
		
		sscanf(input_line, "%d %d", &a, &b); // 读入操作数 

		
		/* 判断具体运算类型,用sprintf函数将字符串写入字符数组中,方便后续输出和计算长度!!!*/ 
		char str[20];
		//我这里为何不需要清空字符数组呢?因为我这里的是局部变量,每次循环自动更新! 
//		memset(str,0,sizeof(str));//清空原有的字符串,防止长度判断错误(比如像上一个长度大于这个,就会留个小尾巴)  
		switch(ch){
			case 'a': {
				ans = a + b;
				sprintf(str, "%d+%d=%d", a ,b ,ans);
				break;
			}
			case 'b':{
				ans = a - b;
				sprintf(str, "%d-%d=%d", a ,b ,ans);
				break;
			}
			case 'c':{
				ans = a * b;
				sprintf(str, "%d*%d=%d", a ,b ,ans);
				break;
			}
		}
		cout << str << endl << strlen(str) << endl; //strlen专门用于获取字符串长度,而且不计算'\0' 
		
	}	
}

2.2 使用字符串

思路

1.这里后续每次读入不是读一行,而是先读一个,如果是字符,处理后再读入两个;如果不是字符,将读入的这个转换为int存入第一个操作数,再读第二个操作数
2.后续基本上一致,要注意这里的s不是局部变量,每次用完要清空的,不然测试长度会有问题

代码

#include <iostream> //cin & cout 用iostream
#include <cstring> //memset & strlen 用cstring
#include <cstdio> //sscanf & sprintf 用cstdio
using namespace std;
int main(){
	char a;//a用于存储运算符
	int n,c,d;//n存储个数不解释,cd存储两个数字
	char s[100],b[10];//s存储最终的字符串,b临时变量
	
	cin>>n; //读入总个数 
	
	for(int i=0;i<n;i++){
		cin>>b;//输入一串字符,有可能是运算符,也有可能是数字
		
		if(b[0]>='a' && b[0]<='z'){
			a=b[0];//如果是运算符就存入a,然后输入数字
			cin>>c>>d; //读入两个操作数 
		}else{
			sscanf(b,"%d",&c);//如果是数字就转换b为int存储到第一个数字
			cin>>d;//然后输入剩下的第二个数字
		}
		
		memset(s,0,sizeof(s));//清空原有的字符串,防止长度判断错误	
		if(a=='a')//用sprintf格式化
			sprintf(s,"%d+%d=%d",c,d,c+d);
		else if(a=='b')
			sprintf(s,"%d-%d=%d",c,d,c-d);
		else if(a=='c')
			sprintf(s,"%d*%d=%d",c,d,c*d);
		cout<<s<<endl<<strlen(s)<<endl;//输出字符串和字符串长度
	}
	return 0;
}

标签:练习题,P1957,运算,cin,sprintf,读入,input,line
From: https://www.cnblogs.com/trmbh12/p/17982294

相关文章

  • $20240119$ 练习题解
    \(20240119\)练习题解CF472D通过尝试我们容易发现,与点\(1\)最近的点一定是直接儿子。我们要是把它作为突破点,那就成功了一半了。假设点\(2\)与点\(1\)最近,又假设我们可以用函数\(F(x)\)来确定\(x\)点的子树形态。那我们会发现如果我们还有剩余的节点,那么剩余的节点......
  • 练习题1
    使用面向对象的思想,编写自定义描述狗的信息。设定属性包括:品种,年龄,心情,名字;方法包括:叫,跑。要求:1)设置属性的私有访问权限,通过公有的get,set方法实现对属性的访问2)限定心情只能有“心情好”和“心情不好”两种情况,如果无效输入进行提示,默认设置“心情好”。3)设置构造......
  • PA0:关于练习题
    练习11: 附加题:复制操作:i=0;while(i<argc){states[i]=argv[i];i++;}如果还要考虑安全性,那就在循环体里面增加判断:i=0;j=0;while(i<argc){states[i]=argv[i];i++;j++;if(j>=......
  • PA0:关于练习题
    网页浏览体验很差,希望下次不要再找广告满天飞的网站搭翻译博客。网页做的很好,以后别做了。  不使用stdio库。gcc在make时会提示存在implicitdeclaration(隐式声明)--------------------------------------------makefile基本指令解释:CFLAGS=-Wall-g clean:   rm......
  • 算法练习题-系列一
    目录柯里化实现柯里化函数柯里化函数作用扁平化[双指针]有序数组合并判断一个字符串是否是回文字符串[字符串]两个版本号version1和version2[字符串]版本号大小比较排序['1.45.0','1.5','6','3.3.3.3.3.3.3']=>['1.5','1.45.0','3.3.3.3.3.3','6']给定两个......
  • 队列练习题
    求m区间内的最小值(洛谷P1440)题目大意对一序列a,从左至右扫描,取每个位置前m个数的最小值,位置为首位置时输出0,不足m个数时就取这段范围内的最小值。解题思路使用单调队列,保持队头存最小元素下标,从队尾更新最值,超出窗口范围时队头出队。未知的代码#include<bits/stdc++.h>u......
  • 栈练习题
    单调栈(洛谷P5788)题目大意与栈中的向右看齐相同题解未知的代码#include<bits/stdc++.h>usingnamespacestd;constintN=3e6+5;inta[N],ans[N],n;stack<int>s;intmain(){cin>>n;for(inti=1;i<=n;i++)cin>>a[i];for(inti=n;i>=1;i--){......
  • Java多线程​(五)练习题7道
    练习多线程练习1(卖电影票)一共有1000张电影票,可以在两个窗口领取,假设每次领取的时间为3000毫秒,要求:请用多线程模拟卖票过程并打印剩余电影票的数量线程类实现:publicclassTicketWindowextendsThread{publicTicketWindow(){}publicTicketWindow(Stringname){super(nam......
  • 洛谷 P1957
    题目链接:在每一行,因为不确定第一个输入数据的类型,所以要用字符串输入。值得注意的是,\(\sfsprintf\)的函数原型为intsprintf(char*buffer,constchar*format[,argument]…);,其第一个参数是char*类型,因此在使用\(\sfsprintf\)时一般使用字符串数组charstr[]而不用\(\s......
  • c203数据库练习题下半
    2、视图练习(1)建立视图v_xs_1,要求包含男生的学号,姓名,性别,出生日期,班级编号,专业名称字段,并要求视图操作数据时进行检查。使用select命令查询创建的视图。createviewv_xs_1asselectxh,xm,xb,csrq,bjbh,zymcfromxsjbxxbwherexb='男'withcheckoption;建立一个学院教师......