表达式求值(中缀转后缀及后缀表达式求值)

表达式求值(中缀转后缀及后缀表达式求值)。中缀表达式转后缀表达式:中缀表达式转后缀表达式遵循以下原则:1.遇到操作数,直接输出;2.栈为空时,遇到运算符,入栈;3.遇到左括号,将其入栈;4.遇到右括号,执行出栈操作,并将出栈的元素输出,直到弹出栈的是左括号,左括号不输出;5.遇到其他运算符’+”-”*”/’时,弹出所有优先级大于或等于该运

大家好,又见面了,我是你们的朋友全栈君。

。中缀表达式转后缀表达式:

中缀表达式转后缀表达式遵循以下原则:

1.遇到操作数,直接输出;
2.栈为空时,遇到运算符,入栈;
3.遇到左括号,将其入栈;
4.遇到右括号,执行出栈操作,并将出栈的元素输出,直到弹出栈的是左括号,左括号不输出;
5.遇到其他运算符’+”-”*”/’时,弹出所有优先级大于或等于该运算符的栈顶元素,然后将该运算符入栈;
6.最终将栈中的元素依次出栈,输出。
经过上面的步骤,得到的输出既是转换得到的后缀表达式。
举例:a+b*c+(d*e+f)g ———> abc+de*f+g*+

图示上述过程:

因为比较懒,而刚好在网上看到画的还不错的图,所以就直接贴过来了哦。希望作者不要怪罪哦。。。
遇到a,直接输出:

这里写图片描述

遇到+,此时栈为空,入栈:
这里写图片描述

遇到b,直接输出:
这里写图片描述
遇到*,优先级大于栈顶符号优先级,入栈:
这里写图片描述

遇到c,输出:
这里写图片描述
到+,目前站内的与+优先级都大于或等于它,因此将栈内的,+依次弹出并且输出,并且将遇到的这个+入栈:
这里写图片描述
遇到(,将其入栈:
这里写图片描述

遇到d,直接输出:
这里写图片描述
遇到*,由于的优先级高于处在栈中的(,因此入栈:
这里写图片描述
遇到e,直接输出:
这里写图片描述
遇到+,栈顶的优先级高于+,但是栈内的(低于+,将出栈输出,+入栈:这里写图片描述

遇到f,直接输出:
这里写图片描述
遇到),弹出栈顶元素并且输出,直到弹出(才结束,在这里也就是弹出+输出,弹出(不输出:
这里写图片描述
遇到*,优先级高于栈顶+,将*入栈
这里写图片描述
遇到g,直接输出: :
这里写图片描述
此时已经没有新的字符了,依次出栈并输出操作直到栈为空:
这里写图片描述

因为后缀表达式求值过程较为简单:
所以在这里我只进行简单说明:
1.扫描后缀表达式:
①如果是数字,则让其进栈
②若为操作符,则从栈中取出两个操作数,先取出的作为右操作数,后取出的作为左操作数,然后进行该操作符的运算,并使其结果入栈。
③重复上述过程,直至表达式扫描完成。
2.最终栈中只留一个元素—–>即就是结果。

下面代码实现中缀转后缀以及后缀表达式求值:

使用的栈是自定义栈(自己实现的):
//stack.h

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
#include<cassert>

//------------使用类型萃取来拷贝栈内元素-------------
struct _TrueType
{
    bool Get()
    {
        return true;
    }
};

struct _FalseType
{
    bool Get()
    {
        return false;
    }
};

template<class _Tp>
struct TypeTraits
{
    typedef _FalseType _IsPODType;
};

template<>
struct TypeTraits<bool>
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits<int>
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits<unsigned int>
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits<char>
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits< float >
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits< double >
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits<long>
{
    typedef _TrueType _IsPODType;
};

template<>
struct TypeTraits< unsigned long>
{
    typedef _TrueType _IsPODType;
};

template<class T>
void Copy(T* dst, T* src, size_t size)
{
    if (TypeTraits<T>::_IsPODType().Get())
    {
        memcpy(dst, src, size);
    }
    else
    {
        for (int i = 0; i < size; ++i)
        {
            dst[i] = src[i];
        }
    }
}

template<class T>
struct TypeTraits< T* >
{
    typedef _TrueType _IsPODype;
};

//-------------------------栈的基本操作----------------
template<class T>
class Stack
{
public:
    //构造函数
    Stack(int capacity = 10)
        :_pData(NULL)
        , _capacity(capacity)
        , _size(0)
    {
        _pData = new T[capacity];
    }

    //拷贝构造函数
    Stack(const Stack<T>& s)
        :_pData(new T[s._capacity])
        , _size(s._size)
        , _capacity(s._capacity)
    {
        for (int i = 0; i < _size; ++i)
        {
            _pData[i] = s._pData[i];
        }
    }

    //赋值运算符函数
    Stack& operator=(Stack<T> s)
    {
        std::swap(_pData, s._pData);
        _size = s._size;
        _capacity = s._capacity;
        return *this;
    }

    //入栈
    void Push(const T& data)
    {
        CheckCapacity();
        _pData[_size++] = data;
    }
    //出栈
    void Pop()
    {
        if (!Empty())
        {
            --_size;
        }
    }

    //获取栈顶元素
    T& Top()
    {
        if (!Empty())
        {
            return _pData[_size - 1];
        }
    }

    const T& Top()const
    {
        if (!Empty())
        {
            return _pData[_size - 1];
        }
    }

    size_t Size()const
    {
        return _size;
    }

    bool Empty()const
    {
        return 0 == _size;
    }

    //析构函数(释放资源)
    ~Stack()
    {
        if (_pData)
        {
            delete[] _pData;
            _pData = NULL;
        }
    }

private:
    //增容
    void CheckCapacity()
    {
        if (_size >= _capacity)
        {
            _capacity = 2 * _capacity + 3;
            T* tmp = new T[_capacity];
            //拷贝原数据
            //释放旧空间
            //指向新空间
           //需要进行类型萃取
            Copy<T>(_pData, tmp, _size);
            delete[] _pData;
            _pData = tmp;
        }
    }

    T* _pData;
    int _capacity;
    int _size;
};

//----------------------------------------------------------
//需要用到的函数的声明
int GetPriority(char ch, int flag);//获取优先级
bool IsOperator(char ch);//判断是否为操作符
void prefixionToSuffix(char* dst, char* src);//中缀表达式转后缀表达式
int  SuffixToValue(char *suffix, char *prefixion);//后缀表达式求值


中缀表达式转后缀表达式:
//prefixionToSuffix.cpp

#include"Stack.h"

//flag为1时表示栈内优先级  flag为0表示栈外优先级
int GetPriority(char ch, int flag)
{
    if (ch == '+' || ch == '-')
    {
        if (flag)
        {
            return 3;
        }
        else
        {
            return 2;
        }
    }
    else if (ch == '*' || ch == '/' || ch == '%')
    {
        if (flag)
        {
            return 5;
        }
        else
        {
            return 4;
        }
    }
    else if (ch == '(')
    {
        if (flag)
        {
            return 1;
        }
        return 6;
    }
    else if (ch == ')')
    {
        if (flag)
        {
            return 6;
        }
        else
        {
            return 1;
        }
    }
}

bool IsOperator(char ch)
{
    if (ch == '+' || ch == '-' || ch == '*' || ch == '/' || ch == '%' || ch == '(' || ch == ')')
    {
        return true;
    }
    return false;
}

//中缀表达式转后缀表达式
void prefixionToSuffix(char* dst, char* src)
{
    assert(dst);
    assert(src);
    Stack<char> s;
    char * cur = src;
    char* tmp = dst;
    while (*cur != '\0')
    {
        if (*cur >= '0' && *cur <= '9')
        {
            *tmp++ = *cur;
            cur++;
            continue;
        }
        else if (IsOperator(*cur))
        {
            if (s.Empty())//如果栈为空,直接入栈
            {
                s.Push(*cur);
                cur++;
            }
            else//如果栈不空,则需要判断栈顶元素和当前操作符的优先级
            {
                if (*cur == ')')
                {
                    while (*cur == ')' && s.Top() != '(')
                    {
                        *tmp++ = s.Top();
                        s.Pop();
                    }
                    s.Pop();
                    cur++;
                }
                if (GetPriority(*cur, 0) > GetPriority(s.Top(), 1))
                {
                    s.Push(*cur);
                    cur++;
                }
                else
                {
                    while (!s.Empty() && GetPriority(*cur, 0) < GetPriority(s.Top(), 1))
                    {
                        *tmp++ = s.Top();
                        s.Pop();
                    }
                    s.Push(*cur);
                    cur++;
                }

            }
        }
        else
        {
            *tmp++ = *cur++;
        }
    }
    while(!s.Empty())
    {
        *tmp++ = s.Top();
        s.Pop();
    }
}

后缀表达式求值:
//SuffixToValue.cpp

#include"Stack.h"

//12 3 4 + * 6 - 8 2 / +
int  SuffixToValue(char *suffix, char *prefixion)
{
    prefixionToSuffix(suffix, prefixion);

    Stack<int> s;
    char* cur = suffix;
    int res = 0;
    int tmp = 0;
    while (*cur != '\0')
    {
        if (*cur == '+' || *cur == '-' || *cur == '*' || *cur == '/' || *cur == '%')
        {
            char ch = *cur;
            int right = s.Top();
            s.Pop();
            int left = s.Top();
            s.Pop();
            switch (ch)
            {
            case '+':
                s.Push(left + right);
                break;
            case '-':
                s.Push(left - right);
                break;
            case '*':
                s.Push(left * right);
                break;
            case '%':
                s.Push(left % right);
                break;
            case '/':
                if (right)
                {
                    s.Push(left / right);
                }
                break;
            }
            cur++;
        }
        else if (*cur >= '0' && *cur <= '9')
        {
            res = 0;
            while (!isspace(*cur) && *cur >= '0' && *cur <= '9')
            {
                tmp = *cur - '0';
                res = res * 10 + tmp;
                cur++;
            }
            s.Push(res);
            //cur++;
        }
        else
        {
            cur++;
        }
    }
    if (!s.Empty())
    {
        res = s.Top();
        return res;
    }
}

main.cpp

#include"Stack.h"

void Test()
{
    char prefixion[] = "12 * (3 + 4) - 6 + 8 / 2 ";//保存前缀表达式
    char suffix[25] = {};//保存后缀表达式
    int res = SuffixToValue(suffix, prefixion);
    cout << res << endl;
}

int main()
{
    Test();
    return 0;
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/149735.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • PID控制算法的C语言实现

    PID控制算法的C语言实现前言最近在学习PID算法,在了解了算法的套路以后,就要进行实验。如何用C语言实现呢?在网络搜索发现了一篇很好的博客,不过里面的数据又臭又长。在这里转载过来,重下新整理了一下。(原文链接)整理中发现,原文参考的原理在工业应用中PID及其衍生算法是应用最广泛的算法之一,是当之无愧的万能算法,如果能够熟练掌握PID算法的设计与实现过程,对于一般的研发人员来讲,应该是足够应对一般研发问题了,而难能可…

    2022年6月6日
    28
  • HandlerThread详解「建议收藏」

    HandlerThread详解「建议收藏」转载请注明链接:HandlerThread版本:2018/9/9-1(10:10)HandlerThread问题汇总基本使用(7)源码(7)问题汇总【☆】是补充问题,直接给答案。其余问题答案都在文中。HandlerThread是什么?【☆】HandlerThread任务是并行处理的?不是!是串行处理的。如果一个任务执行…

    2022年7月12日
    17
  • 用户态与内核态之间切换详解[通俗易懂]

    用户态与内核态之间切换详解[通俗易懂]用户态切换到内核态CPU中有一个标志字段,标志着线程的运行状态。用户态和内核态对应着不同的值,用户态为3,内核态为0.每个线程都对应着一个用户栈和内核栈,分别用来执行用户方法和内核方法。用户方法就是普通的操作。内核方法就是访问磁盘、内存分配、网卡、声卡等敏感操作。当用户尝试调用内核方法的时候,就会发生用户态切换到内核态的转变。切换流程:1、每个线程都对应这一个TCB,TCB中有一个TSS字段,存储着线程对应的内核栈的地址,也就是内核栈的栈顶指针。2、因为从用户态切换到内核态时,需要将用户态.

    2025年12月16日
    3
  • 地理加权回归学习记录「建议收藏」

    地理加权回归学习记录「建议收藏」地理加权回归GWR白话空间统计二十四:地理加权回归(三)地理加权回归可以用来量化空间异质性。研究区域——利用每个要素的不同空间位置计算距离衰减函数——把每个要素的空间位置(一般是坐标信息(x,y))和要素的值带入到这个函数里面之后,就可以得到一个权重值,这个值就可以带入到回归方程里面去这个衰减函数的理论基础,正是Tobler提出所谓的“地理学第一定律(Tobler’sFirstLaw或者Tobler’sFirstLawofGeography):位置越接近的数据,比远处的数据对结果的影响更

    2022年9月28日
    4
  • C++实现远程桌面集群软件[通俗易懂]

    C++实现远程桌面集群软件[通俗易懂]由于在学校需要管理很多主机的需要,自己动手写了个3389桌面集群的软件。软件很简单,分别用2种方式实现:(1)快速登入模式:微软的MsRdpClientActiveX控件实现(2)远程桌面模式:生成.rdp文件实现

    2022年10月15日
    2
  • Java的文件读写操作

    Java的文件读写操作file(内存)—-输入流—->【程序】—-输出流—->file(内存)当我们读写文本文件的时候,采用Reader是非常方便的,比如FileReader,InputStreamReader和BufferedReader。其中最重要的类是InputStreamReader,它是字节转换为字符的桥梁。你可以在构造器重指定编码的方式,如果不指定的话将采用底层操作系统的默认编

    2022年7月26日
    8

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号