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 前言: 匹夫在日常和别人交流的时候,常常会发现一旦讨论涉及到“类型”,话题的热度就会立马升温,因为很多似是而非.或者片面的概念常常被人们当做是全面和正确的答案.加之最近在园子看到有人翻译的& ...
随机推荐
- EMCAscript6随心所记
es6的支持情况http://kangax.github.io/compat-table/es6/ 1.let命令 基本用法 ES6新增了let命令,用来声明变量.它的用法类似于var,但是所声明的变 ...
- 集成Ribbon的客户端调用工具——Feign
什么是Feign? 客户端调用工具 Ribbon+RestTemplate Feign Feign特性: Feign本身包含Ribbon Fegin是一个采用基于接口的注解的声明式客户端调用工具,更加 ...
- Ng-Matero 0.1 发布了!
Ng-Matero 0.1 终于发布了!周末折腾了两天,结果最后发版还是出了点意外,好在今天补了一版. 距离 Ng-Matero 发布第一版已经过去了一个多月,然后很颓废地休息了半个多月,最近项目的关 ...
- 教你用原生CSS写炫酷页面切换效果,跟第三方组件说拜拜
因为项目需要,别人想让我给他写一个个人博客,并且给了我一个其他人的网页,可以点此查看.有的同学可能说了,第三方博客框架这么多,为什么还要去手写的,你说这个有可能是没有看到打开这个博客. 样式介绍 给大 ...
- java 判断 string 转 integer 判断
NumberUtils.isDigits("1") NumberUtils.isDigits("/") 根据返回 true false 再确定是否转换即可 需要 ...
- 渗透之路基础 -- 跨站脚本攻击XSS
目录 漏洞原理及防御 XSS 原理分析:输出问题导致js代码被识别执行 XSS 技术分类 Cookie盗取 基于Xss的WebShell箱子的攻击 XSS相关防护过滤及绕过分析(参考链接) 防护: 绕 ...
- 开发APP必须知道的API集合,来源http://www.cnblogs.com/wikiki/p/7232388.html
笔记 OneNote - OneNote支持获取,复制,创建,更新,导入与导出笔记,支持为笔记添加多媒体内容,管理权限等.提供SDK和Demo. 为知笔记 - 为知笔记Windows客户端开放了大量的 ...
- arcgis三维球中加载2000坐标系出现错误(The tiling scheme of this layer is not supported by SceneView)
目前我们国家测绘地理信息的坐标体系基准是国家2000坐标系CGCS2000.各类地图组件如OpenLayers.Mapbox.Cesuim和ArcGIS Javascrip等都主要是支持WGS84(w ...
- .NET CORE 怎么样从控制台中读取输入流
.NET CORE 怎么样从控制台中读取输入流 从Console.ReadList/Read 的源码中,可学习到.NET CORE 是怎么样来读取输入流. 也可以学习到是如何使用P/Invoke来调用 ...
- 学生管理系统 Python语言
def show_student(): print(('*'*20).center(55)) print('1.添加学生信息'.center(50)) print('2.修改学生信息'.center( ...