蓝桥杯《操作格子》

  解题过程:题意通俗易懂,看完题目以为是普通的对数组进行操作的题目,但是看到输入的数据范围后,发现并不是。但是因为没有好的想法,只能用最简单的方式做,毫无意外的超时了。然后找到这篇博文:http://blog.csdn.net/qq_33245342/article/details/54892576

问题描述 

有n个格子,从左到右放成一排,编号为1-n。 
共有m次操作,有3种操作类型: 
1.修改一个格子的权值, 
2.求连续一段格子权值和, 
3.求连续一段格子的最大值。 
对于每个2、3操作输出你所求出的结果。

输入格式 
第一行2个整数n,m。 
接下来一行n个整数表示n个格子的初始权值。 
接下来m行,每行3个整数p,x,y,p表示操作类型,p=1时表示修改格子x的权值为y,p=2时表示求区间[x,y]内格子权值和,p=3时表示求区间[x,y]内格子最大的权值。

输出格式 
有若干行,行数等于p=2或3的操作总数。 
每行1个整数,对应了每个p=2或3操作的结果。

样例输入 
4  3 
1  2  3  4 
2  1  3 
1  4  3 
3  1  4

样例输出 

3

数据规模与约定 
对于20%的数据n  < =  100,m  < =  200。 
对于50%的数据n  < =  5000,m  < =  5000。 
对于100%的数据1  < =  n  < =  100000,m  < =  100000,0  < =  格子权值  < =  10000。

细节:

  1. 关于树节点的顺序排号:某个节点的子节点序号=自身节点*2(+1右子节点才需要)

  2. 关于区间的控制

  每个几点都有两个int变量存储当前节点所存储的数据表示是与这一区间相关。

//  操作格子
#include<stdio.h>
#define Num 100000

// 创建树的结点
struct Node{
    // [left, right]    记录这个结点和这个结点的子节点(如果有)的取值范围
    // sum记录当前结点的取值范围的总和
    // max记录当前结点的最大值
    int left,right,sum,max;
}Tree[Num*3];

int length=0;

// 初始化树
// p    是当前的位置
// left 是当前结点要存储的左值
// right 是当前结点要存储的右值
void SetTree(int p, int left, int right){
    // 记录结点的个数
    length++;

    if(left==right){
        Tree[p].left=left;
        Tree[p].right=left;
        Tree[p].sum=left;
        Tree[p].max=left;
        return;
    }
    Tree[p].left=left;
    Tree[p].right=right;

    // 右边的子树    [left, right/2]
    int mid = (left+right)>>1;
    SetTree(p*2,left,(mid));
    // 左边的子树    [right/2+1, right];
    SetTree(p*2+1,mid+1,right);

    Tree[p].sum = Tree[p*2].sum+Tree[p*2+1].sum;
    Tree[p].max = Tree[p*2].max>Tree[p*2+1].max?Tree[p*2].max:Tree[p*2+1].max;

    return;
}

void Change(int p, int x, int y){
    if(Tree[p].left==Tree[p].right){
        // 不改变左右值,因为左右值用来做位置的标记
        // Tree[p].left=y;
        // Tree[p].right=y;
        Tree[p].sum=y;
        Tree[p].max=y;
        return;
    }
    int mid=Tree[p*2].right;
    if(mid>=x)  Change(p*2, x, y);
    else Change(p*2+1,x,y);

    Tree[p].sum = Tree[p*2].sum+Tree[p*2+1].sum;
    Tree[p].max = Tree[p*2].max>Tree[p*2+1].max?Tree[p*2].max:Tree[p*2+1].max;
    return;
}

int Find(int p, int x, int y, int choose){
    if(Tree[p].left==x && Tree[p].right==y){
        if(choose==2) return Tree[p].sum;
        else return Tree[p].max;
    }
    int mid=Tree[p*2].right;
    if(mid>=y)  return(Find(p*2, x, y, choose));
    else if(mid<x) return(Find(p*2+1, x, y, choose));

    // 考虑选择的区间刚好不是先有的结点,需要额外的处理
    int le = Find(p * 2, x, mid,choose);
    int ri = Find(p * 2 + 1, mid + 1, y,choose);  

    if(choose==2)   return le+ri;
    else return (le>ri?le:ri);
}

int main(void){
    // nc   =   numcount
    // cc   =   changecount
    int nc,cc;
    int r1=1,num;

    scanf("%d%d",&nc,&cc);
    SetTree(1,1,nc);

    // [0,0]
    // 检查区间
    // for(;r1<=length;r1++) printf("[%2d,%2d]\n",Tree[r1].left,Tree[r1].right);
    // return 0;

    // 1.修改一个格子的权值
    // 2.求连续一段格子权值和
    // 3.求连续一段格子的最大值
    for(r1=1;r1<=nc;r1++){
        scanf("%d",&num);
        // 因为当前树已经按照自然数排序初始化了,如果当前输入的数字与序号相同,那么就不需要做处理
        if(r1==num) continue;
        Change(1,r1,num);
    }

    int a,b;
    for(r1=1;r1<=cc;r1++){
        scanf("%d%d%d",&num,&a,&b);
        switch(num){
            case 1: Change(1,a,b); break;
            case 2: printf("%d\n",Find(1,a,b,2));  break;
            case 3: printf("%d\n",Find(1,a,b,3));  break;
        }
    }

    return 0;
}

Control Lattic

时间: 2024-08-24 20:21:39

蓝桥杯《操作格子》的相关文章

2015年蓝桥杯省赛B组C/C++(试题+答案)

首先说,这次我是第二次参加蓝桥杯(大学里最后一次),可这次去连个三等都没拿到,有些心灰意冷,比上一次还差, 当时看到成绩出来的时候有些失落,但是跌倒了,再站起来继续跑就可以了.可能是状态不好吧,纯属自我安慰. 接下来我把今年的题目又重新做了一遍,写下了这篇博客,如果也有需要探讨答案的,希望可以有帮助. 第一题: 第1题:统计不含4的数字 题目大意 统计10000至99999中,不包含4的数值个数. 解题分析: 第一种解法: 数学方法,这种是在网上看到的一种解法: 最高位除了0.4不能使用,其余8

蓝桥杯——算法训练之乘积最大

问题描述 今年是国际数学联盟确定的"2000--世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告 勘误1:第6题第4个 if最后一个条件粗心写错了,答案应为1580. 条件应为abs(a[3]-a[7])!=1,宝宝心理苦啊.!感谢zzh童鞋的提醒. 勘误2:第7题在推断连通的时候条件写错了,后两个if条件中是应该是<=12 落了一个等于号.正确答案应为116. 1.煤球数目 有一堆煤球.堆成三角棱锥形.详细: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形), 第四层10个(排列成三角形). -. 假设一共

蓝桥杯——判定字符的位置。

蓝桥杯——判断字符的位置 题目: 输入一个字符串,编写程序输出该字符串中元音字母的首次出现位置,如果没有元音字母输出0.英语元音字母只有‘a’.‘e’.‘i’.‘o’.‘u’五个. 样例输入: hello样例输出:2 样例输入: apple样例输出:1 样例输入: pmp样例输出:0 java code: import java.util.*;public class Yuanyingzifu {    public static int fun(String str)    {        

蓝桥杯 地宫取宝(12&#39;)

X 国王有一个地宫宝库.是n x m 个格子的矩阵.每个格子放一件宝贝.每个宝贝贴着价值标签. 地宫的入口在左上角,出口在右下角. 小明被带到地宫的入口,国王要求他只能向右或向下行走. 走过某个格子时,如果那个格子中的宝贝价值比小明手中任意宝贝价值都大, 小明就可以拿起它(当然,也可以不拿). 当小明走到出口时,如果他手中的宝贝恰好是k件,则这些宝贝就可以送给小明. 请你帮小明算一算,在给定的局面下,他有多少种不同的行动方案能获得这k件宝贝. [数据格式] 输入一行 3个整数,用空格分开:n m

蓝桥杯-开心的金明

//蓝桥杯-算法训练 开心的金明 //评测结果 AC //动态规划 01背包 #include <cstdio> #include <algorithm> #include <cstring> using namespace std; const int maxn = 30005; int main() { int N, m; scanf( "%d%d", &N, &m ); int f[maxn]; int v, w; memset

第六届蓝桥杯2015-省赛-C语言大学B组 个人题解

题目连接:http://course.baidu.com/view/2d86a6c1960590c69fc37622.html 1,奖券数目52488 #include <iostream> using namespace std; bool isOK(int a) { while(a) { if(a%10==4) return false; a /= 10; } return true; } int main() { int ans = 0; for(int i=10000; i<=9

第五届蓝桥杯全国软件设计大赛--2013年校内选拔赛Java题目

第五届蓝桥杯全国软件设计大赛 2013年校内选拔赛Java题目 一.考生注意: (1)[结果填空题]要求参赛选手根据题目描述直接填写结果.求解方式不限.不要求源代码. 把答案存入[考生文件夹]下对应题号的文件中即可. (2)[代码填空题]要求参赛选手在弄清给定代码工作原理的基础上填写缺失的部分,使得程序逻辑正确.完整.所填写的代码不超过一条语句(即中间不能出现分号). 把填空的答案(仅填空处的答案,不包括题面已存在的代码)存入[考生文件夹]下对应题号的文件中中即可. (3)[编程题]要求选手设计

蓝桥杯——真题训练之蚂蚁感冒

标题:蚂蚁感冒 长100厘米的细长直杆子上有n只蚂蚁.它们的头有的朝左,有的朝右. 每只蚂蚁都只能沿着杆子向前爬,速度是1厘米/秒. 当两只蚂蚁碰面时,它们会同时掉头往相反的方向爬行. 这些蚂蚁中,有1只蚂蚁感冒了.并且在和其它蚂蚁碰面时,会把感冒传染给碰到的蚂蚁. 请你计算,当所有蚂蚁都爬离杆子时,有多少只蚂蚁患上了感冒. [数据格式] 第一行输入一个整数n (1 < n < 50), 表示蚂蚁的总数. 接着的一行是n个用空格分开的整数 Xi (-100 < Xi < 100),

【蓝桥杯】PrimeDistance

如果p1和p2(p1< p2)均为素数,且任意整数p(p1< p< p2)都不是素数,则说素数p1与p2是相邻的,并定义它们之间的距离为d= p2 - p1.给定正整数L和U,求出区间[L, U]中距离最小的两个相邻素数C1与C2和距离最大的两个相邻素数D1与D2. 例如,L=2,U=17时,C1=2,C2=3,D1=7,D2=11. import java.util.ArrayList; import java.util.Scanner; public class PrimeDista