堆栈(Stack)是一种特殊的线性表,是一种操作只允许在尾端进行插入或删除等操作的线性表。表尾允许进行插入删除操作,称为栈顶(Top),另一端是固定的,称为栈底(Bottom).栈的操作使按照先进后出或后进先出的原则进行的。

用一片连续的存储空间来存储栈中的数据元素,称为顺序栈(Sequence Stack)。类似于顺序表,用一维数组来存放栈中的数据元素。缺点:浪费存储空间。

用链式存储结构来存储的栈为链栈(Linked Stack).链栈通常用单链表来表示。

Stack

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace DataStructure
{ interface IStack<T>
{
void Push(T item); //入栈操作
T Pop(); //出栈操作
T GetTop(); //取栈顶元素
int GetLength(); //求栈的长度
bool IsEmpty(); //判断栈是否为空
void Clear(); //清空操作
}
/// <summary>
/// 顺序栈
/// </summary>
/// <typeparam name="T"></typeparam>
class SequenceStack<T> : IStack<T>
{
private int maxsize; //顺序栈的容量
private T[] data; //数组,用于存储顺序栈中的数据元素
private int top; //指示顺序栈的栈顶 //索引器
public T this[int index]
{
get { return data[index]; }
set { data[index] = value; }
} //容量属性
public int Maxsize
{
get { return maxsize; }
set { maxsize = value; }
} //栈顶属性
public int Top
{
get
{
return top;
}
} public SequenceStack(int size)
{
data = new T[size];
maxsize = size;
top = -;
} //求栈的长度
public int GetLength()
{
return top + ;
}
//清空顺序栈
public void Clear()
{
top = -;
}
//判断顺序栈是否为空
public bool IsEmpty()
{
if (top == -)
{
return true;
}
else
return false;
}
//判断栈是否为满
public bool IsFull()
{
if (top == maxsize - )
{
return true;
}
else
return false;
} //入栈
public void Push(T elem)
{
if (IsFull())
{
Console.WriteLine("Stack is Full !");
return;
}
data[++top] = elem; } //出栈
public T Pop()
{
T tem = default(T);
if (IsEmpty())
{
Console.WriteLine("Stack is Empty !");
return default(T);
}
tem = data[top];
--top;
return tem;
} //获取栈顶元素
public T GetTop()
{
if (IsEmpty())
{
Console.WriteLine("Stack is Empty !");
return default(T);
}
return data[top];
} } /// <summary>
/// 用顺序栈解决火车车厢重排问题
/// </summary>
class TrainArrangwBySeqStack
{
//车厢重排算法,K个缓冲铁轨,车厢初始排序存放在P中
public bool RailRoad(int[] p, int n, int k)
{
//创建与缓冲铁轨对应的堆栈
SequenceStack<int>[] H;
H = new SequenceStack<int>[k + ];
for (int i = ; i <= k; i++)
H[i] = new SequenceStack<int>(p.Length);
int NowOut = ; //下次要输出的车厢
int minH = n + ; //缓冲铁轨中编号最小的车厢
int minS = ; //minH号车厢对应的缓冲铁轨 //车厢重排
for (int i = ; i < n; i++)
{
if (p[i] == NowOut)
{
Console.WriteLine("Move car {0} from input to output", p[i]);
NowOut++;
//从缓冲铁轨中输出
while (minH == NowOut)
{
Output(ref minH, ref minS, ref H, k, n);
NowOut++;
}
}
else
{
if (!Hold(p[i], ref minH, ref minS, ref H, k, n))
{
return false;
}
}
}
return true;
} //在一个缓冲区中放入车厢C
public bool Hold(int c, ref int minH, ref int minS, ref SequenceStack<int>[] H, int k, int n)
{
//如果没有可用的缓冲铁轨,则返回false
//否则返回true
//为车厢c寻找最优的铁轨
//初始化 int BestTrack = ; //目前最优的铁轨
int BestTop = n + ; //最优铁轨上的头辆车厢
int x; //车厢索引 //扫描缓冲铁轨
for (int i = ; i <= k; i++)//!!!i=1
{
if (!H[i].IsEmpty())
{
//铁轨i不为空
x = H[i][H[i].Top];
if (c < x && x < BestTop)
{
//铁轨i顶部的车厢编号最小
BestTop = x;
BestTrack = i;
}
}
else//铁轨i为空
{
if (BestTrack == )
{
BestTrack = i;
}
break;
}
}
if (BestTrack == )
return false;//没有可用铁轨 //把车厢c送入缓冲铁轨
H[BestTrack].Push(c);
Console.WriteLine("Move car{0} from input to holding track {1}", c, BestTrack);
//必要时修改minH minS
if (c < minH)
{
minH = c;
minS = BestTrack;
}
return true;
} //把车厢从缓冲区铁轨送至出轨处,同时修改minH minS
public void Output(ref int minH, ref int minS, ref SequenceStack<int>[] H, int k, int n)
{
int c;//车厢索引
//从堆栈minS中删除编写、好最小的车厢minH
c = H[minS].Pop();
Console.WriteLine("Move car{0} from holding track {1} to output", minH, minS);
//通过检查所有的栈顶,搜索新的minH minS
minH = n + ;
for (int i = ; i <= k; i++)
{
if (!H[i].IsEmpty() && (c = H[i][H[i].Top]) < minH)
{
minH = c;
minS = i;
}
}
}
} /// <summary>
/// 链栈结点
/// </summary>
class StackNode<T>
{
private T data; //数据域
private StackNode<T> next; //引用域 public StackNode()
{
data = default(T);
next = null;
} public StackNode(T val)
{
data = val;
next = null;
} public StackNode(T val, StackNode<T> p)
{
data = val;
next = p;
} //数据域属性
public T Data
{
get { return data; }
set { data = value; }
} //引用域属性
public StackNode<T> Next
{
get { return next; }
set { next = value; }
} } /// <summary>
/// 链栈
/// </summary>
/// <typeparam name="T"></typeparam>
class LinkStack<T> : IStack<T>
{
private StackNode<T> top; //栈顶指示器
private int size; //栈中元素的个数 //栈顶指示器属性
public StackNode<T> Top
{
get { return top; }
set { top = value; }
} //元素个数属性
public int Size
{
get { return size; }
set { size = value; }
} public LinkStack()
{
top = null;
size = ;
} //判断链栈是否为空
public bool IsEmpty()
{
if ((top == null) && (size == ))
return true;
else
return false;
} public int GetLength()
{
return size;
} public void Clear()
{
top = null;
size = ;
} //入栈操作
//在单链表的起始处插入一个结点
public void Push(T item)
{
StackNode<T> q = new StackNode<T>(item);
if (top == null)
{
top = q;
}
else
{
//将新结点的next指向栈顶指示器top所指向的结点
q.Next = top;
//将栈顶指示器top指向新结点
top = q;
}
++size;
} //出栈操作
public T Pop()
{
if (IsEmpty())
{
Console.WriteLine("Stack is empty !");
return default(T);
}
StackNode<T> p = top;
top = top.Next;
--size;
return p.Data;
} //获取栈顶结点的值
public T GetTop()
{
if (IsEmpty())
{
Console.WriteLine("Stack is empty !");
return default(T);
}
return top.Data;
}
} /// <summary>
/// 用链栈解决火车车厢重排问题
/// </summary>
class TrainArrangeByLinkStack
{
//k个缓冲铁轨,车厢初始排序存储在p中
public bool RailRoad(int[] p, int n, int k)
{
//创建与缓冲铁轨对应的堆栈
LinkStack<int>[] H;
H = new LinkStack<int>[k + ];
for (int i = ; i <= k; i++)
{
H[i] = new LinkStack<int>();
} int NowOut = ; //下一次要输出的车厢
int minH = n + ; //缓冲铁轨中编号最小的车厢
int minS = ; //minH号车厢对应的缓冲铁轨 //车厢重排
for (int i = ; i < n; i++)
{
if (p[i] == NowOut)
{
Console.WriteLine("Move car {0} from input to output", p[i]);
NowOut++; //从缓冲铁轨中输出
while (minH == NowOut)
{
Output(ref minH, ref minS, ref H, k, n);
NowOut++;
}
}
else
{
//将p[i]送入缓冲铁轨
if (!Hold(p[i], ref minH, ref minS, ref H, k, n))
return false;
}
}
return true;
} //在一个缓冲铁轨中放入车厢c
public bool Hold(int c, ref int minH, ref int minS, ref LinkStack<int>[] H, int k, int n)
{
//如果没有可用缓冲铁轨,则返回false
//否则返回true
//为车厢c寻找最优的缓冲铁轨
//初始化
int BestTrack = ; //目前最优的铁轨
int BestTop = n + ; //最优铁轨上的头辆车厢
int x; //车厢索引 //扫描缓冲铁轨
for (int i = ; i <= k; i++)
{
if (!H[i].IsEmpty())
{
//铁轨不为空
x = H[i].Top.Data;
if (c < x && x < BestTop)
{
BestTop = x;
BestTrack = i;
}
}
else
{
if (BestTrack == )
BestTrack = i;
break;
}
}
if (BestTrack == )
return false;//没有可用铁轨
//把车厢c送入缓冲铁轨
H[BestTrack].Push(c);
Console.WriteLine("Move car {0} from input to holding track {1}", c, BestTrack); if (c < minH)
{
minH = c;
minS = BestTrack;
}
return true;
} //把车厢从缓冲铁轨送至出轨处,同时修改minH minS
public void Output(ref int minH,ref int minS ,ref LinkStack <int>[] H,int k,int n)
{
int c; //车厢索引
c = H[minS].Pop();
Console.WriteLine("Move car {0} form holding track {1} to output", minH, minS);
//通过检查所有的栈顶,搜索新的minH和minS
minH = n + ;
for (int i = ; i <= k; i++)
{
if (!H[i].IsEmpty() && (c = H[i].Top.Data) < minH)
{
minH = c;
minS = i;
}
}
}
} class Stack
{ static void Main()
{
int[] p = new int[] { , , , , , , , , };
int k = ; //用顺序栈解决火车车厢重排问题
TrainArrangwBySeqStack tas = new TrainArrangwBySeqStack();
bool results;
results = tas.RailRoad(p, p.Length, k);
do
{
if (results == false)
{
Console.WriteLine("need more holding track, please enter additional number:");
k = k + Convert.ToInt32(Console.ReadLine());
results = tas.RailRoad(p, p.Length, k);
}
} while (results == false);
Console.ReadLine(); //用链栈解决火车车厢重排问题
TrainArrangeByLinkStack ta = new TrainArrangeByLinkStack();
bool result;
result = ta.RailRoad(p, p.Length, k);
do
{
if (result == false)
{
Console.WriteLine("need more holding track,please enter additional number:");
k = k + Convert.ToInt32(Console.ReadLine());
result = ta.RailRoad(p, p.Length, k);
}
} while (result == false);
Console.ReadLine();
}
}
}

C#实现堆栈的更多相关文章

  1. 谈谈一些有趣的CSS题目(三)-- 层叠顺序与堆栈上下文知多少

    开本系列,讨论一些有趣的 CSS 题目,抛开实用性而言,一些题目为了拓宽一下解决问题的思路,此外,涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题中有你感觉 ...

  2. History API与浏览器历史堆栈管理

    移动端开发在某些场景中有着特殊需求,如为了提高用户体验和加快响应速度,常常在部分工程采用SPA架构.传统的单页应用基于url的hash值进行路由,这种实现不存在兼容性问题,但是缺点也有--针对不支持o ...

  3. JVM学习(2)——技术文章里常说的堆,栈,堆栈到底是什么,从os的角度总结

    俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及到的知识点总结如下: 堆栈是栈 JVM栈和本地方法栈划分 Java中的堆,栈和c/c++中的堆,栈 数据结构层面的堆,栈 os层面 ...

  4. arcgis engine 中出现的内存堆栈溢出问题。

    两种解决方案: 1.循环加载mxd文档的时候出现的堆栈溢出,解决办法是每次循环结束时清空FeatureLayer,感觉并不好,但是确实可以实现功能. 2.循环调取featureclass的search ...

  5. java 堆栈 理解

    Java 中的堆和栈 堆和栈:分为数据结构的堆和栈以及内存中的堆和栈,两种理解应区分开. 数据结构中的堆: 堆实际上指的就是(满足堆性质的)优先队列的一种数据结构,第1个元素有最高的优先权. 堆性质: ...

  6. python列表模拟堆栈和队列

    对列特点:先进先出.后进后出 用列表insert.pop模拟进队出队: >>> l = [] >>> l.insert(0,'p1') >>> l ...

  7. GDB调试汇编堆栈过程分析

    GDB调试汇编堆栈过程分析 分析过程 这是我的C源文件:click here 使用gcc - g example.c -o example -m32指令在64位的机器上产生32位汇编,然后使用gdb ...

  8. 20145212——GDB调试汇编堆栈过程分析

    GDB调试汇编堆栈过程分析 测试代码 #include <stdio.h> short val = 1; int vv = 2; int g(int xxx) { return xxx + ...

  9. C和指针 第十七章 经典数据类型 堆栈 队列 二叉树

    堆栈: // // Created by mao on 16-9-16. // #ifndef UNTITLED_STACK_H #define UNTITLED_STACK_H #define TR ...

  10. gdb调试汇编堆栈过程的学习

    gdb调试汇编堆栈过程的学习 以下为C源文件 使用gcc - g code.c -o code -m32指令在64位的机器上产生32位汇编,然后使用gdb example指令进入gdb调试器: 进入之 ...

随机推荐

  1. robotframework接口测试初探1

    robotframework这个框架最近很多人在使用它,大部分是和selenium结合的,大概看了下,然后发现这个做接口测试感觉也还不错,初步研究了下 环境安装: robotframework这个环境 ...

  2. Centos 6.7 安装smokeping (最完整教程)

    本教程需要的源码包一并上传了,届时可以直接上传到linux系统里面! 需要编译的fping.echoping.smokeping源码包,链接:http://pan.baidu.com/s/1pL4HL ...

  3. Matlab 2013b 在El Capitan 中无法使用问题解决

    更新了mac的操作系统到El capitan, 结果发现Matlab打不开了,每次都弹出一个Java error的窗口.现实如下内容 java.lang.NullPointerException at ...

  4. seo查询命令

    以下内容均来自网络,只是稍微整理 感觉比较好用的是1.2.5.6.11.12.13条 1. site指令: 查询某个特定网站收录情况. 比如查询6676小游戏收录格式即为:site:www.6676. ...

  5. 遥感影像滤波处理软件 — timesat3.2

    最近因为要做遥感影像的滤波处理,经过女神推荐,决定用Timesat,可是该软件3.1版本只适合xp系统以及2011的matlab,后来在官网上找到了最新的3.2版本.支持64位操作系统以及2014的m ...

  6. 2016年中国大学生程序设计竞赛(合肥)-重现赛1008 HDU 5968

    异或密码 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submis ...

  7. SpringMVC中使用Cron表达式的定时器

    SpringMVC中使用Cron表达式的定时器 cron(定时策略)简要说明 顺序: 秒 分 时 日 月 星期 年份 (7个参数,空格隔开各个参数,年份非必须参数) 通配符: , 如果分钟位置为* 1 ...

  8. telnet -测试端口号

    点击计算机的开始菜单-->运行 ,输入CMD命令,然后确定.打开cmd命令行.   输入telnet测试端口命令: telnet IP 端口 或者 telnet 域名 端口 回车   如果端口关 ...

  9. Android驱动开发前的准备(一)

    Android系统移植与驱动开发概述 1.1 Android 系统架构 1.2 Android系统移植的主要工作 1.3 查看linux内核版本 1.4 linux内核版本号的定义规则 1.5 lin ...

  10. Pocscan搭建详解

    0x01 关于Pocscan Pocscan是一款开源 Poc 调用框架,可轻松调用Pocsuite,Tangscan,Beebeeto,Knowsec老版本POC 按照官方规范编写的 Poc对目标域 ...