栈与栈的实现

栈是一种基础的数据结构,只从一端读写数据。基本特点就”后进先出“,例如顺序入栈1,2,3,4,5,再顺序出栈是5,4,3,2,1

栈的基本操作

栈的基本操作有如下几种:

  • 检测栈是否为空
  • 返回栈存储数据的数量
  • 返回栈顶数据/返回栈顶数据并将其弹出
  • 将数据压入栈
  • 清空栈

栈的实现

软件实现——GO语言

软件的栈可以使用链表基本结构实现或使用数组实现:使用链表栈的优势是栈的容量几乎不限,确定是入栈出栈都需要开销较大的声明结构体;数组实现的优势是速度快(自增自减一般有指令实现),但是空间必须预先指定。

统一adt接口

type Stack_adt interface {
    Is_empty() bool
    Get_depth() int
    Push(data Stack_data)
    Pop() (Stack_data, error)
    Get_head() (Stack_data, error)
    Clear()
}

链表栈

数据结构

type stack_node struct {
    data Stack_data
    next *stack_node
}

type Link_stack struct {
    length int
    head   *stack_node
}

判空方法

func (l *Link_stack) Is_empty() bool {
    if l.head.next == nil {
        return true
    } else {
        return false
    }
}

获取栈中数据量方法

func (l *Link_stack) Get_depth() int {
    return l.length
}

压栈方法

func (l *Link_stack) Push(data Stack_data) {
    new_node := stack_node{data, l.head.next}
    l.head.next = &new_node
    l.length++
}

对于入栈数据,建立链表节点并将其插入到头结点后(读取位置),保证后进先出

弹栈方法

func (l *Link_stack) Pop() (Stack_data, error) {
    if l.Is_empty() {
        return Stack_data{}, errors.New("empty stack")
    } else {
        data := l.head.next.data
        l.head.next = l.head.next.next
        l.length--
        return data, nil
    }
}

直接取出头结点后的节点,若本来就是空栈则返回一个空栈的errors异常,否则该异常为nil

获取栈顶数据(仅获取,不弹出)

func (l *Link_stack) Get_head() (Stack_data, error) {
    if l.Is_empty() {
        return Stack_data{}, errors.New("empty stack")
    } else {
        return l.head.next.data, nil
    }
}

与弹栈相同,不同的是读取后不将读取的节点移出链表

清空栈

func (l *Link_stack) Clear() {
    _, err := l.Pop()
    for err == nil {
        _, err = l.Pop()
    }
}

不断弹栈并查看异常(抛弃读出数据),直到报出空栈异常(返回异常不为nil

数组栈

数据结构

const DEPTH = 10

type Array_stack struct {
    data   [DEPTH]Stack_data
    length int
}

data为数组,用于存储数据;length为存入数据的数量,同时也是“栈顶指针”,标记入栈位置

判空方法

func (a *Array_stack) Is_empty() bool {
    if a.length == 0 {
        return true
    } else {
        return false
    }
}

获取栈中数据量方法

func (a *Array_stack) Get_depth() int {
    return a.length
}

压栈方法

func (a *Array_stack) Push(data Stack_data) {
    if a.length >= DEPTH {
        return
    } else {
        a.data[a.length] = data
        a.length++
    }
}

对于入栈数据,若入栈位置已经超出数组尺寸,则栈满,不入栈。否则将输入数据插入length标记的入栈位置,并将length自加

弹栈方法

func (a *Array_stack) Pop() (Stack_data, error) {
    if a.length == 0 {
        return Stack_data{}, errors.New("empty stack")
    } else {
        a.length--
        return a.data[a.length], nil
    }
}

先将length自减,获得上一个入栈元素的位置,再返回该值

获取栈顶数据(仅获取,不弹出)

func (a *Array_stack) Get_head() (Stack_data, error) {
    if a.length == 0 {
        return Stack_data{}, errors.New("empty stack")
    } else {
        return a.data[a.length-1], nil
    }
}

与弹栈相同,不同的是读取后不改变“栈顶指针”的位置

清空栈

func (a *Array_stack) Clear() {
    a.length = 0
}

直接将“栈顶指针”清零即可实现清空栈

切片栈

切片是一种Go语言特有的数据结构,类似于动态数组,使用切片可以实现深度可变的栈。

硬件实现——Verilog语言

module stack_controller #(
    parameter DEPTH_LOG = 4,
    parameter WIDTH = 8
)(
    input clk,    // Clock
    input rst_n,  // Asynchronous reset active low

    input stack_write_req,
    input [WIDTH - 1:0]stack_write_data,
    input stack_read_req,

    output reg stack_empty,
    output stack_full,

    output reg ram_write_req,
    output reg [DEPTH_LOG - 1:0]ram_addr,
    output reg [WIDTH - 1:0]ram_write_data
);

reg [DEPTH_LOG:0]stack_point;
wire is_full = (stack_point == 2 ** DEPTH_LOG)?1'b1:1'b0;
wire is_empty = (stack_point == 'b0)?1'b1:1'b0;
always @ (posedge clk or negedge rst_n) begin //control point of stack
    if(~rst_n) begin
        stack_point <= 'b0;
    end else if(stack_write_req && stack_read_req) begin //lock
        stack_point <= stack_point;
    end else if(stack_write_req && !is_full) begin //write when stack is not full
        stack_point <= stack_point + 1'b1;
    end else if(stack_read_req && !is_empty) begin // read when stack is not empty
        stack_point <= stack_point - 1'b1;
    end
end
assign stack_full = stack_point[DEPTH_LOG];

always @ (posedge clk or negedge rst_n) begin //generate empty signal
    if(~rst_n) begin
        stack_empty <= 'b0;
    end else if(ram_addr == 'b0 && is_empty) begin // delay signal
        stack_empty <= 1'b1;
    end else begin
        stack_empty <= 'b0;
    end
end

always @ (posedge clk or negedge rst_n) begin //generate ram_write_req
    if(~rst_n) begin
        ram_write_req <= 'b0;
    end else if(!is_full) begin
        ram_write_req <= stack_write_req;
    end else begin
        ram_write_req <= 'b0;
    end
end

always @ (posedge clk or negedge rst_n) begin //prepare the addr and data for push
    if(~rst_n) begin
        ram_addr <= 'b0;
        ram_write_data <= stack_write_data;
    end else begin
        ram_addr <= stack_point[DEPTH_LOG - 1:0];
        ram_write_data <= stack_write_data;
    end
end

endmodule

Verilog实现栈的关键点有三个:

  • 控制栈顶指针
  • 栈满信号生成
  • 栈空信号生成

该硬件栈的栈顶指针指向下一个入栈的位置,且位数比ram地址位多一位,当最高位为1时,可认为栈溢出,停止写入;同理,当栈顶指针指向0,该栈为空栈。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容