第十四届蓝桥杯集训——练习解题阶段(无序阶段)-ALGO-7 算法训练 逆序对 平衡二叉树

简介: 第十四届蓝桥杯集训——练习解题阶段(无序阶段)-ALGO-7 算法训练 逆序对 平衡二叉树

第十四届蓝桥杯集训——练习解题阶段(无序阶段)-ALGO-7 算法训练 逆序对 平衡二叉树


前言

       最近的一些文章都可能会很碎,写到哪里是哪里,过一阵子会具体的整理一遍,这里其它的类型题先往后排一排,因为蓝桥最后考的也就是对题目逻辑的理解能力,也就是dp分析能力了,所以就主要目标定在这里,最近的题目会很散,很多,基本上都是网罗全网的一些dp练习题进行二次训练,准备比赛的学生底子薄的先不建议看啊,当然,脑子快的例外,可以直接跳过之前的一切直接来看即可,只需要你在高中的时候数学成绩还可以那就没啥问题,其实,dp就是规律总结,我们只需要推导出对应题目的数学规律就可以直接操作,可能是一维数组,也可能是二维数组,总体来看二维数组的较多,但是如果能降为的话建议降为,因为如果降为起来你看看时间复杂度就知道咋回事了,那么在这里祝大家能无序的各种看明白,争取能帮助到大家。


算法训练 逆序对

资源限制

内存限制:256.0MB   C/C++时间限制:1.0s   Java时间限制:3.0s   Python时间限制:5.0s

问题描述

Alice是一个让人非常愉跃的人!他总是去学习一些他不懂的问题,然后再想出许多稀奇古怪的题目。这几天,Alice又沉浸在逆序对的快乐当中,他已近学会了如何求逆序对对数,动态维护逆序对对数等等题目,他认为把这些题让你做简直是太没追求了,于是,经过一天的思考和完善,Alice终于拿出了一道他认为差不多的题目:

有一颗2n-1个节点的二叉树,它有恰好n个叶子节点,每个节点上写了一个整数。如果将这棵树的所有叶子节点上的数从左到右写下来,便得到一个序列a[1]…a[n]。现在想让这个序列中的逆序对数量最少,但唯一的操作就是选树上一个非叶子节点,将它的左右两颗子树交换。他可以做任意多次这个操作。求在最优方案下,该序列的逆序对数最少有多少。

Alice自己已近想出了题目的正解,他打算拿来和你分享,他要求你在最短的时间内完成。

输入格式

第一行一个整数n。

下面每行,一个数x。

如果x=0,表示这个节点非叶子节点,递归地向下读入其左孩子和右孩子的信息,如果x≠0,表示这个节点是叶子节点,权值为x。

输出格式

输出一个整数,表示最少有多少逆序对。

样例输入

3

0

0

3

1

2

样例输出

1

数据规模与约定

对于20%的数据,n <= 5000。

对于100%的数据,1 <= n <= 200000,0 <= a[i]<2^31

题解:

C语言

#include<stdio.h>
#define N 200010
long long ans = 0; 
int n,val,index=1; 
 
int S[N],left[N],right[N],key[N];
int rRotate(int t){ 
  int tem = left[t];
  left[t] = right[tem];
  right[tem] = t;
  S[t] = S[left[t]]+S[right[t]]+1;
  S[tem] = S[left[tem]]+S[right[tem]]+1;
  return tem;
} 
int lRotate(int t){
  int tem = right[t];
  right[t] = left[tem];
  left[tem] = t;
  S[t] = S[right[t]]+S[left[t]]+1;
  S[tem] = S[right[tem]]+S[left[tem]]+1;
  return tem;
}
int adjust(int t,int flag){
  if(flag){
    if(S[left[left[t]]]>S[right[t]] || S[right[left[t]]]>S[right[t]]){
      if(S[right[left[t]]]>S[right[t]]){
        left[t] = lRotate(left[t]);
      }
      return rRotate(t);
    }
  }else{
    if(S[right[right[t]]]>S[left[t]] || S[left[right[t]]]>S[left[t]]){
      if(S[left[right[t]]]>S[left[t]]){
        right[t] = rRotate(right[t]);
      }
      return lRotate(t);
    }
  }
  return t; 
}
int insert(int t,int node){ 
  S[t]++;
  if(key[node]<key[t]){ 
    if(left[t]==0){ 
      left[t]=node;
    }else{ 
      left[t] = insert(left[t],node);
    }
  }else{
    if(right[t]==0){
      right[t]=node;
    }else{
      right[t] = insert(right[t],node);
    }
  }
  return adjust(t,key[node]<key[t]); 
}
int rank(int t,int val){ 
  if(t==0)return 0; 
  if(val>=key[t]){
    return rank(right[t],val);
  }else{
    return rank(left[t],val)+1+S[right[t]];
  }
}
int merge(int node,int begin,int end){
  long long lens=0,rans=0; 
  int i;
  for(i=begin;i<end;i++){
    int tem = rank(node,key[i]); 
    lens += tem;
    rans += S[node] - tem;
  }
  ans += lens < rans ? lens : rans;
  for(i=begin;i<end;i++){
    left[i]=right[i]=0;
    S[i]=1;
    node = insert(node,i);
  }
  return node;
} 
int buildTree(){
  scanf("%d",&val);
  if(val!=0){
    left[index]=right[index]=0;
    S[index]=1;
    key[index]=val;
    return index++;
  }
  int a = index;
  int lt = buildTree();
  int b = index;
  int rt = buildTree();
  int c = index;
  if(b - a > c - b){ 
    return merge(lt,b,c);
  }else{
    return merge(rt,a,b);
  }
}
int main(){
  scanf("%d",&n);
  buildTree();
  printf("%I64d\n",ans); 
  return 0; 
}

C++语言

#include<stdio.h>
#include<iostream>
using namespace std;
#define ForD(i,n) for(int i=n;i;i--)
#define F (100000007)
#define MAXN (2*200000+10)
long long mul(long long a,long long b){return (a*b)%F;}
long long add(long long a,long long b){return (a+b)%F;}
long long sub(long long a,long long b){return (a-b+(a-b)/F*F+F)%F;}
int n,root=0;/*n->叶子结点数,root->根序号?*/
struct node
{
    int fa;        /*父结点的下标*/
    int ch[2];    /*0->左树下标    1->右树下标*/
    int size;    /*size->当前结点含有的有效结点数*/ 
    int c;        /*当前结点的数值*/
    node():size(0),c(0){ch[0]=ch[1]=fa=0;}    /*初始化为0*/
}a[MAXN];                                    /*树形数组->纪录各叶结点的数值*/
void update(int x)/*更新叶结点个数*/
{
    a[x].size=a[a[x].ch[0]].size+a[a[x].ch[1]].size+(a[x].c>0);
}
int tail=0;
void pushdown(int x)/*将叶结点的父结点指向当前结点*/
{
    a[a[x].ch[0]].fa=a[a[x].ch[1]].fa=x;
}
/*创建树*/
void build(int &x)
{
    if (!x) x=++tail;
    scanf("%d",&a[x].c);
    if (a[x].c==0)
    {
        build(a[x].ch[0]);/*创建左子树*/
        build(a[x].ch[1]);/*创建右子树*/
        update(x);pushdown(x);/*更新当前结点的有效叶结点个数,以及父结点指向*/
    }else a[x].size=1;
}
void rotate(int x)/*旋转*/
{
    int y=a[x].fa,z=a[y].fa;
    bool p=a[y].ch[0]==x;
    if (z)        /*有爷爷*/
    {
        if (a[z].ch[0]==y) /*未旋转*/
            a[z].ch[0]=x;/*将子树提拔为父树(升序)*/
        else 
            a[z].ch[1]=x;    /*还原状态*/
    }
    a[x].fa=z,a[y].fa=x;/*当前结点与父结点交换(父结点指向)*/ 
    if (a[x].ch[p])     /*原子树是否有右树(隔代转移)*/
        a[a[x].ch[p]].fa=y;
    a[y].ch[p^1]=a[x].ch[p];
    a[x].ch[p]=y;        /*父树移至子树的右端(右树)*/
    update(y);    /*更新旋转后,子树的结点的有效结点数*/
}
void splay(int x)
{
    while (a[x].fa)/*不为根结点*/
    {
        int y=a[x].fa,z=a[y].fa;
        if (z)        /*有爷爷*/
            if ((a[y].ch[0]==x)^(a[z].ch[0]==y)) rotate(x);/*旋转*/
            else rotate(y);
        rotate(x);
    }
    update(x);
}
void ins(long long &tot,int x,int y)
{
    a[x].size++;            /*插入+1*/
    if (a[y].c<=a[x].c)        /*是逆序对*/
    {
        if (a[x].ch[0])     /*左树有子*/
            ins(tot,a[x].ch[0],y);
        else                 /*左树无子*/
            a[y].fa=x,splay(a[x].ch[0]=y);/*右数插入到左树子*/
    }
    else
    {
        tot+=a[a[x].ch[0]].size+(a[x].c>0);
        if (a[x].ch[1]) ins(tot,a[x].ch[1],y);
        else a[y].fa=x,splay(a[x].ch[1]=y);
    }
}
int q[MAXN],size;
void clac(int x,int y)
{
    if (a[y].ch[0]) clac(x,a[y].ch[0]);
    if (a[y].c) q[++size]=y;
    if (a[y].ch[1]) clac(x,a[y].ch[1]);
}
long long merge(bool &lor,int z)/*分治*/
{
    int x=a[z].ch[0],y=a[z].ch[1];
    if (a[x].size<a[y].size) /*判断叶结点多的往前调(平衡树?)*/ 
        swap(x,y);
 
    a[x].fa=0;a[y].fa=0;q[1]=y;
    size=0;clac(x,y);
    long long tot=0;    /*最少逆序对*/
    ForD(i,size)        /*循环(子结点数)次*/
    {
        int now=q[i];
        a[now].ch[0]=a[now].ch[1]=a[now].fa=0;
        a[now].size=1;
        ins(tot,x,now);
        x=now;
    }
    a[x].fa=z;
    a[z].ch[0]=0,a[z].ch[1]=x;
    return tot;
}
long long qur(int &x)
{
    if (a[x].c) return 0;/*若根结点没有叶结点,则逆序对为0*/
    else
    {
        long long lson=a[a[x].ch[0]].size,rson=a[a[x].ch[1]].size,ls=qur(a[x].ch[0]),rs=qur(a[x].ch[1]);
        bool lor=0;
        long long ms=merge(lor,x);
        return ls+rs+min(lson*rson-ms,ms);
    }
}
int main()
{
    scanf("%d",&n);
    build(root);
    cout<<qur(root)<<endl;
    return 0;
}

Java语言

import java.util.LinkedList;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.StringTokenizer;
import java.io.*;
public class Main {
    static Node[] tree;
    static int inReturn;
    static int[] noLeaf;
    static Queue<PQItem> inQ;
    public static void main(String[] args) throws IOException {
        InputReader in = new InputReader(System.in);
        int n = in.nextInt(); long cnt = 0;
        noLeaf = new int[n];
        tree = new Node[n*2];
        tree[0] = new Node(0);
        tree[1] = new Node(1);
        build(tree[1], 1, in);
        inQ = new PriorityQueue();
        for (int i = n - 2; i >= 0; i--) cnt += merge(noLeaf[i]);
        System.out.print(cnt);
    }
    static void build(Node node, int offset, InputReader in) throws IOException {
        LinkedList<Node> queue = new LinkedList();
        int index = 0;
        queue.addFirst(node);
        while (queue.size() > 0) {
            Node now = queue.poll();
            noLeaf[index] = now.weight;
            now.weight = in.nextInt();
            if (now.weight == 0) {
                now.left = ++offset;
                queue.addFirst(tree[++offset] = new Node(offset));
                now.right = offset;
                queue.addFirst(tree[offset - 1] = new Node(offset - 1));
                index++;
            } else now.size = 1;
        }
    }
    static long merge(int node) {
        long LS = tree[tree[node].left].size, RS = tree[tree[node].right].size, cnt = 0;
        if (LS > RS) {
            Queue(tree[node].right);
            tree[node] = tree[tree[node].left];
        } else {
            Queue(tree[node].left);
            tree[node] = tree[tree[node].right];
        }
        while (inQ.size() > 0) {
            inReturn = 0;
            insert(node, reSet(inQ.poll().node));
            cnt += inReturn;
        }
        return Math.min(cnt, LS * RS - cnt);
    }
    static void Queue(int node) {
        if (tree[node].left != 0) Queue(tree[node].left);
        if (tree[node].right != 0) Queue(tree[node].right);
        if (tree[node].weight != 0) inQ.offer(new PQItem(tree[node].weight, node));
    }
    static void insert(int node, int value) {
        ++tree[node].size;
        if (tree[node].weight >= tree[value].weight) {
            if (tree[node].left != 0) insert(tree[node].left, value);
            else  tree[node].left = value;
            maintain(node, true);
        } else {
            inReturn += tree[tree[node].left].size + 1;
            if (tree[node].right != 0) insert(tree[node].right, value);
            else tree[node].right = value;
            maintain(node, false);
        }
    }
    static int reSet(int node) {
        tree[node].size = 1;
        tree[node].left = 0;
        tree[node].right = 0;
        return node;
    }
    static void LeftRotate(int node) {
        Node temp = tree[node];
        int r = temp.right;
        tree[r].size = temp.size;
        temp.right = tree[r].left;
        temp.size = tree[temp.left].size + tree[temp.right].size + 1;
        tree[r].left = r;
        tree[node] = tree[r];
        tree[r] = temp;
    }
    static void RightRotate(int node) {
        Node temp = tree[node];
        int l = temp.left;
        tree[l].size = temp.size;
        temp.left = tree[l].right;
        temp.size = tree[temp.left].size + tree[temp.right].size + 1;
        tree[l].right = l;
        tree[node] = tree[l];
        tree[l] = temp;
    }
    static void maintain(int node, boolean flag) {
        if (flag) {
            if (tree[tree[tree[node].left].left].size > tree[tree[node].right].size) RightRotate(node);
            else if (tree[tree[tree[node].left].right].size > tree[tree[node].right].size) {
                LeftRotate(tree[node].left);
                RightRotate(node);
            } else return;
        } else {
            if (tree[tree[tree[node].right].right].size > tree[tree[node].left].size) LeftRotate(node);
            else if (tree[tree[tree[node].right].left].size > tree[tree[node].left].size) {
                RightRotate(tree[node].right);
                LeftRotate(node);
            } else return;
        }
        maintain(tree[node].left, true);
        maintain(tree[node].right, false);
    }
    static class Node {
        int weight;
        int size;
        int left;
        int right;
        Node(int weight) { this.weight = weight; }
    }
    static class PQItem implements Comparable<PQItem> {
        int index;
        int node;
        PQItem (int index, int node) {
            this.node = node;
            this.index = index;
        }
        @Override
        public int compareTo(PQItem o) {
            if (this.index > o.index) return -1;
            return +1;
        }
    }
    static class InputReader {
        BufferedReader read;
        StringTokenizer tokens;
        String delimiters;
        InputReader (InputStream in, String delimiters) {
            this.read = new BufferedReader(new InputStreamReader(in));
            this.delimiters = delimiters;
            this.tokens = new StringTokenizer("", delimiters);
        }
        InputReader (InputStream in) { this(in, " \t\n\r\f"); }
        String next() throws IOException {
            while (!tokens.hasMoreTokens()) tokens = new StringTokenizer(read.readLine(), delimiters);
            return tokens.nextToken();
        }
        int nextInt() throws IOException { return Integer.parseInt(next()); }
        void close() throws IOException { read.close(); }
    }
}

总结

这个题目能看到是对【平衡二叉树】的理解与使用,直接套用即可,这里面没有给定Python的解法。可以自己去填充一下。

没有什么不付出就能拿到的结果,我们都是在负重前行,最终结果与自身先天的脑力有一定的关系,但是还是有很大一部分看自己后天的努力,其实从报名到比赛也就5个月左右,真正刷题的事件也就2个月,2个月回忆一下你真正的认真刷过题吗,如果你真的用尽所有的精力去努力了,那么我相信你最终的成绩一定会让你满意的,加油。


相关文章
|
8月前
|
算法 测试技术 C++
【动态规划】【滑动窗口】【C++算法】 629K 个逆序对数组
【动态规划】【滑动窗口】【C++算法】 629K 个逆序对数组
|
3月前
|
Python
蓝桥杯练习题(一):Python组之入门训练题
这篇文章是关于蓝桥杯Python组的入门训练题,包括Fibonacci数列、圆的面积、序列求和和A+B问题的具体代码实现和样例输出。
159 0
|
3月前
|
存储 机器学习/深度学习 算法
蓝桥杯练习题(三):Python组之算法训练提高综合五十题
蓝桥杯Python编程练习题的集合,涵盖了从基础到提高的多个算法题目及其解答。
136 3
蓝桥杯练习题(三):Python组之算法训练提高综合五十题
|
3月前
|
存储 算法 数据管理
数据结构与算法学习二零:二叉排序树(BST)、平衡二叉树(AVL)
这篇文章通过需求分析、代码实现和测试验证,详细介绍了二叉排序树的创建、遍历和删除操作,以及二叉平衡树(AVL)的自平衡特性和单旋转操作,旨在提高树结构在数据管理中的效率和性能。
68 0
数据结构与算法学习二零:二叉排序树(BST)、平衡二叉树(AVL)
|
3月前
|
算法 Java 程序员
【算法每日一练及解题思路】有n级台阶,一次只能上1级或2级,共有多少种走法?
本文深入解析了“爬楼梯问题”,探讨了递归与迭代两种解法,并提供了Java代码实现。通过分析问题本质,帮助读者理解动态规划技巧,提高解决实际编程问题的能力。关键词:Java, 算法, 动态规划, 爬楼梯问题, 递归, 迭代。
141 0
|
3月前
|
算法 C++
【算法解题思想】动态规划+深度优先搜索(C/C++)
【算法解题思想】动态规划+深度优先搜索(C/C++)
|
7月前
|
算法 C语言
数据结构和算法学习记录——栈和队列习题-用队列实现栈、用栈实现队列(核心思路、解题过程、完整题解)二
数据结构和算法学习记录——栈和队列习题-用队列实现栈、用栈实现队列(核心思路、解题过程、完整题解)二
46 2
|
7月前
|
存储 算法
算法训练,牛客.判断是不是平衡二叉树 牛客.最大子矩阵两个数组的交集牛客.数组中两个字符串的最小距离
算法训练,牛客.判断是不是平衡二叉树 牛客.最大子矩阵两个数组的交集牛客.数组中两个字符串的最小距离
算法训练,牛客.判断是不是平衡二叉树 牛客.最大子矩阵两个数组的交集牛客.数组中两个字符串的最小距离
|
8月前
|
算法 安全
死锁相关知识点以及银行家算法(解题详细步骤)
死锁相关知识点以及银行家算法(解题详细步骤)
367 2
|
7月前
|
算法
数据结构和算法学习记录——平衡二叉树(基本介绍、平衡因子、平衡二叉树的定义、平衡二叉树的高度)
数据结构和算法学习记录——平衡二叉树(基本介绍、平衡因子、平衡二叉树的定义、平衡二叉树的高度)
173 0