hdu线段树专题训练


单点更新:

这是线段树中最基本的类型,只更新叶子节点,然后把信息用PushUP(int r)这个函数更新上来。

hdu 1166   敌兵布阵

代码如下

#include<iostream>
#include<stdlib.h>
#include<cstdio>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
const int maxn=55555;
int sum[maxn<<2];
void PushUP(int rt)
{
    sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}
void build(int l,int r,int rt )
{
    if(l==r)
    {
       scanf("%d",&sum[rt]);
       return;
    }
    int m=(l+r)>>1;
    build(lson);
    build(rson);
    PushUP(rt);
}
void update(int p,int add,int l,int r,int rt)
{
    if(l==r)
    {
        sum[rt]=sum[rt]+add;
        return;
    }
    int m=(l+r)>>1;
    if(p<=m)
        update(p,add,lson);
    else
        update(p,add,rson);
    PushUP(rt);
}
int query(int L,int R,int l,int r,int rt)
{
    if(L<=l&&r<=R)
        return sum[rt];
    int m=(l+r)>>1;
    int ret=0;
    if(L<=m)
        ret=ret+query(L,R,lson);
    if(R>m)
    ret=ret+query(L,R,rson);
    return ret;
}

int main(void)
{
    int T,n,t=1;
    int i,j;
    scanf("%d",&T);
    for(t=1;t<=T;t++)
    {   printf("Case %d:\n",t);
        scanf("%d",&n);
        build(1,n,1);
        char s[10];
        while(scanf("%s",s))
        {
            
            if(s[0]=='E')
                break;
            int a,b;
            scanf("%d%d",&a,&b);
            if(s[0]=='Q')
                printf("%d\n",query(a,b,1,n,1));
            else if(s[0]=='S')
                update(a,-b,1,n,1);
            else 
                update(a,b,1,n,1);
        }
        
    }
    system("Pause");
    return 0;
}

 
 

hdu 1754 I Hate It
 代码如下:

#include<iostream>
#include<cstdio>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
const int maxn=222222;
int MAX[maxn<<2];
void PushUP(int rt)
{
	MAX[rt]=max(MAX[rt<<1],MAX[rt<<1|1]);
}
void build(int l,int r,int rt)
{
	if(l==r)
	{
		scanf("%d",&MAX[rt]);
		return;
	}
	int m=(l+r)>>1;
	build(lson);
	build(rson);
	PushUP(rt);
}
void update(int p,int add,int l,int r,int rt)
{
	if(l==r)
	{
		MAX[rt]=add;
		return;
	}
	int m=(l+r)>>1;
	if(p<=m)
		update(p,add,lson);
	else
		update(p,add,rson);
	PushUP(rt);
}
int query(int L,int R,int l,int r,int rt)
{
	if(L<=l&&r<=R)
		return MAX[rt];
	int m=(l+r)>>1;
	int ret=0;
	if(L<=m)
	 ret=max(ret,query(L,R,lson));
	if(R>m)
		ret=max(ret,query(L,R,rson));
	return ret;
}
int main(void)
{
	int N,M;
	while(scanf("%d%d",&N,&M)!=EOF)
	{
		build(1,N,1);
		char ch[2];
		int a,b;
		for(int i=1;i<=M;i++)
		{
			scanf("%s",ch);
			scanf("%d%d",&a,&b);
			if(ch[0]=='Q')
			printf("%d\n",query(a,b,1,N,1));
			if(ch[0]=='U')
				update(a,b,1,N,1);
		}
	}
	return 0;
}


 hdu  1394 Minimum Inversion Number

题意:求Inversion后的最小逆序数
思路:用O(nlogn)复杂度求出最初逆序数后,就可以用O(1)的复杂度分别递推出其他解
线段树功能:update:单点增减 query:区间求和

代码如下:

#include<iostream>
#include<cstdio>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
const int maxn=5500;
int sum[maxn<<2];
void PushUP(int rt)
{
	sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}
void build(int l,int r,int rt)
{
	if(l==r)
	{
		sum[rt]=0;
		return;
	}
	int m=(l+r)>>1;
	build(lson);
	build(rson);
	PushUP(rt);
}
void update(int p,int l,int r,int rt)
{
	if(l==r)
	{
		sum[rt]++;
		return ;
	}
	int m=(l+r)>>1;
	if(p<=m)
		update(p,lson);
	else
		update(p,rson);
	PushUP(rt);
}
int query(int L,int R,int l,int r,int rt)
{
	if(L<=l&&r<=R)
	 return sum[rt];
	int m=(l+r)>>1;
	int ret=0;
	if(L<=m)
		ret=ret+query(L,R,lson);
	if(R>=m)
		ret=ret+query(L,R,rson);
	return ret;
}
int main(void)
{
	int i,n;
	while(scanf("%d",&n)!=EOF)
	{
		build(0,n-1,1);
		int a[maxn],sum=0;
		for(i=0;i<n;i++)
		{
			scanf("%d",&a[i]);
			sum=sum+query(a[i],n-1,0,n-1,1);
		    update(a[i],0,n-1,1);
		}
		int ret=sum;
		for(i=0;i<n;i++)
		{
			sum+=n-a[i]-a[i]-1;
			ret=min(ret,sum);
		}
		printf("%d\n",ret);
	}
	return 0;
}


hdu 2795 BillBoard

代码如下:

#include <cstdio>
#include <algorithm>
using namespace std;
 
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
const int maxn = 222222;
int h , w , n;
int MAX[maxn<<2];
void PushUP(int rt) {
    MAX[rt] = max(MAX[rt<<1] , MAX[rt<<1|1]);
}
void build(int l,int r,int rt) {
    MAX[rt] = w;
    if (l == r) return ;
    int m = (l + r) >> 1;
    build(lson);
    build(rson);
}
int query(int x,int l,int r,int rt) {
    if (l == r) {
        MAX[rt] -= x;
        return l;
    }
    int m = (l + r) >> 1;
    int ret = (MAX[rt<<1] >= x) ? query(x , lson) : query(x , rson);
    PushUP(rt);
    return ret;
}
int main() {
    while (~scanf("%d%d%d",&h,&w,&n)) {
        if (h > n) h = n;
        build(1 , h , 1);
        while (n --) {
            int x;
            scanf("%d",&x);
            if (MAX[1] < x) puts("-1");
            else printf("%d\n",query(x , 1 , h , 1));
        }
    }
    return 0;
}

poj 2828 Buy Tickets

题目大意:

这道题是从最后一个数开始建立线段树,区间存储的范围是该区间有多少个空位剩余,对于最后一个插入的数,它肯定和其插入的位置一致,例如对于测试数据的第一组,69插入2这个位置,是肯定的,它将前面占有2位置的数向后面推,同理对于倒数第二个数,占有1这个位置,如果前面有数占有1这个位置,此时前面的必定是向后面推。假设P为插入的位置,因次每次查看左区间有没有大于等于p的空位,如果有则搜索,否者搜索右孩子。

代码如下:

#include<iostream>
#include<cstdio>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
const int maxn=200000;
int sum[maxn<<2];
int n,cur,ans[maxn];
void PushUP(int rt)
{
	sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}
void build(int l,int r,int rt)
{
	if(l==r)
	{
		sum[rt]=1;
		return ;
	}
	int m=(l+r)>>1;
	build(lson);
	build(rson);
	PushUP(rt);
}
void update(int L,int R,int l,int r,int rt)
{
	if(l==r)
	{
		sum[rt]--;
		ans[l]=cur;
		return ;
	}
	int m=(l+r)>>1;
	if(L<=sum[rt<<1])
		update(L,R,lson);
	else
	{
		L-=sum[rt<<1];
		update(L,R,rson);
	}
	PushUP(rt);
}

struct node
{
	int p,v;
}queue[maxn];
int main(void)
{
	int i,j;
	while(scanf("%d",&n)!=EOF)
	{
		build(1,n,1);
		for(i=1;i<=n;i++)
		scanf("%d%d",&queue[i].p,&queue[i].v);
		for(i=n;i>0;i--)
		{
			cur=queue[i].v;
			update(queue[i].p+1,n,1,n,1);
		}
		for(i=1;i<=n;i++)
		{
			if(i-1)
				printf(" ");
				printf("%d",ans[i]);
		}
		printf("\n");
	}
	return 0;
}



 



 


 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于计算机专业的学生而言,参加各类比赛能够带来多方面的益处,具体包括但不限于以下几点: 技能提升: 参与比赛促使学生深入学习和掌握计算机领域的专业知识与技能,如编程语言、算法设计、软件工程、网络安全等。 比赛通常涉及实际问题的解决,有助于将理论知识应用于实践中,增强问题解决能力。 实践经验: 大多数比赛都要求参赛者设计并实现解决方案,这提供了宝贵的动手操作机会,有助于积累项目经验。 实践经验对于计算机专业的学生尤为重要,因为雇主往往更青睐有实际项目背景的候选人。 团队合作: 许多比赛鼓励团队协作,这有助于培养学生的团队精神、沟通技巧和领导能力。 团队合作还能促进学生之间的知识共享和思维碰撞,有助于形成更全面的解决方案。 职业发展: 获奖经历可以显著增强简历的吸引力,为求职或继续深造提供有力支持。 某些比赛可能直接与企业合作,提供实习、工作机会或奖学金,为学生的职业生涯打开更多门路。 网络拓展: 比赛是结识同行业人才的好机会,可以帮助学生建立行业联系,这对于未来的职业发展非常重要。 奖金与荣誉: 许多比赛提供奖金或奖品,这不仅能给予学生经济上的奖励,还能增强其成就感和自信心。 荣誉证书或奖状可以证明学生的成就,对个人品牌建设有积极作用。 创新与研究: 参加比赛可以激发学生的创新思维,推动科研项目的开展,有时甚至能促成学术论文的发表。 个人成长: 在准备和参加比赛的过程中,学生将面临压力与挑战,这有助于培养良好的心理素质和抗压能力。 自我挑战和克服困难的经历对个人成长有着深远的影响。 综上所述,参加计算机领域的比赛对于学生来说是一个全面发展的平台,不仅可以提升专业技能,还能增强团队协作、沟通、解决问题的能力,并为未来的职业生涯奠定坚实的基础。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值