今天又看了遍《effective C++》,手动实现了一下条款42中的栈,贴出来当博客的处女贴。

首先栈的声明如下,采用了模板传入类型,而栈的底层采用是个链表。

// stack.h
// by Chwen 2014-10-27 #include<stdio.h>
#include <stdlib.h>
#include <iostream> using namespace std; // 栈声明 template<typename T>
class Stack
{
public:
Stack();
~Stack(); void push(const T& node);
T Top();
void pop();
int size()const;
bool Empty() const;
void clear(); private:
struct StackNode
{
T data;
StackNode* next;
StackNode(const T& Newdata, StackNode* nextNode)
:data(Newdata),next(nextNode)
{}
};
  
StackNode * top;
// 防止默认拷贝和默认赋值
Stack(const Stack& rhs);
Stack& operator=(const Stack& rhs);
int mySize;
};

  而对应的cpp实现如下:

// stack.cpp

#include "stack.h"
using namespace std; // 栈实现 template<typename T>
Stack<T>::Stack()
:top(nullptr),mySize()
{ } template<typename T>
Stack<T>::~Stack()
{
clear();
} template<typename T>
void Stack<T>::push(const T& node)
{
top = new StackNode(node,top);
mySize ++;
} template<typename T>
T Stack<T>::Top()
{
if (Empty())
{
_DEBUG_ERROR("Error, stack is empty!");
}
return top->data;
} template<typename T>
void Stack<T>::pop()
{
if (Empty())
{
_DEBUG_ERROR("Error, stack is empty!");
}
StackNode* topOfStack = top;
top = top->next;
delete topOfStack;
topOfStack = nullptr;
mySize --;
return;
} template<typename T>
bool Stack<T>::Empty() const
{
return top == nullptr;
} template<typename T>
void Stack<T>::clear()
{
while (top)
{
StackNode* topOfStack = top;
top = top->next;
delete topOfStack; }
mySize = ;
}
template<typename T>
int Stack<T>::size()const
{
return mySize;
}

以上即是采用模板实现的栈的所有代码,可以实现栈的push,  pop, top, clear 等操作。

以下写了一个简单的测试代码:

void funv()
{ Stack<int> s; for(int i = ; i < ; ++i)
{
s.push(i);
}
for(int j = s.size()- ; j >= ; --j)
{
cout<< "node: " << s.Top() <<endl;
s.pop();
} s.clear(); }

int main ()
{
  funv();
  getchar();
  return 0;
}

之后effective C++指出了另一种更精巧的方式实现,即私有继承。

代码实现如下:

// stack.h
// by Chwen 2014-10-27
#include<stdio.h>
#include <stdlib.h>
#include <iostream> class commonStack
{
protected:
commonStack();
~commonStack(); void push(void* node);
void* Top();
void pop();
int size()const;
bool Empty() const;
void clear(); private:
struct StackNode
{
void* data;
StackNode* next;
StackNode(void* Newdata, StackNode* nextNode)
:data(Newdata),next(nextNode)
{}
}; StackNode * top;
// 防止默认拷贝和默认赋值
commonStack(const commonStack& rhs);
commonStack& operator=(const commonStack& rhs);
int mySize;
}; template <typename T>
class Stack:private commonStack
{
public:
void push(T * ty){commonStack::push(static_cast<void*>(ty));}
T* top(){return static_cast<T*>(commonStack::Top());}
void pop(){return commonStack::pop();}
int size(){return commonStack::size();}
bool Empty()const{ return commonStack::Empty(); }
void clear(){return commonStack::clear();} };

对应的cpp 如下:

#include "stack.h"

using namespace std;

commonStack::commonStack()
:top(nullptr),mySize()
{ } commonStack::~commonStack()
{
clear();
} void commonStack::push(void* node)
{
top = new StackNode(node,top);
mySize ++;
} void* commonStack::Top()
{
if (Empty())
{
_DEBUG_ERROR("Error, stack is empty!");
}
return top->data;
} void commonStack::pop()
{
if (Empty())
{
_DEBUG_ERROR("Error, stack is empty!");
}
StackNode* topOfStack = top;
top = top->next;
delete topOfStack;
topOfStack = nullptr;
mySize --;
return;
} bool commonStack::Empty() const
{
return top == nullptr;
} void commonStack::clear()
{
while (top)
{
StackNode* topOfStack = top;
top = top->next;
delete topOfStack; }
mySize = ;
} int commonStack::size()const
{
return mySize;
}

这里commonStack将原模板类的T改为了void*, 之后使用protected保护该类不会被其他不明群众调用,而是给出了一个模板接口类私有继承这个类,这样一来,既起到了保护作用,又在低损耗的情况下给出了方便易用的接口,巧夺天工的设计。

测试代码如下:

void funcInt()
{
int* a[];
for(int i = ; i < ; ++i)
{
a[i] = new int(i);
} Stack<int> s; for(int j = ; j < ; ++j)
{
s.push(a[j]);
} int k = s.size();
int* t = s.top();
s.pop();
if(s.Empty())
{
cout<<"empty"<<endl;
}
s.clear(); for(int i = ; i < ; ++i)
{
delete a[i] ;
} } void funcString()
{
string* str[];
for(int i = ; i < ; ++i)
{
str[i] = new string("a");
} Stack<string> s; for(int j = ; j < ; ++j)
{
s.push(str[j]);
} int k = s.size();
string* t = s.top();
s.pop();
if(s.Empty())
{
cout<<"empty"<<endl;
}
s.clear(); for(int i = ; i < ; ++i)
{
delete str[i] ;
}
} int main ()
{
funcInt();
funcString();
getchar();
return ;
}

测试代码没有输出,可以断点看数据。

之后我又去看了一眼STL对栈的实现,它默认使用deque作为栈的底层实现。

template<class _Ty,
class _Container = deque<_Ty> > class stack
{
// 栈实现
}

调用的时候直接std::stack<int> 即默认使用deque作为栈底层容器。

用户也可以指定其他方式,比如std::stack<int, std::list<int> >, 这样就使用了list作为栈的底层容器。

让我感到awesome的是STL实现的精巧和灵活性,居然是可指定底层的一种实现方法,太精巧了。回头再去看一下《STL源码剖析》。

只罗列了代码,没太详细的介绍原理,感兴趣的可以直接去看《effective C++》和《STL源码剖析》,以及STL的stack代码。

引用请注明出处,http://www.cnblogs.com/chwen/p/4055474.html 非常感谢,随时交流。

C++采用模板实现栈的方法的更多相关文章

  1. java内存管理(堆、栈、方法区)

    java内存管理 简介 首先我们要了解我们为什么要学习java虚拟机的内存管理,不是java的gc垃圾回收机制都帮我们释放了内存了吗?但是在写程序的过程中却也往往因为不懂内存管理而造成了一些不容易察觉 ...

  2. JVM 运行时数据区:程序计数器、Java 虚拟机栈和本地方法栈,方法区、堆以及直接内存

    Java 虚拟机可以看作一台抽象的计算机,如同真实的计算机,它也有自己的指令集和运行时内存区域. Java 虚拟机在执行 Java 程序的过程中会把它所管理的内存(运行时内存区域)划分为若干个不同的数 ...

  3. 单链表的C++实现(采用模板类)

    采用模板类实现的好处是,不用拘泥于特定的数据类型.就像活字印刷术,制定好模板,就可以批量印刷,比手抄要强多少倍! 此处不具体介绍泛型编程,还是着重叙述链表的定义和相关操作.  链表结构定义 定义单链表 ...

  4. (转) ThinkPHP模板自定义标签使用方法

    这篇文章主要介绍了ThinkPHP模板自定义标签使用方法,需要的朋友可以参考下  转之--http://www.jb51.net/article/51584.htm   使用模板标签可以让网站前台开发 ...

  5. Angular 向组件传递模板的几种方法

    最近在写一个日期选择器组件,为了满足将来可能出现的各种需求,所以需要能够高度的自定义组件的样式.为了达到这个目的,需要能够在日期选择器组件外控制每个日期格子内要显示的内容,比如,标上节假日之类的.这时 ...

  6. 简单了解下java中的堆、栈和方法区。

    堆.栈.方法区 1,首先了解下java中的数据类型. ①java中的八大基本数据类型:boolean, char , byte, short, int, long , float , double. ...

  7. JVM内存结构之堆、栈、方法区以及直接内存、堆和栈区别

    JVM内存结构之堆.栈.方法区以及直接内存.堆和栈区别 一.  理解JVM中堆与栈以及方法区 堆(heap):FIFO(队列优先,先进先出):二级缓存:*JVM中只有一个堆区被所有线程所共享:对象和数 ...

  8. ThinkPHP问题收集:模板中使用U方法时无法嵌套大括号,For标签,插入数据,新增的表字段缓存问题

    ThinkPHP模板中使用U方法时无法嵌套大括号需要在control里面用U方法赋值给变量传到模版如:{:U('/Blog/comment/',array('id'=>$id)}$comment ...

  9. JVM,JMM,虚拟机栈,本地方法栈

    JVM 虚拟机栈 本地方法栈:本地方法(使用native关键词修饰的方法,是由JVM底层用C,C++实现的),运行这部份代码使用的栈就是本地方法栈

随机推荐

  1. NTOPNG,用来平时优化网站性能,用处大的

    最近考察一下NTOPNG和NGX-REQ模块,看哪个对网站优化性能用户更大... 参考URL: http://www.68idc.cn/help/jiabenmake/qita/20150109164 ...

  2. android环境下两种md5加密方式

    在平时开发过程中,MD5加密是一个比较常用的算法,最常见的使用场景就是在帐号注册时,用户输入的密码经md5加密后,传输至服务器保存起来.虽然md5加密经常用,但是md5的加密原理我还真说不上来,对md ...

  3. ehci符合USB2.0,uhci,ohci,

    uhci   ohci   ehci他们都是主机控制器的规格,OHCI主要为非PC系统上以及带有SiShe ALi芯片组的 PC主板上的USB芯片,UHCI大多为Intel和Via主板上的USB控制器 ...

  4. 维基百科上—数据仓库、数据挖掘、OLAP三者之间的区别

    数据仓库可以作为数据挖掘和OLAP等分析工具的资料来源,由于存放于数据仓库中的资料,必需经过筛选与转换,因此可以避免分析工具使用错误的资料,而得到不正确的分析结果. 数据挖掘和OLAP同为分析工具,其 ...

  5. 解读sample3

    说明 理解被测试代码 理解测试代码:test fixture简介 编写fixture class TEST_F宏 其他 不应该被忽略的注释 说明 被测试代码文件 sample3-inl.h 测试代码文 ...

  6. C++ 匿名名字空间及静态非成员函数

    在C++中,static有一个感觉被较少提及的用法:修饰非成员函数,这个用法实际是从C语言继承来的.其作用是表明这个函数只在当前编译单元中有效.这就使这个函数的所有引用在编译时就可以全部确定,无需进入 ...

  7. sicily 1119 Factstone Benchmark

    题意:求满足n! < 2^k,n的最大值! 解题:指数比较转换成对数比较,达到降幂! 其中: log (n!) = log(n)+log(n-1)+...+log(1); log(2^k) = ...

  8. 二、Linux文件系统之内存管理

    虚拟内存  32位:4G 64位:2^64 内存管理: 进程管理 自动分配和管理 支持模块化程序设计 保护和访问控制 长期存储 虚拟内存  <---MMU-->物理内存

  9. 写自己的WPF样式 - 窗体

    初试WPF样式,感觉还不错.上篇写完了按钮的样式下面写窗体,废话不多说直接上代码: (1)定义一个窗体样式"MyWpfWindow" <Style x:Key="M ...

  10. 安装redis,含安装步骤和安装中出现的详细错误分析

    1.wget http://download.redis.io/releases/redis-2.8.13.tar.gz 2.解压文件 tar -zxvf redis-2.8.13.tar.gz 3. ...