[蓝桥杯][算法提高VIP]上帝造题五分钟 (线段树+区间最小值)

蓝桥杯历年真题题目及题解目录汇总 

 

题目描述

第一分钟,上帝说:要有题。于是就有了L,Y,M,C
第二分钟,LYC说:要有向量。于是就有了长度为n写满随机整数的向量
第三分钟,YUHCH说:要有查询。于是就有了Q个查询,查询向量的一段区间内元素的最小值
第四分钟,MZC说:要有限。于是就有了数据范围
第五分钟,CS说:要有做题的。说完众神一哄而散,留你来收拾此题

输入

第一行两个正整数n和Q,表示向量长度和查询个数
接下来一行n个整数,依次对应向量中元素:a[0],a[1],…,a[n-1]
接下来Q行,每行两个正整数lo,hi,表示查询区间[lo, hi]中的最小值,即min(a[lo],a[lo+1],…,a[hi])。

输出

共Q行,依次对应每个查询的结果,即向量在对应查询区间中的最小值。

样例输入

7 4
1 -1 -4 8 1 2 -7
0 0
1 3
4 5
0 6

样例输出

1
-4
1
-7

提示

第一个查询[0,0]表示求min{a[0]}=min{1}=1
第二个查询[1,3]表示求min{a[1],a[2],a[3]}=min{-1,-4,8}=-4
第三个查询[4,5]表示求min{a[4],a[5]}=min{1,2}=1
第四个查询[0,6]表示查询整个向量,求min{a[0..6]}=min{1,-1,-4,8,1,2,-7}=-7

 

没什么好解释啦,目前还处于线段树入门阶段,这题只需要建树+查询就好,没涉及更新

import java.util.Scanner;

public class 蓝桥杯_上帝造题五分钟 {

	public static void main(String[] args) {
		Scanner in = new Scanner(System.in);
		n = in.nextInt();
		m = in.nextInt();
		a = new int[n+1];
		tree = new Segtree[4*n+5];
		for(int i=1;i<=n;i++)
			a[i] = in.nextInt();
		for(int i=1;i<=4*n+4;i++)
			tree[i] = new Segtree();
		buildtree(1, 1, n);
		for(int i=1;i<=m;i++)
			System.out.println(query(1, 1, n, in.nextInt()+1, in.nextInt()+1));//注意题目给的是0开头的下标
	}
	
	static int n,m;
	static Segtree[] tree;
	static int[] a;
	
	static class Segtree {
		int min,l,r;
		public Segtree(int l,int r) {
			this.l = l;
			this.r = r;
		}
		public Segtree() {}
	}
	
	static void buildtree(int now,int l,int r) {
		tree[now].l = l;
		tree[now].r = r;
		if(l==r) {
			tree[now].min = a[l];
			return;
		}
		int mid = l + (r-l)/2;
		buildtree(2*now, l, mid);
		buildtree(2*now+1, mid+1, r);
		tree[now].min = Math.min(tree[2*now].min, tree[2*now+1].min);
	}
	
	static int query(int now,int l,int r,int x,int y) {
		if(y<l || x>r)
			return Integer.MAX_VALUE;
		if(x<=l && y>=r) {
			return tree[now].min;
		}
		
		int mid = l + (r-l)/2;
		return Math.min(query(2*now, l, mid, x, y),query(2*now+1, mid+1, r, x, y));
	}
	
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值