C#

사칙연산 문제 생성

안녕1999 2020. 11. 7. 23:50
		void DEBUG_rnd_4_CAL()//사칙연산 문제 생성
		{
			Random rnd = new Random();
			string[] op= { "+", "-", "/", "*" };
			string s;
			int n=50,m;
			while (n > 0)
			{
				s = "";
				m= rnd.Next(3)+3;
				s = C.Ftoa2(rnd.Next(100) / (rnd.Next(10) + 1.0f)     - rnd.Next(50));
				while (m > 0)
				{
					s= s + " " + op[rnd.Next(op.Length)] +" "+ C.Ftoa2(rnd.Next(100) / (rnd.Next(10) + 1.0f) - rnd.Next(50));
					m--;
				}
				Console.WriteLine(s+",");
				n--;
			}
		}
		void DEBUG_test_Str_cal_with_brace()
		{
			string[] a = {
				"1 + 2 * (3 + 4)",
				"1 + 2 * 3",
				"111+1*2",
				//"48/2*(9+3)",
				"(2 + 3) * 4",
				"2 + (3 * 4)",
				"(6 / 2) * 3",
				"6 / (2 * 3)",
				"2 + 3 * 4",
				"6 / 2 * 3",
				"5 + 10 + 10",
				"7 / 3 - 9",
				//"7 / 3 - 9",
				"(123*456)+789 /123*1245",
				//------------------------
				"-5.14 * -7.5 * 56 + -29.4 / -24",
				"-23.5 + -2 * -6.43 * -23.2",
				"-38.29 + -3.56 / -4.5 * -37.33 + -8 + -34",
				"-32.78 * 22 - 18 / -20.75 - -22.33",
				"-29.67 + 54 + -31.17 + 20.33 * -35.67 - 15",
				"18.25 + -38.8 - -2.22 * -5.71 / 2.67 + -20",
				"-7.33 + -11.5 + -8.8 / -3.33 + -4.8 + -40.25",
				"-27 - -13.1 - -20.67 - -2.25 - 19",
				"-27.33 * -9.5 + -21.14 * -36.13",
				"-8.67 + -38.13 * -13.29 + -18.1 + 4.75 / -5.89",
				"-6.25 * 17 * 17.67 / -20.88 / -5.1 / -21",
				"-9.57 * -11.75 - -11.5 + -7.6",
				"3.4 * 0.57 * -40.5 - 4.14 - -24.25 * -6.75",
				"3.5 - -10.67 * -22.5 + -14 / -32.2",
				"-20.5 * -7 * -45 + -9.33",
				"-15.67 / -33 + -5.3 / -12",
				"45 / -4 + -6.75 - 8.83",
				"-25.67 / -11 - 4.11 / 41",
				"-6.63 + 0.71 - -28.67 / -17 - 1.5 * -7.75",
				"4.33 * -14.25 * -5.43 * -8.7 + -28",
				"-36.33 * -25.57 / -38.6 - -5.17 - 0.33 / 7.14",
				"-44.75 * -32.22 / -42.4 + -13 - -29.14 / -4.7",
				"-19.9 / -39.9 + -14.5 / -30.29 - 2 + 21",
				"-36.11 - -34.67 + 28 * -6.5",
				"-31.44 - 11 + -6.67 / 30.5",
				"-1 - -12 * -2.17 / 3 * -15.78",
				"-27.75 / -18.63 - -9.25 - -34 * -5 + -6.4",
				"5.5 + -36.4 + -0.83 * -7.33 - -33 * -42.89",
				"50 / 3.4 - -27.88 * -17.5 / -18",
				"35 - -32.22 / -28.5 + 2.5",
				"-3.33 + -29.33 * -1 * -23.4 + -42.25",
				"-37 * -0.14 / -31.4 / -5 + -25.6 / -19.33",
				"-16.6 + -29.33 - -19 + -19.83",
				"-29.5 - -19.22 * 6.5 * 5 + -7.38 / -22",
				"26.5 + -26.67 - 45.5 - 63 / -16.57",
				"3 + 4.56 / -22.75 / -40.33 / 1.4",
				"6 / 13 * -14 / 5.75 - -13.89",
				"1.5 + 1.75 / -34.56 * 2.9 - 1.44",
				"22.5 - -0.1 * -1.6 + -14.43 + 9.14",
				"-24.17 - -12 / -8.56 / -38.25 + -37.63 / -21.4",
				"-33.5 / -12.89 / -16 + -6.75 * -11.2 + -6.25",
				"15.75 + -3.75 - 65 - -39.14",
				"-38.88 / -17.78 - 2 * -0.75 - -41",
				"10 - 3.17 - -18.78 + 12 - -9",
				"-26.57 + -11.88 / -26 * -20.43 + -42.8 - 12",
				"29 * 2 * -15.5 + 4.17 * -36.4",
				"3.4 * -14.71 + -0.5 / -8 - -0.6 / 7",
				"-38.8 - -1.8 + -20.75 * -5.75 * 2.67",
				"61 - -11.5 * -14.25 * -35",
				"-20 * 0.2 + -19 + -23 / -6.25 * -14",
				//--------------------------------
				"-5.14 * (-7.5 * 56 + -29.4) / -24",
				"-23.5 + (-2 * -6.43) * -23.2",
				"(-38.29 + -3.56 / -4.5) * -37.33 + -8 + -34",
				"-32.78 * 22 - 18 / (-20.75 - -22.33)",
				"-29.67 + (54 + -31.17 + 20.33) * -35.67 - 15",
				"(18.25 + -38.8 - -2.22 * -5.71 / 2.67 + -20)",
				"(-7.33 + -11.5) + -8.8 / -3.33 + -4.8 + -40.25",
				"(-27 - -13.1 - -20.67) - -2.25 - 19",
				"-27.33 * (-9.5 + -21.14) * -36.13",
				"-8.67 + -38.13 * -13.29 + (-18.1 + 4.75 / -5.89)",
				"-6.25 * 17 * (17.67 / -20.88 / -5.1) / -21",
				"-9.57 * (-11.75 - -11.5) + -7.6",
				//--------------------------------
				"-5.14 * (-7.5 * (56 + -29.4)) / -24",
				"(-23.5 + (-2 * -6.43)) * -23.2",
				"((-38.29 + -3.56) / -4.5) * -37.33 + -8 + -34",
				"-32.78 * ((22 - 18) / (-20.75 - -22.33))",
				"((-29.67 + (54 + -31.17 + 20.33)) * -35.67) - 15",
				"(18.25 + (-38.8 - (-2.22 * (-5.71 / (2.67 + -20)))))",
				"(-7.33 + -11.5) + (-8.8 / -3.33) + (-4.8 + -40.25)",
				"((-27 - -13.1 - -20.67) - -2.25 - 19)",
				"(-27.33 * (-9.5 + -21.14) * -36.13)",
				"-8.67 + -38.13 * (-13.29 + (-18.1 + 4.75 / -5.89))",
				"((-6.25 * 17 )* (17.67 / -20.88 / -5.1)) / -21",
				"-9.57 * ((-11.75 - -11.5) + -7.6)"
			};
			double[] r =
			{
				1 + 2 * (3 + 4),
				1 + 2 * 3,
				111+1*2,
				//48/2.0D*(9+3),//나눗셈은 실수로 지정해주어야한다. double은 D
				(2 + 3) * 4,
				2 + (3 * 4),
				(6 / 2.0D) * 3,
				6.0D / (2 * 3),
				2 + 3 * 4,
				6 / 2.0D * 3,
				5 + 10 + 10,
				7.0D / 3D - 9.0D,
				//7 / 3 - 9,//정수를 넣어주면, 정수형으로 계산된다.
				(123*456)+789 /123.0D*1245,
				//------------------------
				-5.14D * -7.5D * 56D + -29.4D / -24D,
				-23.5D + -2 * -6.43D * -23.2,
				-38.29D + -3.56D / -4.5D * -37.33 + -8 + -34,
				-32.78D * 22 - 18D / -20.75D - -22.33,
				-29.67D + 54 + -31.17D + 20.33 * -35.67D - 15,
				18.25D + -38.8 - -2.22 * -5.71 / 2.67D + -20,
				-7.33D + -11.5 + -8.8 / -3.33 + -4.8 + -40.25,
				-27D - -13.1 - -20.67 - -2.25 - 19,
				-27.33D * -9.5 + -21.14 * -36.13,
				-8.67D + -38.13 * -13.29 + -18.1 + 4.75D / -5.89D,
				-6.25D * 17 * 17.67D / -20.88D / -5.1D / -21D,
				-9.57D * -11.75 - -11.5 + -7.6D,
				3.4D * 0.57 * -40.5 - 4.14 - -24.25 * -6.75D,
				3.5D - -10.67 * -22.5 + -14 / -32.2D,
				-20.5D * -7 * -45 + -9.33,
				-15.67D / -33 + -5.3 / -12D,
				45 / -4D + -6.75 - 8.83,
				-25.67 / -11D - 4.11 / 41,
				-6.63 + 0.71 - -28.67 / -17D - 1.5 * -7.75,
				4.33 * -14.25 * -5.43 * -8.7 + -28,
				-36.33 * -25.57 / -38.6 - -5.17 - 0.33 / 7.14,
				-44.75 * -32.22 / -42.4 + -13 - -29.14 / -4.7,
				-19.9 / -39.9 + -14.5 / -30.29 - 2 + 21,
				-36.11 - -34.67 + 28 * -6.5,
				-31.44 - 11 + -6.67 / 30.5,
				-1 - -12 * -2.17 / 3 * -15.78,
				-27.75 / -18.63 - -9.25 - -34 * -5 + -6.4,
				5.5 + -36.4 + -0.83 * -7.33 - -33 * -42.89,
				50 / 3.4 - -27.88 * -17.5 / -18,
				35 - -32.22 / -28.5 + 2.5,
				-3.33 + -29.33 * -1 * -23.4 + -42.25,
				-37 * -0.14 / -31.4 / -5 + -25.6 / -19.33,
				-16.6 + -29.33 - -19 + -19.83,
				-29.5 - -19.22 * 6.5 * 5 + -7.38 / -22,
				26.5 + -26.67 - 45.5 - 63 / -16.57,
				3 + 4.56 / -22.75 / -40.33 / 1.4,
				6 / 13D * -14 / 5.75D - -13.89,
				1.5D + 1.75D / -34.56D * 2.9 - 1.44D,
				22.5 - -0.1 * -1.6 + -14.43 + 9.14,
				-24.17 - -12 / -8.56D / -38.25D + -37.63 / -21.4D,
				-33.5 / -12.89D / -16D + -6.75 * -11.2 + -6.25,
				15.75 + -3.75 - 65 - -39.14,
				-38.88 / -17.78D - 2 * -0.75 - -41,
				10 - 3.17 - -18.78 + 12 - -9,
				-26.57 + -11.88 / -26D * -20.43 + -42.8 - 12,
				29 * 2 * -15.5 + 4.17 * -36.4,
				3.4 * -14.71 + -0.5 / -8D - -0.6 / 7D,
				-38.8 - -1.8 + -20.75 * -5.75 * 2.67,
				61 - -11.5 * -14.25 * -35,
				-20 * 0.2 + -19 + -23 / -6.25D * -14,
				//--------------------------------
				-5.14D * (-7.5D * 56D + -29.4D) / -24D,
				-23.5D + (-2 * -6.43) * -23.2,
				(-38.29D + -3.56 / -4.5D) * -37.33 + -8 + -34,
				-32.78D * 22 - 18 / (-20.75D - -22.33),
				-29.67D + (54 + -31.17 + 20.33) * -35.67 - 15,
				(18.25D + -38.8 - -2.22 * -5.71 / 2.67D + -20),
				(-7.33D + -11.5) + -8.8 / -3.33D + -4.8 + -40.25,
				(-27D - -13.1 - -20.67) - -2.25 - 19,
				-27.33D * (-9.5 + -21.14) * -36.13,
				-8.67D + -38.13 * -13.29 + (-18.1 + 4.75 / -5.89D),
				-6.25D * 17 * (17.67 / -20.88D / -5.1D) / -21D,
				-9.57D * (-11.75 - -11.5) + -7.6,
				//--------------------------------
				-5.14D * (-7.5D * (56D + -29.4D)) / -24D,
				(-23.5D + (-2 * -6.43)) * -23.2,
				((-38.29D + -3.56) / -4.5D) * -37.33 + -8 + -34,
				-32.78D * ((22 - 18) / (-20.75D - -22.33)),
				((-29.67D + (54 + -31.17 + 20.33)) * -35.67) - 15,
				(18.25D + (-38.8 - (-2.22 * (-5.71 / (2.67D + -20))))),
				(-7.33D + -11.5) + (-8.8 / -3.33D) + (-4.8 + -40.25),
				((-27D - -13.1 - -20.67) - -2.25 - 19),
				(-27.33D * (-9.5 + -21.14) * -36.13),
				-8.67D + -38.13 * (-13.29 + (-18.1 + 4.75 / -5.89D)),
				((-6.25D * 17 )* (17.67 / -20.88D / -5.1D)) / -21D,
				-9.57D * ((-11.75 - -11.5) + -7.6)
			};
			double d;
			int i,err_cnt=0, ok_cnt=0;
			//DEBUG_rnd_4_CAL();//사칙연산 문제 생성
			for (i = 0; i < a.Length; i++)
			{
				Console.WriteLine(a[i]);
				d = Str_cal_with_brace(a[i]);

				float f1, f2;
				f1 = ((float)d);
				f2 = ((float)r[i]);
				if (d != r[i])
				//if (((float)d) != ((float)r[i]))
				//if (f1 != f2)
				{
					//err
					err_cnt++;
					Console.WriteLine(a[i]);
					Console.WriteLine("ERR : " + a[i]+"="+C.Ftoa(r[i])+","+C.Ftoa(d));

					double d2 = 7 / 3.0f - 9;//-6.6666669845581055

				}
				else
				{
					Console.WriteLine("OK : " + a[i] + "=" + C.Ftoa(r[i]) + "," + C.Ftoa(d));
					ok_cnt++;
				}
			}
			Console.WriteLine("Total_cnt=" + a.Length.ToString());
			Console.WriteLine("ok_cnt="+ ok_cnt.ToString());
			Console.WriteLine("err_cnt=" + err_cnt.ToString());
		}
        
        
        
        
        
        
        int Find_and_Cal(double[]v,char[]op,int cnt,int i)//다음 숫자와 연산자 찾아서 계산
		{
			string msg = "";
			int find = -1, next;
			DEBUG_view_double_char(v, op, cnt);
			//다음 숫자와 연산자 찾기
			for (next = i + 1; next < cnt; next++)
			{
				//마지막 숫자는 연산자가 없다. 삭제된것이 아니면 ok
				if (op[next] !='D')//D=Delete
				{
					//find
					find = next;
					if (op[i] == '*')
					{
						v[i] = v[i] * v[next];
					}
					else if (op[i] == '/')
					{
						v[i] = v[i] / v[next];
					}
					else if (op[i] == '%')
					{
						v[i] = v[i] % v[next];
					}
					else if (op[i] == '+')
					{
						v[i] = v[i] + v[next];
					}
					else if (op[i] == '-')
					{
						v[i] = v[i] - v[next];
					}
					else
					{
						msg = "ERR : 알 수 없는 연산자";
					}
					//next 삭제
					op[i] = op[next];//다음연산자로 변경

					v[next] = 0;
					op[next] = 'D';//D=Delete

					DEBUG_view_double_char(v, op, cnt);

					i--;//다시한번 진행
					break;
				}
				else
				{
					//비어있는 슬롯.skip
				}
			}
			DEBUG_view_double_char(v, op, cnt);
			/*if(find==-1)
			{
				msg = "다음숫자를 못찾았습니다. or 계산 완료";
			}
			else { }*/
			return find;
		}
        bool HasCal(string s)//계산 수식이 있나?
		{
			bool ret = false;
			if (s != "")
			{
				ret = (s.IndexOf('+') >= 0) || (s.IndexOf('-') >= 0) ||
					(s.IndexOf('*') >= 0) || (s.IndexOf('/') >= 0) ||
					(s.IndexOf('%') >= 0);
			}
			else { }
			return ret;
		}
		double Str_cal_with_brace(string s)//괄호없는 문자열 사칙연산 문자열 수식 계산. s="1+2*3+4"
		{
			double ret = 0;
			if (s != "")
			{
				//수식이 있나?
				if (HasCal(s))
				{
					//수식(+-*/%) 있음
					bool bStart = true;
					int cnt = 0, i = 0, j, n = s.Length / 2 + 1, next, brace_cnt;
					double[] v = new double[n];
					char[] op = new char[n];
					string buf = "", msg = "", buf2 = "";
					v[0] = 0;
					s = s.Replace(" ", "");
					s = s.Replace("\t", "");
					s = s.Replace("\r", "");
					s = s.Replace("\n", "");
					while (i < s.Length)
					{
						if (s[i] == '(')
						{
							//괄호 처리. (앞에는 연산자
							i = C.Brace_cpy(s, i , out buf2);//'('을 발견하면 짝이 맞는 ')'이전까지만 복사한다.
							v[cnt] = Str_cal_with_brace(buf2);
							//다음 연산자 찾기
							//공백제거
							//i = C.Skip_space(s, i);
							if (i >= s.Length)
							{
								//마지막
								op[cnt] = ' ';
							}
							else
							{
								if ("+-/%*".IndexOf(s[i]) >= 0)
								{
									//연산자
									op[cnt] = s[i];
									i++;
								}
								else
								{
									//err
									msg = "ERR : +-/%* 연산자가 나올 차례입니다.";
									op[cnt] = '*';//곱셈연산자로 지정
								}
								bStart = true;
							}
							cnt++;//부호는 아직...
							DEBUG_view_double_char(v, op, cnt);
							buf = "";
						}
						else
						{
							//괄호 없음

							//"3-1"은? 첫번째 연산자, 부호 없음
							//"3--1"은? 첫번째 연산자, 부호 -
							//"-3-1"은? 첫번째 부호
							//"3-+1"은?
							if (bStart)//수식의 시작(부호만 올 수 있다 연산자가 앞에 올 수 없다
							{
								//부호 또는 숫자
								buf = buf + s[i];
								i++;
								bStart = false;
							}
							else
							{
								//연산자인가?
								//첫번째가 연산자(op)이고, 2번째 이후는 부호
								j = "+-/%*".IndexOf(s[i]);
								if ((j >= 0) &&
									(i != 0))//부호
								{
									//연산자
									//연산자.지금까지 저장한 값을 push
									v[cnt] = C.Atod(buf);
									op[cnt] = s[i];
									cnt++;
									DEBUG_view_double_char(v, op, cnt);
									buf = "";
									bStart = true;
								}
								else
								{
									//숫자
									buf = buf + s[i];
									bStart = false;
								}
								i++;
							}
							//DEBUG_puts("buf=" + buf);
						}
					}
					//마지막 숫자 넣기
					if (buf != "")
					{
						//push
						v[cnt] = C.Atod(buf);
						op[cnt] = ' ';//마지막은 연산자가 없다
						cnt++;
					}
					else { }
					DEBUG_view_double_char(v, op, cnt);

					//곱셈,나눗셈,나머지 부터 계산
					for (i = 0; i < cnt; i++)
					{
						if ((op[i] == '*') || (op[i] == '/') || (op[i] == '%'))
						{
							Find_and_Cal(v, op, cnt, i);//다음 숫자와 연산자 찾아서 계산
							i = -1;//다시 계산
						}
						else { }
					}
					DEBUG_view_double_char(v, op, cnt);

					//덧셈, 뺄셈 계산
					for (i = 0; i < cnt; i++)
					{
						if ((op[i] == '+') || (op[i] == '-'))
						{
							Find_and_Cal(v, op, cnt, i);//다음 숫자와 연산자 찾아서 계산
							i = -1;//다시 계산
						}
						else { }
					}
					DEBUG_view_double_char(v, op, cnt);
					ret = v[0];
				}
				else
				{
					//수식없음
				}
			}
			else { }
			return ret;
		}