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 ...
随机推荐
- SQLSERVER中的资源调控器
SQLSERVER中的资源调控器 转载自: http://wenku.baidu.com/view/0d92380cf78a6529647d5375.html http://www.cnblogs.c ...
- memcached 查看所有的key
1. cmd上登录memcache 1 > telnet 127.0.0.1 11211 2. 列出所有keys 1 2 3 4 stats items // 这条是命令 STAT it ...
- windows10如何打开vhd文件
本人电脑安装了Visual Studio 2017,但是由于项目需求需要Core SDK(2.0)的版本支持,也就是2017最新版.所以现在需要利用visual Studio 2017最新版本的安装包 ...
- Oracle EBS 查看双节点是否做了信任
perl $AD_TOP/patch/115/bin/txkRunSSHSetup.pl verifyssh -contextfile=$CONTEXT_FILE -hosts=erpapp1,erp ...
- MySQL分析函数实现
| MySQL分析函数实现还好MySQL8.0已经实现了与Oracle相同的分析函数. 1. 实现rownumSET @rn:=0;SELECT @rn:=@rn+1 AS rownum ,e.* F ...
- gh-ost和pt-osc性能对比
haughty_xiao 基于MySQL row格式的复制现在趋于主流,因此可以使用此格式的binlog来跟踪改变而不是触发器.与percona toolkit的pt-online-schem ...
- java读取txt文件的2中方法---并将内容(每一行以固定的字符分割切成2段)存到map中去
#java读取txt文件的第一种方法 /** * 方法:readTxt * 功能:读取txt文件并把txt文件的内容---每一行作为一个字符串加入到List中去 * 参数:txt文件的地址 * 返回: ...
- Win7下的C盘重新划分为两个盘
Win 7分盘 注意事项:操作之前,先备份好重要数据,以免误操作导致数据丢失 . 方法步骤如下: 1.在桌面右键点击"计算机"-"管理": 2.鼠标左键单击选& ...
- Linux之因BASH造成的键盘错误和环境问题
对于Linux我们习惯使用/bin/bash.并且大多数人操作在Centos系统上,但是仍有不少人在ubuntu上使用,两个操作系统大同小异.都是使用了Linux内核.接下来就来讲讲我使用过程中两个系 ...
- SDN2017 第三次实验作业
实验目的 在给定如上实验拓扑情况下,用vlan得到下列虚拟网段 h1--h4互通 h2--h5互通 h3--h6互通 其余主机间无法通信 实验步骤 1. 创建拓扑 #! /usr/bin/python ...