//---------------------------15/04/01----------------------------

//inplace_merge(要求有序)

template<class BidirectionalIterator>

inline void inplace_merge(BidirectionalIterator first,

BidirectionalIterator middle,

BidirectionalIterator last)

{

if(first == middle || middle == last)

return;

__inplace_merge_aux(first, middle, last, value_type(first),

distance_type(first));

}

template<class BidirectionalIterator,
class T, class Distance>

inline void __inplace_merge_aux(BidirectionalIterator first,

BidirectionalIterator middle,

BidirectionalIterator last,

T*, Distance*)

{

Distance len1 =
;

distance(first, middle, len1);

Distance len2 =
;

distance(middle, last, len2);

//用来申请暂时的缓冲区

temporary_buffer<BidirectionalIterator, T> buf(first, last);

)

__merge_without_buffer(first, middle, last, len1, len2);

else

__merge_adaptive(first, middle, last, len1, len2,

buf.begin(), Distance(buf.size()));

}

//有缓冲区的情况

template<class BidirectionalIterator,
class Distance, class Pointer>

void __merge_adaptive(BidirectionalIterator first,

BidirectionalIterator middle,

BidirectionalIterator last,

Distance len1, Distance len2,

Pointer buffer, Distance buffer_size)

{

if(len1 <= len2 && len1 <= buffer_size)

{

//先复制[first,middle)区间的元素到缓冲区

Pointer end_buffer = copy(first, middle, buffer);

//调用merge函数,把元素放入first开始的区间,也就是first到last

merge(buffer, end_buffer, middle, last, first);

}

else if(len2 <= buffer_size)

{

Pointer end_buffer = copy(middle, last, buffer);

//从后面开始merge。

__merge_backward(first, middle, buffer, end_buffer, last);

}

else

{

//缓冲区放不下len1
或 len2,需要进行裁剪

//搞不懂为什么前面是lower_bound,后面是upper_bound

//有一种可能是:len1 > len2时,取lower_bound可以使更少的元素换到左边的区间

//len1 < len2时,取upper_bound可以时更少的元素换到右边。

//这么做可以平衡两个区间的元素量

BidirectionalIterator first_cut =first;

BidirectionalIterator second_cut = middle;

Distance len11 =
;

Distance len22 =
;

if(len1 > len2)

{

len11 = len1 /
;

advance(first_cut, len11);

second_cut = lower_bound(middle, last, *first_cut);

distance(middle, second_cut, len22);

}

else

{

len22 = len2 /
;

advance(second_cut, len22);

first_cut = upper_bound(first, middle, *second_cut);

distance(first, first_cut, len11);

}

//到这时区间是这样的 first      first_cut        middle        second_cut      last

//为了让为了merge必须让要merge的区间并起来 
所以把middle到second_cut,换到first_cut的位置就可以了

//rotate后 first      first_cut(原middle)        new_middle(原first_cut)

//          second_cut(原second_cut)      last

BidirectionalIterator new_middle =

__rotate_adaptive(first_cut, middle, second_cut, len1 - len11,

len22, buffer, buffer_size);

__merge_adaptive(first, first_cut, new_middle, len11, len22, buffer,

buffer_size);

__merge_adaptive(new_middle, second_cut, last, len1 -len11,

len2 - len22, buffer, buffer_size);

}

}

template<class BidirectionalIterator1,
class BidirectionalIterator2,

class Distance>

BidirectionalIterator __rotate_adaptive(BidirectionalIterator1 first,

BidirectionalIterator1 middle,

BidirectionalIterator1 last,

Distance len1, Distance len2,

BidirectionalIterator2 buffer,

Distance buffer_size)

{

//缓冲区足够
就利用缓冲区翻转,不够就调用全局rotate

BidirectionalIterator2 buffer_end;

if(len1 > len2 && len2 <= buffer_size)

{

buffer_end = copy(middle, last, buffer);

copy_backward(first, middle, last);

return copy(buffer, buffer_end, first);

}

else if(len1 <= buffer_size)

{

buffer_end = copy(first, middle, buffer);

copy(middle, last, first);

return copy_backward(buffer, buffer_end, last);

}

else

{

rotate(first, middle, last);

advance(first, len2);

return first;

}

}

//nth_element

//使的nth的元素处在完全排序后的位置,只能保证这一个元素处在对的位置

template<class RandomAccessIterator>

inline void nth_element(RandomAccessIterator first,

RandomAccessIterator nth,

RandomAccessIterator last)

{

__nth_element(first, nth, last, value_type(first));

}

template<class RandomAccessIterator,
class T>

void __nth_element(RandomAccessIterator first,

RandomAccessIterator nth,

RandomAccessIterator last, T*)

{

)

{

//划分一次

RandomAccessIterator cut = __unguarded_partition

(first, last, T(__median(*first,

*(first + (last - first) /
),

*(last -
))));

//nth处在哪边就继续划分那边

if(cut <= nth)

first = cut;

else

last = cut;

}

__insertion_sort(first, last);

}

//merge sort

template<class BidirectionalIterator>

void mergesort(BidirectionalIterator first, BidirectionalIterator last)

{

typename iterator_traits<BidirectionalIterator>::difference_type n

=distance(first, last);

|| n ==
)

return;

else

{

BidirectionalIterator mid = first + n /
;

mergesort(first, mid);

mergesort(mid, last);

inplace_merge(first, mid, last);

}

}


stl源码剖析 详细学习笔记 算法(5)的更多相关文章

  1. stl源码剖析 详细学习笔记 算法(1)

    //---------------------------15/03/27---------------------------- //算法 { /* 质变算法:会改变操作对象之值 所有的stl算法都 ...

  2. stl源码剖析 详细学习笔记 算法总览

    //****************************基本算法***************************** /* stl算法总览,不在stl标准规格的sgi专属算法,都以 *加以标 ...

  3. stl源码剖析 详细学习笔记 算法(2)

    //---------------------------15/03/29---------------------------- //****************************set相 ...

  4. stl源码剖析 详细学习笔记 算法(4)

    //---------------------------15/03/31---------------------------- //lower_bound(要求有序) template<cl ...

  5. stl源码剖析 详细学习笔记 算法(3)

    //---------------------------15/03/30---------------------------- //min_element template<class Fo ...

  6. stl源码剖析 详细学习笔记 hashtable

    //---------------------------15/03/24---------------------------- //hashtable { /* 概述: sgi采用的是开链法完成h ...

  7. stl源码剖析 详细学习笔记 set map

    // //  set map.cpp //  笔记 // //  Created by fam on 15/3/23. // // //---------------------------15/03 ...

  8. stl源码剖析 详细学习笔记 RB_tree (2)

    //---------------------------15/03/22---------------------------- //一直好奇KeyOfValue是什么,查了下就是一个和仿函数差不多 ...

  9. stl源码剖析 详细学习笔记 RB_tree (1)

    // //  RB_tree_STL.cpp //  笔记 // //  Created by fam on 15/3/21. // // #include "RB_tree_STL.h&q ...

随机推荐

  1. SQLSERVER中的元数据锁

    SQLSERVER中的元数据锁 网上对于元数据锁的资料真的非常少 元数据锁一般会出现在DDL语句里 下面列出数据库引擎可以锁定的资源 资源 说明 RID 用于锁定堆(heap)中的某一行 KEY 用于 ...

  2. Android应用耗电量统计,无需USB连接

    Android应用耗电量统计一直是一个很头疼的问题,手工统计耗时太长,自动化统计又不是非常精准(执行自动化代码需要通过USB连接,而USB又会充电,这就造成统计数据不准).后来从前辈那里得知可以通过a ...

  3. Oracle 当数据库的表没有drop操作就可以通过如下方式恢复表数据

    --执行下列语句可查询出相关时间点 select * from sys.smon_scn_time order by time_dp desc; --执行下列语句可将某个时间点的数据恢复 insert ...

  4. 使用postman给servlet传各种参数

    web开发中经常会使用到postman软件,常用的方法涉及到get和post方法去获取对应json数据,get方法直接传url就可以,返回对应json数据.但是post请求就需要json数据提交,而且 ...

  5. 接口调用,输出结果为Json格式(ConvertTo-Json),提交参数给URL(WebRequest)

    1.直接输出为json格式: Get-Process -Id $pid | ConvertTo-Json | clip.exe 2.自定义结果为json格式: $serverinfoj = @&quo ...

  6. Python学习--- requests库中文编码问题

    为什么会有ISO-8859-1这样的字符集编码 requests会从服务器返回的响应头的 Content-Type 去获取字符集编码,如果content-type有charset字段那么request ...

  7. FinalShell使用---Xshell的良心国产软件

    最近发现了一款同类产品FinalShell,还是一块良心国货.初步体验了一下,确实是良心之作.且免费(通用版),支持国货. FinalShell是一体化的的服务器,网络管理软件,不仅是ssh客户端,还 ...

  8. priority_queue的优先级变化(结构体的写法)

    priority_queue的优先级变化(结构体的写法) 在头文件中加上#include <queue> 即可使用stl中的库函数priority_queue,优先队列默认的是从大到小的优 ...

  9. File API

    ES5 推出了一系列的 API: BLOB (二进制大对象) File (文件接口,基于 BLOB,但是增加了文件相关的方法,比如路径,大小) FileList (借助 <input type= ...

  10. Volley源码分析(一)RequestQueue分析

    Volley源码分析 虽然在2017年,volley已经是一个逐渐被淘汰的框架,但其代码短小精悍,网络架构设计巧妙,还是有很多值得学习的地方. 第一篇文章,分析了请求队列的代码,请求队列也是我们使用V ...