博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
winner tree 胜者树
阅读量:6147 次
发布时间:2019-06-21

本文共 3589 字,大约阅读时间需要 11 分钟。

在树形选择排序中,利用锦标赛思想建立的树称为胜者树。

1、每个非终端节点存储的是左右孩子节点中的优胜者

2、通过减少比较次数,提高效率

3、胜者树就是一颗特殊的线段树

 

 

一、构建树

Procedure buildmint;  复杂度 O(n)var i,j,k:longint;begin  i:=N; j:=N+n-1; {
其中 N 为大于等于 n 的最小 2 次方数 }  while i>1 do  begin    k:=i;    while k

 

 

二、查询

核心思想:自下向上,奇偶缩进。

Function findmin(x,y:longint):longint;  复杂度 O(logn)var i,j,k,r:longint;begin  i:=x+N-1; j:=y+N-1; r:=i;  while i<=j do  begin    if (f[r]>f[i]) then r:=i;    if (f[r]>f[j]) then r:=j;    if (i and 1=1) then inc(i);    if (j and 1=0) then dec(j);    i:=i shr 1; j:=j shr 1;  end;  exit(f[r]);  while r

 

 

三、更新

核心思想:自下而上,更新父节点。

胜者树不擅长更新,因为,某个结点更新的时候,它还需要跟兄弟结点比较。更新频繁的情况,用败者树

 

四、败者树

败者树是胜者树的一种变体。在败者树中,用父结点记录其左右子结点进行比赛的败者,而让胜者参加下一轮的比赛。败者树的根结点记录的是败者,需要加一个结点来记录整个比赛的胜利者。采用败者树可以简化重构的过程。

典型应用:多路归并排序,可以减少比较次数

 

 

五、代码实现胜者树

poj2328 

Description

An array of size 
n ≤ 10
6 is given to you. There is a sliding window of size 
k which is moving from the very left of the array to the very right. You can only see the 
k numbers in the window. Each time the sliding window moves rightwards by one position. Following is an example: 
The array is 
[1 3 -1 -3 5 3 6 7], and 
k is 3.
Window position Minimum value Maximum value
[1  3  -1] -3  5  3  6  7  -1 3
 1 [3  -1  -3] 5  3  6  7  -3 3
 1  3 [-1  -3  5] 3  6  7  -3 5
 1  3  -1 [-3  5  3] 6  7  -3 5
 1  3  -1  -3 [5  3  6] 7  3 6
 1  3  -1  -3  5 [3  6  7] 3 7

Your task is to determine the maximum and minimum values in the sliding window at each position. 

Input

The input consists of two lines. The first line contains two integers 
n and 
k which are the lengths of the array and the sliding window. There are 
n integers in the second line. 

Output

There are two lines in the output. The first line gives the minimum values in the window at each position, from left to right, respectively. The second line gives the maximum values. 

Sample Input

8 31 3 -1 -3 5 3 6 7

Sample Output

-1 -3 -3 -3 3 33 3 5 5 6 7

Source

, Ikki
#include 
#include
using namespace std;const int MX = 1000011;int n, k;int a[MX], minRes[MX], maxRes[MX];int treeMin[5*MX], treeMax[5*MX];int N;//构建一颗winner tree,tree用数组存储,根结点是数组的第二个结点,即A[1]//叶子结点是原始数组。树的最左结点的下标是2的指数void build() { int i = N, j = N+n-1; while (i > 1) { int t = i; while (t < j) { treeMin[t/2] = min(treeMin[t], treeMin[t+1]); treeMax[t/2] = max(treeMax[t], treeMax[t+1]); t += 2; } if ((j&1) == 0) {
//下标j是偶数,说明这个结点是落单的 treeMin[j/2] = treeMin[j]; treeMax[j/2] = treeMax[j]; } i = i>>1; j = j>>1; }}int query(int x, int y) { int i=x+N, j=y+N, t = i; while (i <= j) { if (treeMin[t] > treeMin[i]) t = i; if (treeMin[t] > treeMin[j]) t = j; if ((i&1) == 1) i++; if ((j&1) == 0) j--; i = i>>1; j = j>>1; } //return treeMin[t]; while (t < N) { if (treeMin[t] == treeMin[t*2]) t = t*2; else t = t*2+1; } return treeMin[t]; //return t - N;}void work() { for (int i=0; i
treeMin[x]) t = x; if (treeMin[t] > treeMin[y]) t = y; if (treeMax[r] < treeMax[x]) r = x; if (treeMax[r] < treeMax[y]) r = y; if ((x&1) == 1) x++; if ((y&1) == 0) y--; x = x>>1; y = y>>1; } minRes[i] = treeMin[t]; maxRes[i] = treeMax[r]; }}void printRes() { for (int i=0; i
> n >> k; N = calN(n); for (int i=0; i
> a[i]; treeMax[N+i] = treeMin[N+i] = a[i]; } build(); //printTree(); work(); printRes(); //cout<<"query:"<< query(0, 2)<

 

 

转载地址:http://nelya.baihongyu.com/

你可能感兴趣的文章
舍弃Python,为什么知乎选用Go重构推荐系统?
查看>>
在soapui上踩过的坑
查看>>
MySQL的字符集和字符编码笔记
查看>>
ntpd同步时间
查看>>
must implement java.io.Serializable hessian
查看>>
Microsoft Licenses Flash Lite for Windows Mobile Users
查看>>
HDOJ 2020 绝对值排序
查看>>
HDOJ/HDU 2560 Buildings(嗯~水题)
查看>>
Maven编译时跳过Test
查看>>
Spring Boot 整合Spring Security 和Swagger2 遇到的问题小结
查看>>
[20170628]12C ORA-54032.txt
查看>>
除以2
查看>>
高可用集群原理解析
查看>>
Nginx配置URL转向tomcat
查看>>
极客Web前端开发资源大荟萃#001
查看>>
让div固定在某个位置
查看>>
Java开发环境Docker镜像
查看>>
从无到有,WebService Apache Axis2初步实践
查看>>
任务调度(一)——jdk自带的Timer
查看>>
UIKit框架(15)PCH头文件
查看>>