C# 栈的应用
栈的特性:后进先出(LIFO)
回文判断
类似123321,123a321即为回文
思路:
- 将字符串前一半入栈
- 依次弹出栈与字符串后一半比较
public static bool IsPlalindrome(string str)
{
var stack = new Stack<char>();
for (int i = 0; i < str.Length / 2; i++)
{
stack.Push(str[i]);
}
var len = str.Length % 2 == 0 ? str.Length / 2 : (str.Length + 1) / 2;
for (int i = len; i < str.Length; i++)
{
if (stack.Pop() != str[i])
{
return false;
}
}
return true;
}
进制转化
10进制转8进制 1024→2000
转换流程如下:
| N | N div | N mod |
|---|---|---|
| 1024 | 128 | 0 |
| 128 | 16 | 0 |
| 16 | 2 | 0 |
| 2 | 0 | 2 |
思路:
- 取模入栈
- 整除运算直至为0
实现如下:
private static void Main()
{
Console.WriteLine(Get(4396, 8));
Console.ReadKey();
}
public static int Get(int value, int i)
{
var stack = new Stack<int>();
while (value != 0)
{
stack.Push(value % i);
value /= i;
}
return int.Parse(string.Join("", stack));
}
括号匹配
- 圆括号、方括号和花括号可以任意嵌套
- 正确格式:{{90[]}}(4)
- 错误格式:{2(1}1)[3]
思路:
- 如果时
(,[,{则入栈 - 如果时
),],}则将对应左边括号弹出栈
实现如下:
private static void Main()
{
Console.WriteLine(Check("{abc[1](2)}sss(aaa)[({})]"));
Console.ReadKey();
}
public static bool Check(string str)
{
var stack = new Stack<char>();
foreach (var c in str)
{
switch (c)
{
case '(':
case '{':
case '[':
stack.Push(c);
break;
case ')':
if (stack.Count == 0 || stack.Pop() != '(')
{
return false;
}
else
{
break;
}
case '}':
if (stack.Count == 0 || stack.Pop() != '{')
{
return false;
}
else
{
break;
}
case ']':
if (stack.Count == 0 || stack.Pop() != '[')
{
return false;
}
else
{
break;
}
}
}
return stack.Count == 0;
}
中缀转后缀表达式求职
运算规则
- 从左算到右
- 先乘除,后加减
- 先括号内,后括号外
相邻两个操作符优先级判断如下:
c1表示前一个操作符,c2表示后一个操作符
| c1/c2 | + | - | * | / | ( | ) |
|---|---|---|---|---|---|---|
| + | > | > | < | < | < | > |
| - | > | > | < | < | < | > |
| * | < | < | > | > | < | > |
| / | < | < | > | > | < | > |
| ( | < | < | < | < | < | = |
| ) | > | > | > | > | > |
思路:
- 分操作数栈和操作符栈
- 操作数进操作数栈
- 当前操作符优先级大于顶栈操作符则入栈
- 当前操作符优先级小于顶栈操作符,则弹出顶栈,弹出两个操作数运算,运算结果再入栈
- 重复上一步骤,直至将当前操作符入栈
- 若最后两栈都不为空,则依次弹出操作符与操作数计算,直至操作符栈为空,此时操作数栈剩一个元素即为最终结果。
实现如下:
private static void Main()
{
Console.WriteLine(Calculation("(2+3)*2+2*(6-3)/(4-2)+2"));
Console.ReadKey();
}
public static int Calculation(string str)
{
//操作数栈
var opndStack = new Stack<int>();
//操作符栈
var optrStack = new Stack<char>();
foreach (var c in str)
{
if (char.IsDigit(c))
{
//当前的字符是操作数
opndStack.Push(int.Parse(c.ToString()));
}
else
{
//当前的字符是操作符
while (optrStack.Count != 0)
{
var priority = Priority(optrStack.Peek(), c);
if (priority == '<')
{
//栈顶优先级小与当前操作符
//入栈
optrStack.Push(c);
break;
}
if (priority == '=')
{
//栈顶优先级等于当前操作符
//就是左右括号匹配,弹出左括号
optrStack.Pop();
break;
}
if (priority != '>')
{
continue;
}
//栈顶优先级大于当前操作符
//需要计算
var optr = optrStack.Pop();
var value2 = opndStack.Pop();
var value1 = opndStack.Pop();
opndStack.Push(Operate(value1, optr, value2));
}
//1.第一次栈为空直接入栈。
//2.退栈直至为空当前操作符也需要入栈,但")"无需入栈
if (optrStack.Count == 0 && c != ')')
{
optrStack.Push(c);
}
}
}
while (optrStack.Count != 0)
{
var optr = optrStack.Pop();
var value2 = opndStack.Pop();
var value1 = opndStack.Pop();
opndStack.Push(Operate(value1, optr, value2));
}
return opndStack.Count == 1 ? opndStack.Pop() : 0;
}
public static int Operate(int value1, char optr, int value2)
{
switch (optr)
{
case '+':
return value1 + value2;
case '-':
return value1 - value2;
case '*':
return value1 * value2;
case '/':
return value1 / value2;
}
return 0;
}
/// <summary>
/// 比较栈顶操作符与当前操作符优先级
/// </summary>
/// <param name="c1">栈顶操作符</param>
/// <param name="c2">当前操作符</param>
/// <returns></returns>
public static char Priority(char c1, char c2)
{
switch (c1)
{
case '+':
case '-':
if (c2 == '+' || c2 == '-' || c2 == ')')
{
return '>';
}
return '<';
case '*':
case '/':
if (c2 == '(')
{
return '<';
}
return '>';
case '(' when c2 == ')':
return '=';
case '(':
return '<';
case ')':
return '>';
}
return '>';
}
reference
C# 栈的应用的更多相关文章
- 通往全栈工程师的捷径 —— react
腾讯Bugly特约作者: 左明 首先,我们来看看 React 在世界范围的热度趋势,下图是关键词“房价”和 “React” 在 Google Trends 上的搜索量对比,蓝色的是 React,红色的 ...
- Java 堆内存与栈内存异同(Java Heap Memory vs Stack Memory Difference)
--reference Java Heap Memory vs Stack Memory Difference 在数据结构中,堆和栈可以说是两种最基础的数据结构,而Java中的栈内存空间和堆内存空间有 ...
- duang~免费的学习视频来啦:学霸君之全栈测试
学霸君向童鞋们推荐一款 同名学霸学习 视频教程 重点是完全免费收看学习噢!!! 今天 学霸君推荐腾讯课堂的学霸君之全栈测试 复制下方链接至腾讯课堂中报名学习 https://ke.qq.com/cou ...
- [数据结构]——链表(list)、队列(queue)和栈(stack)
在前面几篇博文中曾经提到链表(list).队列(queue)和(stack),为了更加系统化,这里统一介绍着三种数据结构及相应实现. 1)链表 首先回想一下基本的数据类型,当需要存储多个相同类型的数据 ...
- BZOJ1012: [JSOI2008]最大数maxnumber [线段树 | 单调栈+二分]
1012: [JSOI2008]最大数maxnumber Time Limit: 3 Sec Memory Limit: 162 MBSubmit: 8748 Solved: 3835[Submi ...
- BZOJ 4453: cys就是要拿英魂![后缀数组 ST表 单调栈类似物]
4453: cys就是要拿英魂! Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 90 Solved: 46[Submit][Status][Discu ...
- BZOJ 3238: [Ahoi2013]差异 [后缀数组 单调栈]
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2326 Solved: 1054[Submit][Status ...
- .NET全栈开发工程师学习路径
PS:最近一直反复地看博客园以前发布的一条.NET全栈开发工程师的招聘启事,觉得这是我看过最有创意也最朴实的一个招聘启事,更为重要的是它更像是一个技术提纲,能够指引我们的学习和提升,现在转载过来与各位 ...
- Nodejs之MEAN栈开发(八)---- 用户认证与会话管理详解
用户认证与会话管理基本上是每个网站必备的一个功能.在Asp.net下做的比较多,大体的思路都是先根据用户提供的用户名和密码到数据库找到用户信息,然后校验,校验成功之后记住用户的姓名和相关信息,这个信息 ...
- 匹夫细说C#:不是“栈类型”的值类型,从生命周期聊存储位置
0x00 前言: 匹夫在日常和别人交流的时候,常常会发现一旦讨论涉及到“类型”,话题的热度就会立马升温,因为很多似是而非.或者片面的概念常常被人们当做是全面和正确的答案.加之最近在园子看到有人翻译的& ...
随机推荐
- 消息中间件——RabbitMQ(二)各大主流消息中间件综合对比介绍!
前言 消息队列已经逐渐成为企业IT系统内部通信的核心手段.它具有低耦合.可靠投递.广播.流量控制.最终一致性等一系列功能,成为异步RPC的主要手段之一.当今市面上有很多主流的消息中间件,如老牌的Act ...
- spring boot 打 war包
spring boot .spring cloud打 war包,并发布到tomcat中运行 1.pom文件修改 <packaging>war</packaging> 2.< ...
- 中间件增强框架之InterceptFramework
本文讲解MOF中的InterceptFramework框架.该框架可以在应用启动过程中获取画像信息,实现应用画像数据采集和存储. 一.前言 在智能运维中,应用服务所使用的组件及JAR包等相关信息非常重 ...
- jmeter学习笔记-----第一天
环境简介:apache-jmeter-3.3 win10系统 一.Jmeter录制操作步骤: 1.工作台---添加HTTP代理服务器: 2.为自己笔记本的浏览器设置相同代理: 3.测试计划下-s ...
- Unity经典案例之:Fire Balls 多个圆环以及圆环的变速变向
版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...
- java 中 size() 和 length()
偶然发现自己不清楚 java size() 和length()是干嘛用的,总结一下: 1.java中的length()方法是针对字符串String说的,如果想看这个字符串的长度则用到length()这 ...
- springbootl用thymeleaf整合htm
pom文件: <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...
- 80后,天才程序员, Facebook 第一任 CTO,看看开挂的人生到底有多变态?
鸡仔说:今天介绍一位天才程序员--亚当·德安格洛(Adam D'Angelo),他被<财富>杂志誉为"科技界最聪明的人之一",大学去了被誉为"天才" ...
- runnable和thread实现多线程的区别
下面以典型的买票程序(基本都是以这个为例子)为例,来说明二者的区别. 首先通过继承Thread类实现,代码如下: class MyThread extends Thread{ private int ...
- Java-手动搭建SSM(Maven)
一.环境部署 操作系统:windows10专业版 jdk:1.8.0_144 IDE:eclipse-oxygen 服务器:tomcat 9.0 数据库:mysql 5.7.18 Maven:3.54 ...