求字符串表达式的值.

 

求字符串表达式的值,这里主要使用的知识是栈.为了更好的使用,这里定义一个栈头文件,其中包括基本的栈操作.

#ifndef STACK_H
#define STACK_H

typedef char datatype;
typedef struct
{
    datatype *arr;
    int top;
    int size;
}stack;

void init_stack(stack *s, int n = 100)/*栈的初始化*/
{
    s->size = n;
    if (!(s->arr = (datatype*)malloc(n * sizeof(datatype))))
    {
        printf("Malloc error!\n");
    }    
    s->top = 0;
}

void destroy_stack(stack *s)/*栈的销毁*/
{
    delete s->arr;
    s->arr = NULL;
}

void push_stack(stack *s, datatype val)/*入栈操作*/
{
    if (s->top >= s->size)
    {
        printf("The stack is full!\n");
    }
    else
    {
        s->arr[s->top++] = val;
    }
}

int is_empty(stack *s)/*查看栈是否为空*/
{
    return (s->top == 0) ? 1 : 0;
}

datatype pop_stack(stack *s)/*出栈操作*/
{
    if (s->top == 0)
    {
        return 0;
        printf("The stack is empty!");
    }
    else
    {
        return s->arr[--(s->top)];
    }
}

datatype get_top(stack *s)/*取得栈顶元素*/
{
    if (s->top == 0)
    {
        return 0;
        printf("The stack is empty!");
    }
    else
    {
        return s->arr[s->top-1];
    }
}

#endif


求表达式的值,首先将表达式转换成后缀表达式, 然后再求后缀表达式的值.(这里字符串表达式的格式未能完全检测是否正确)

/*
  Author: Mcdragon
  Date: 26-10-11 09:23
  Description: 计算字符串表达式的值(支持括号与小数). 
*/
#include <stdio.h>
#include <malloc.h>
#include "stack.h"
#define STR_LEN 100
int braket_match(char *str, stack *s); /*检测一个表达式的括号是否匹配:*/
double calulate(char *str); /*求出后缀表达式的值*/
void change_expr(char *input, char *out, int len); /*将中缀表达式转换成后缀表达式*/
int is_operator(char ch); /*判断是否是运算符*/
int is_digit(char ch);  /*判断是否是数字*/
int priority(char ch); /*返回运算符的优先级*/

int main()
{
    stack s;
    char str[STR_LEN], post_str[STR_LEN];
    init_stack(&s,STR_LEN);
    printf("Please input an express!\n");
    scanf("%s",str);
    printf("---------------------------------\n\n");
    
    if(!braket_match(str,&s))
    {
        printf("Express Error!\n");
		return 0;
    }
    else
    {
        printf("%s",str);
    }    
    change_expr(str, post_str, STR_LEN);
	//printf("%s\n",post_str);
    printf(" = %.3f\n", calulate(post_str));
    destroy_stack(&s);
    getchar();getchar();
    return 0;
}

double calulate(char *str)
{
    double doub_stack[STR_LEN];
    int doub_stack_top = 0;
    int i = 0;
    double tmp = 0;
    int count;
    int point_position = 0;
    
    while(str[i] != '\0')
    {
        if(is_digit(str[i]))
        {
            count = 0;
            point_position = 0;
            tmp = str[i] - '0';
            while(str[++i] != ' ')
            {
                count++;
                if(str[i] == '.')
                {
                    point_position = count;
                }
                else
                {
                    tmp *= 10;
                    tmp += str[i] - '0';
                }
            }
            if (point_position)
			{
				while(count > point_position)					
				{
					tmp /= 10;
					count--;
				}
			}
            doub_stack[doub_stack_top++] = tmp;
        }
        if(is_operator(str[i]))
        {
            switch(str[i])
            {
                case '+':
                    doub_stack[doub_stack_top - 2] = doub_stack[doub_stack_top - 2] 
                        + doub_stack[doub_stack_top - 1];
                    doub_stack_top--;
                    break;
                case '-':
                    doub_stack[doub_stack_top - 2] = doub_stack[doub_stack_top - 2] 
                        - doub_stack[doub_stack_top - 1];
                    doub_stack_top--;
                    break;
                case '*':
                    doub_stack[doub_stack_top - 2] = doub_stack[doub_stack_top - 2] 
                        * doub_stack[doub_stack_top - 1];
                    doub_stack_top--;
                    break;
                case '/':
                    doub_stack[doub_stack_top - 2] = doub_stack[doub_stack_top - 2] 
                        / doub_stack[doub_stack_top - 1];
                    doub_stack_top--;
                    break;
                default:
                    break;
            }
        }
        i++;
    }
    return doub_stack[0];
}
int braket_match(char *str, stack *s)
{
    int i = 0;
    while(str[i] != '\0')
    {
        if(str[i] == '(' || str[i] == '[' || str[i] == '{')
        {
            push_stack(s,(int)str[i]);
        }
        if(str[i] == ')' || str[i] == ']' || str[i] == '}')
        {
            switch(str[i])
            {
                case ')':
                    if (pop_stack(s) != '(')
                    {
                        return 0;
                    }
                    break;
                case ']':
                    if (pop_stack(s) != '[')
                    {
                        return 0;
                    }
                    break;
                case '}':
                    if (pop_stack(s) != '{')
                    {
                        return 0;
                    }
                    break;
                default:
                    break;
            }
        }
        i++;
    }
    return (is_empty(s)) ? 1 : 0;
}

/*将中缀表达式表示成后缀表达式*/
void change_expr(char *input, char *out, int len)
{
    stack num_stack, op_stack;
    int i = 0;
    int j = 0;
    init_stack(&num_stack, len);
    init_stack(&op_stack, len);

    while(input[i] != '\0')
    {
        if(input[i] == '(')
        {
            push_stack(&op_stack, input[i]);
        }
        else if(input[i] == ')')
        {
            if(is_empty(&op_stack))
            {
                printf("Operator error!\n");
                return;
            }
            while(get_top(&op_stack) != '(')
            {
                out[j++] = pop_stack(&op_stack);
                out[j++] = ' ';
            }
            pop_stack(&op_stack);
        }        
        else if(is_operator(input[i]))
        {
            if(is_empty(&op_stack) || priority(input[i]) > priority(get_top(&op_stack)))
            {
                push_stack(&op_stack, input[i]);
            }
            else
            {
                while(!(is_empty(&op_stack)) && priority(input[i]) <= priority(get_top(&op_stack)))
                {
                    out[j++] = pop_stack(&op_stack);
                    out[j++] = ' ';
                }
                if(is_empty(&op_stack) || priority(input[i]) > priority(get_top(&op_stack)))
                {
                    push_stack(&op_stack, input[i]);
                }
            }
        }
        else
        {
            while(is_digit(input[i]))
            {
                out[j++] = input[i++];
            }
            out[j++] = ' ';
            i--;
        }
        i++;
    }
    while(!(is_empty(&op_stack)))
    {
        out[j++] = pop_stack(&op_stack);
    }
	out[j] = '\0';
    destroy_stack(&num_stack);
    destroy_stack(&op_stack);
}

int is_operator(char ch)
{
    return (ch == '+' || ch == '-' || ch == '*' || ch == '/') ? 1 : 0;
}

int is_digit(char ch)
{
    return (ch >= '0' && ch <= '9' || ch == '.') ? 1 : 0;
}

int priority(char ch)
{
    switch(ch)
    {
        case '+':
        case '-':
            return 1; break;
        case '*':
        case '/':
            return 2; break;
        default:
            return 0; break;
    }
}


最后输出结果:


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用Java中的表达式库来解析并计算字符串数学表达式,比如Jep、Javaluator等。这些库可以支持各种数学运算符、函数以及括号,并且能够解析字符串形式的表达式并返回结果。 下面是一个使用Jep库实现字符串表达式计算的示例: ```java import org.nfunk.jep.JEP; public class ExpressionCalculator { public static double calculate(String expression) { JEP jep = new JEP(); jep.parseExpression(expression); return jep.getValue(); } } ``` 上述代码中,我们首先导入了Jep库,然后定义了一个calculate方法,该方法接收一个字符串表达式作为参数,并返回表达式计算结果。在方法中,我们创建了一个JEP对象,通过调用parseExpression方法来解析表达式,然后通过调用getValue方法来获取表达式计算结果。 例如,我们可以使用下面的代码来计算字符串表达式"2*(3+4)"的: ```java double result = ExpressionCalculator.calculate("2*(3+4)"); System.out.println(result); // 输出14.0 ``` 除了支持基本的数学运算符和括号外,Jep库还支持自定义函数的定义和使用。例如,我们可以定义一个名为"square"的函数,用于计算一个数的平方: ```java JEP jep = new JEP(); jep.addFunction("square", new MySquareFunction()); jep.parseExpression("square(4)"); // 解析并计算表达式 double result = jep.getValue(); // 自定义函数实现 public class MySquareFunction extends PostfixMathCommand { public MySquareFunction() { numberOfParameters = 1; } @Override public void run(Stack stack) throws ParseException { checkStack(stack);//检查栈是否为空 Object parameter = stack.pop();//从栈中取出参数 double val = ((Double) parameter).doubleValue();//将参数转化为double类型 stack.push(new Double(val * val));//将计算结果入栈 } } ``` 上述代码中,我们首先通过调用addFunction方法来注册一个名为"square"的函数,并指定该函数的实现。然后我们可以在表达式中使用"square"函数进行计算。 总之,使用Java中的表达式库可以方便地解析和计算字符串数学表达式,包括自定义函数和带括号的表达式。 ### 回答2: Java中可以通过解析字符串的数学表达式来进行简单的计算。一般情况下,我们可以使用Java的内置函数和操作符来进行基本的数学计算,但有时我们可能需要处理包含自定义函数和带括号的复杂数学表达式。 对于自定义函数,我们可以通过编写自己的函数来解析和计算字符串中的函数调用。例如,我们可以通过编写一个函数解析器来解析字符串中的数学函数调用,并计算函数的,然后再将其替换为相应的结果。 对于带括号的数学表达式,一种常见的方法是使用递归来解析和计算表达式。我们可以定义一个递归函数,该函数负责解析和计算表达式中的子表达式,然后再根据操作符进行相应的计算。 在解析和计算表达式时,需要注意运算符的优先级和结合性。可以使用栈或递归的方式来处理。我们可以使用两个栈,一个用来存储操作符,另一个用来存储操作数。当遇到操作符时,我们可以比较其优先级与栈顶操作符的优先级,如果当前操作符优先级较高,则将其入栈;否则,将栈顶操作符出栈,并将其应用于相应的操作数,然后将结果入栈。 通过以上方法,我们可以解析和计算复杂的数学表达式。例如,对于表达式"2 * (3 + 4) - 5",我们可以通过解析括号内的子表达式,分别计算"3 + 4"和"2 * 7",最后再将结果相减得到最终结果。 总之,Java可以通过解析字符串的数学表达式来进行简单的计算。对于自定义函数和带括号的表达式,我们可以通过编写函数解析器和使用递归来进行相应的处理。这样,我们就可以方便地处理各种复杂的数学表达式计算。 ### 回答3: Java字符串表达式计算的主要思路是通过解析字符串中的数学表达式,将其转化为计算机可以识别和计算的形式。 首先,通过使用Java的字符串处理功能,可以将输入的数学表达式按照运算符和括号进行分割和切割。然后,使用递归算法来处理表达式中的括号,将括号内的表达式进行计算。 接着,可以使用Java的内置数学函数库来处理表达式中的各种数学函数。可以通过预定义的函数来计算常见的数学函数,也可以通过自定义函数来实现特定的功能。 在计算过程中,可以使用Java中的Stack(堆栈)数据结构来处理运算符和括号。可以借助Stack来判断运算符的优先级,从而决定计算的顺序和方式。通过设定优先级规则,可以保证表达式计算结果的准确性。 最后,通过逐步计算表达式中的各个部分,可以得到最终的计算结果。在计算过程中,需要注意异常情况的处理,例如除零错误或者其他不合法的输入。可以使用异常处理机制来处理这些情况,从而保证程序的稳定性和健壮性。 综上所述,Java通过解析字符串数学表达式进行计算,可以实现简单的数学计算功能,并且支持自定义函数和带括号的表达式。这样的功能可以应用于各种需要进行数学计算的场景,例如科学计算、数据分析等领域。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值