stl源码剖析 详细学习笔记priority_queue slist
//
// priority_queue.cpp
// 笔记
//
// Created by fam on 15/3/16.
//
//
//---------------------------15/03/16----------------------------
//priority_queue
{
/*
priority_queue概述:
同正常队列一样,队尾进,队首出,不过不是先进后出,
有权值的概念,所以会自动排序(并不是全部排序,只要保证队
首值最大即可,权值最高的最先出队列,这里用的是堆来保证队首
的值
*/
template<class T,
class Sequence = vector<T>,
class Compare = less<typename Sequence::value_type> >
class priority_queue
{
public:
typedef typename Sequence::value_type value_type;
typedef typename Sequence::size_type size_type;
typedef typename Sequence::reference reference;
typedef typename Sequence::const_reference const_reference;
protected:
Sequence c;
Compare comp;
public:
priority_queue() : c(){};
//explict
可以有效防止隐式转化
explicit priority_queue(const Compare& x) : c(), comp(x){}
//下面都是直接调用heap算法来直接实现堆的操作
template<class InputIterator>
priority_queue(InputIterator first, InputIterator last,
const Compare& x)
:c(first,last), comp(x){ make_heap(c.begin(), c.end(), comp);}
template< class InputIterator>
priority_queue(InputIterator first, InputIterator last)
: c(first, last) {make_heap(c.begin(), c.end(), comp);}
bool empty() const {return c.empty();}
size_type size()
const {return c.size();}
const_reference top()
const {return c.front();}
void push(const value_type& x)
{
__STL_TRY
{
c.push_back(x);
push_heap(c.begin(), c.end(), comp);
}
__STL_UNWIND(c.clear());
}
void pop()
{
__STL_TRY
{
pop_heap(c.begin(), c.end(), comp);
c.pop_back();
}
__STL_UNWIND(c.clear());
}
/*
总结:
同queue
不提供迭代器。
不同queue,底层实现是用vector实现的(queue底层是deque实现的)
原因:queue每次进队列都在队尾,出队列都在队首,频繁地对头尾进行操作,
deque比vector性能好很多
priority_queue虽然也是队尾进,队首出,但是不可避免的是每次都要调整位置,
所以采用堆加vector是很好的选择(deque的随机访问是要比vector慢的),采用
堆每次插入,取出,都只用log(n)的时间,所以很好.
*/
};
}
//slist
{
/*
slist概述:
list是双向链表(double linked list),slist是单向链表
slist的迭代器属于单向的ForwardIterator,所以功能被限制了很多
但是耗用的空间小,操作更快.(我们大部分人最先接触的链表数据结构就是单向链表)
slist不提供push_back(),
*/
//__slist_node
struct __slist_node_base
{
__slist_node_base* next;
};
template <class T>
struct __slist_node :
public __slist_node_base
{
T data;
};
//添加一个节点(new_node)到prev_node之后
inline __slist_node_base* __slist_make_link(
__slist_node_base* prev_node,
__slist_node_base* new_node)
{
new_node->next = prev_node->next;
prev_node->next = new_node;
return new_node;
}
//通过循环判断node节点之后(包括node节点)一共有多少节点
inline size_t __slist_size(__slist_node_base* node)
{
size_t result =
;
; node = node->next)
++result;
return result;
}
//__slist_iterator
struct __slist_iterator_base
{
typedef size_t size_type;
typedef ptrdiff_t difference_type;
typedef forward_iterator_tag iterator_category;
//这里用基类创建一个节点
__slist_node_base* node;
__slist_iterator_base(__slist_node_base* x) :node(x){}
//相当于++操作
void incr() { node = node->next; }
//迭代器是否相等取决于他们的节点是否相等
bool operator==(const __slist_iterator_base& x)
const
{
return node == x.node;
}
bool operator!=(const __slist_iterator_base& x)
const{
return node != x.node;
}
};
//T: class Ref class& Ptr class*
template<class T,
class Ref, class Ptr>
struct __slist_iterator :
public __slist_iterator_base
{
typedef __slist_iterator<T, T&, T*> iterator;
typedef __slist_iterator<T,
const T&, const T*> const_iterator;
typedef __slist_iterator<T, Ref, Ptr> self;
typedef T value_type;
typedef Ptr pointer;
typedef Ref reference;
//节点类型
typedef __slist_node<T> list_node;
__slist_iterator(list_node* x) : __slist_iterator_base(x) {}
__slist_iterator() : __slist_iterator_base(){}
__slist_iterator(const iterator& x) : __slist_iterator_base(x.node) {}
//node是__slist_node_base类型
需要强制转化
reference
operator*() const {return ((list_node*) node)->data;}
pointer
operator->() const {return &(operator*());}
self&
operator++()
{
incr();
return *this;
}
self
operator++(int)
{
self temp = *this;
incr();
return temp;
}
//并没有operator--操作
};
//class slist
template< class T,
class Alloc = allic>
class slist
{
public:
typedef T value_type;
typedef value_type* pointer;
typedef const value_type* const_pointer;
typedef value_type& reference;
typedef const value_type& const_reference;
typedef size_t size_type;
typedef ptrdiff_t difference_type;
typedef __slist_iterator<T, T&, T*> iterator;
typedef __slist_iterator<T,
const T&, const T*> const_iterator;
private:
typedef __slist_node<T> list_node;
typedef __slist_node_base list_node_base;
typedef __slist_iterator_base iterator_base;
typedef simple_alloc<list_node, Alloc> list_node_allocator;
//申请内存并调用构造函数
static list_node* create_node(const value_type& x)
{
list_node* node = list_node_allocator::allocate();
__STL_TRY
{
construct(&node->data, x);
node->next=;
}
__STL_UNWIND(list_node_allocator::deallocate(node));
return node;
}
static void destroy_node(list_node* node)
{
destroy(&node->data);
list_node_allocator::deallocate(node);
}
private:
list_node_base head;
public:
slist() { head.next =
; }
//clear()是循环删除所有节点
~slist() {clear();}
iterator begin() {return iterator((list_node*)head.next);}
//强制转化一个0为iterator类型表示end
//最开始的时候,head.next等于0
表示末尾;
iterator end() {);}
size_type size()
const {return __slist_size(head.next);}
bool empty() const {return head.next ==
;}
void swap(slist& L)
{
list_node_base* tmp =head.next;
head.next = L.head.next;
L.head.next = tmp;
}
public:
reference front() {return ((list_node*) head.next)->data;}
void push_front(const value_type& x)
{
__slist_make_link(&head, create_node(x));
}
void pop_front()
{
list_node* node = (list_node*) head.next;
head.next = node->next;
destroy_node(node);
}
};
/*
总结:
同deque一样,我还是不知道为什么在迭代器中会有self和iterator两个typedef
为什么要特地弄出两个基类?
这样的好处貌似是一个链表可以存放不同的数据类型,但是
并没有用啊。在使用slist<T>来声明定义变量时就确定了类型了。
别的没什么特别的技巧,就是一个很普通的list
*/
}
stl源码剖析 详细学习笔记priority_queue slist的更多相关文章
- stl源码剖析 详细学习笔记 hashtable
//---------------------------15/03/24---------------------------- //hashtable { /* 概述: sgi采用的是开链法完成h ...
- stl源码剖析 详细学习笔记 set map
// // set map.cpp // 笔记 // // Created by fam on 15/3/23. // // //---------------------------15/03 ...
- stl源码剖析 详细学习笔记 RB_tree (1)
// // RB_tree_STL.cpp // 笔记 // // Created by fam on 15/3/21. // // #include "RB_tree_STL.h&q ...
- stl源码剖析 详细学习笔记heap
// // heap.cpp // 笔记 // // Created by fam on 15/3/15. // // //---------------------------15/03/15 ...
- stl源码剖析 详细学习笔记 空间配置器
//---------------------------15/04/05---------------------------- /* 空间配置器概述: 1:new操作包含两个阶段操作 1>调 ...
- stl源码剖析 详细学习笔记 算法(1)
//---------------------------15/03/27---------------------------- //算法 { /* 质变算法:会改变操作对象之值 所有的stl算法都 ...
- stl源码剖析 详细学习笔记 算法总览
//****************************基本算法***************************** /* stl算法总览,不在stl标准规格的sgi专属算法,都以 *加以标 ...
- stl源码剖析 详细学习笔记 RB_tree (2)
//---------------------------15/03/22---------------------------- //一直好奇KeyOfValue是什么,查了下就是一个和仿函数差不多 ...
- stl源码剖析 详细学习笔记stack queue
// // stack.cpp // 笔记 // // Created by fam on 15/3/15. // // //---------------------------15/03/1 ...
随机推荐
- android dev概念快速入门
apk: android将源代码依赖库等经过编译后打包分发的应用. 打包详细过程如下: android-studio安装 由于google被qiang,需要制定proxy,可以使用sock,同时安装完 ...
- EJB2.0 ejb-jar.xml配置文件详解
<ejb-jar><!--EJB发布描述器的根元素,它包含一个可选的EJB-JAR文件的描述,可选显示的名称,可选小图标的文件名,可选的大图标的文件名,必须的所有被包含的enterp ...
- IE8 下面通过滤镜的方式进行图片旋转
首先,为什么我会提出这样的方式来进行操作呢?原因还是需求导致: 在做项目中,有这样一个需求,在进行网页中图片查看的时候,需要对图片的操作有支持旋转和缩放这些操作,看似这样的网上插件有很多,对!但是对于 ...
- spring-bean 版本的问题(报错:org.xml.sax.SAXParseException; lineNumber: 14; columnNumber: 75;)
当XML中配置的xsd是4.0,而引用的包是4以下的spring-bean.jar时,当服务器能连网时没问题,不能连网时,就报以下类似错误: org.xml.sax.SAXParseException ...
- python3.6和pip3安装
CenOS7 安装依赖环境 yum -y install openssl-devel ncurses-devel sqlite-devel readline-devel tk-devel gcc 编译 ...
- OWASP TOP10(2017)
原文链接:https://www.t00ls.net/viewthread.php?from=notice&tid=39385
- 深入浅出SharePoint——Search疑难排除
通过Search log http://richardstk.com/2013/12/23/using-the-sharepoint-2013-search-query-tool-with-searc ...
- JFreeChart绘制XY折线图(工具类设计)
准备用Java写通信的仿真平台作为毕业设计,相比matlab绘图,Java绘图需要自己去写很多工具类,博主在这采用了JFreeChart的开源解决方案,摸索着自己写了一个XY折线图工具类,话不多说贴源 ...
- 使用mpVue开发小程序实战总结
1.图形验证码接口返回base64格式的数据,使用image标签接收不显示问题. 解决方法: 使用wx.base64ToArrayBuffer和wx.arrayBufferToBase64转化一遍数据 ...
- BZOJ2095:[POI2010]Bridges(最大流,欧拉图)
Description YYD为了减肥,他来到了瘦海,这是一个巨大的海,海中有n个小岛,小岛之间有m座桥连接,两个小岛之间不会有两座桥,并且从一个小岛可以到另外任意一个小岛.现在YYD想骑单车从小岛1 ...