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

2023-02-16 16:06:24 浏览数 (1)

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


目录

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

前言

算法训练 逆序对

C语言

C 语言

Java语言

总结

第六届——第十三届省赛题解

第六届——第十二届省赛题解


前言

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


算法训练 逆序对

资源限制

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

问题描述

Alice是一个让人非常愉跃的人!他总是去学习一些他不懂的问题,然后再想出许多稀奇古怪的题目。这几天,Alice又沉浸在逆序对的快乐当中,他已近学会了如何求逆序对对数,动态维护逆序对对数等等题目,他认为把这些题让你做简直是太没追求了,于是,经过一天的思考和完善,Alice终于拿出了一道他认为差不多的题目: 有一颗2n-1个节点的二叉树,它有恰好n个叶子节点,每个节点上写了一个整数。如果将这棵树的所有叶子节点上的数从左到右写下来,便得到一个序列a1…an。现在想让这个序列中的逆序对数量最少,但唯一的操作就是选树上一个非叶子节点,将它的左右两颗子树交换。他可以做任意多次这个操作。求在最优方案下,该序列的逆序对数最少有多少。 Alice自己已近想出了题目的正解,他打算拿来和你分享,他要求你在最短的时间内完成。

输入格式

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

输出格式

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

样例输入

3 0 0 3 1 2

样例输出

1

数据规模与约定

对于20%的数据,n <= 5000。 对于100%的数据,1 <= n <= 200000,0 <= ai<2^31

题解:

C语言

代码语言:javascript复制
#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("%I64dn",ans); 
	return 0; 
}

C 语言

代码语言:javascript复制
#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语言

代码语言:javascript复制
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, " tnrf"); }

        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个月回忆一下你真正的认真刷过题吗,如果你真的用尽所有的精力去努力了,那么我相信你最终的成绩一定会让你满意的,加油。

第六届——第十三届省赛题解

所有的题目都做了讲解,最难的有配套的视频,视频提供者是【2020级的弓家宜】先生。

第六届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/123284163

第七届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/123285783

第八届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/123302677

第九届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/123303285

第十届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/123319090

第十一届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/123320205

第十二届Java省赛C组第一套

https://laoshifu.blog.csdn.net/article/details/123413141

第十二届Java省赛C组第二套

https://laoshifu.blog.csdn.net/article/details/123413271

第十三届Java省赛C组

https://laoshifu.blog.csdn.net/article/details/128891276

第六届——第十二届省赛题解

所有题目均有题解,部分第10题非最优解,至少跑20%数据。

第六届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123440705

第七届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123442982

第八届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123443626

第九届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123443908

第十届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123444946

第十一届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123445601

第十二届Java国赛C组

https://laoshifu.blog.csdn.net/article/details/123446589

0 人点赞