题目名称

改造二叉树

数字对

交换

英文名称

binary

pair

swap

输入文件名

binary.in

pair.in

swap.in

输出文件名

binary.out

pair.out

swap.out

时间限制

1s

2s

1s

空间限制

256M

256M

256M

测试点数目

20

20

10

测试点分值

5

5

10

是否有部分分

题目类型

传统

传统

传统

是否有SPJ

1.改造二叉树

【题目描述】

小Y在学树论时看到了有关二叉树的介绍:在计算机科学中,二叉树是每个结点最多有两个子结点的有序树。通常子结点被称作“左孩子”和“右孩子”。二叉树被用作二叉搜索树和二叉堆。随后他又和他人讨论起了二叉搜索树。

什么是二叉搜索树呢?二叉搜索树首先是一棵二叉树。设key[p]表示结点p上的数值。对于其中的每个结点p,若其存在左孩子lch,则key[p]>key[lch];若其存在右孩子rch,则key[p]<key[rch];注意,本题中的二叉搜索树应满足对于所有结点,其左子树中的key小于当前结点的key,其右子树中的key大于当前结点的key。

小Y与他人讨论的内容则是,现在给定一棵二叉树,可以任意修改结点的数值。修改一个结点的数值算作一次修改,且这个结点不能再被修改。若要将其变成一棵二叉搜索树,且任意时刻结点的数值必须是整数(可以是负整数或0),所要的最少修改次数。

相信这一定难不倒你!请帮助小Y解决这个问题吧。

【输入格式】

第一行一个正整数n表示二叉树结点数。结点从1~n进行编号。

第二行n个正整数用空格分隔开,第i个数ai表示结点i的原始数值。

此后n - 1行每行两个非负整数fa, ch,第i + 2行描述结点i + 1的父亲编号fa,以及父子关系ch,(ch
= 0 表示i + 1为左儿子,ch = 1表示i
+ 1为右儿子)。

结点1一定是二叉树的根。

【输出格式】

仅一行包含一个整数,表示最少的修改次数。

【样例输入】

3

2 2 2

1 0

1 1

【样例输出】

2

【数据范围】

20 % :n <= 10 , ai <= 100.

40 % :n <= 100 , ai <= 200

60 % :n <= 2000 .

100 % :n <= 10 ^ 5 ,  ai < 2 ^
31.

 

/*
想到了中序遍历然后做lis
但是只25分 有一种情况是
2 5 3 4
这样lis是3 按说改一个5就行
但是题目限制了改成整数 2 3之间不能填了
这就wawawawawa~
这里学习了一种常见的将严格上升的的序列等价转化成不降的序列的方法
a1 a2 a3 -> a1-1 a2-2 a3-3
这样再做不降的lis就好了
*/
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define maxn 200010
using namespace std;
int n,a[maxn],b[maxn],len,lc[maxn],rc[maxn],c[maxn],r;
void Dfs(int x){
if(lc[x])Dfs(lc[x]);
b[++len]=x;
if(rc[x])Dfs(rc[x]);
}
void LIS(){
for(int i=;i<=len;i++){
int x=a[b[i]];x-=i;
if(x>=c[r]){
c[++r]=x;
continue;
}
int p=upper_bound(c+,c++r,x)-c;
c[p]=x;
}
}
int main()
{
freopen("binary.in","r",stdin);
freopen("binary.out","w",stdout);
scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%d",&a[i]);
int x,y;
for(int i=;i<=n;i++){
scanf("%d%d",&x,&y);
if(y==)lc[x]=i;
if(y==)rc[x]=i;
}
Dfs();
LIS();
printf("%d\n",len-r);
return ;
}

 

2.数字对

【题目描述】

小H是个善于思考的学生,现在她又在思考一个有关序列的问题。

她的面前浮现出一个长度为n的序列{ai},她想找出一段区间[L, R](1 <= L <= R <= n)。

这个特殊区间满足,存在一个k(L <= k <= R),并且对于任意的i(L <= i <= R),ai都能被ak整除。这样的一个特殊区间 [L, R]价值为R - L。

小H想知道序列中所有特殊区间的最大价值是多少,而有多少个这样的区间呢?这些区间又分别是哪些呢?你能帮助她吧。

【输入格式】

第一行,一个整数n.

第二行,n个整数,代表ai.

【输出格式】

第一行两个整数,num和val,表示价值最大的特殊区间的个数以及最大价值。

第二行num个整数,按升序输出每个价值最大的特殊区间的L.

【样例输入1】

5

4 6 9 3 6

【样例输出1】

1 3

2

【样例输入2】

5

2 3 5 7 11

【样例输出2】

5 0

1 2 3 4 5

【数据范围】

30%:
1 <= n <= 30 , 1 <= ai <= 32.

60%:
1 <= n <= 3000 , 1 <= ai <= 1024.

80%:
1 <= n <= 300000 , 1 <= ai <= 1048576.

100%: 1 <= n <= 500000 , 1 <= ai < 2 ^ 31.

暴力:

/*
n*n暴力 枚举ak 然后左右拓展
虽然是暴力 但是跑的飞快~~
注意输出的是不一样的区间....
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 500010
using namespace std;
int n,a[maxn],mxx,mxl,sum,Ans[maxn];
struct node{
int len,L;
}ans[maxn];
int init(){
int x=;char s=getchar();
while(s<''||s>'')s=getchar();
while(s>=''&&s<=''){x=x*+s-'';s=getchar();}
return x;
}
int cmp(const node &x,const node &y){
if(x.len==y.len)return x.L<y.L;
return x.len>y.len;
}
int Gcd(int a,int b){
return !b?a:Gcd(b,a%b);
}
int main()
{
freopen("pair.in","r",stdin);
freopen("pair.out","w",stdout);
n=init();
for(int i=;i<=n;i++)
a[i]=init();
for(int k=;k<=n;k++){
int li=k,ri=k,gcd=a[k];
for(int i=k+;i<=n;i++)
if(Gcd(gcd,a[i])!=gcd)break;
else ri++;
for(int i=k-;i>=;i--)
if(Gcd(gcd,a[i])!=gcd)break;
else li--;
ans[k].len=ri-li;ans[k].L=li;
}
sort(ans+,ans++n,cmp);
mxx=ans[].len;mxl=ans[].L;
Ans[++sum]=mxl;
for(int k=;k<=n;k++)
if(ans[k].len!=mxx)break;
else if(ans[k].L!=mxl){
Ans[++sum]=ans[k].L;
mxl=ans[k].L;
}
printf("%d %d\n",sum,mxx);
for(int i=;i<=sum;i++)
printf("%d ",Ans[i]);
return ;
}

ST表:

/*
nlogn做法
同样的枚举ak 左右拓展的时候二分来搞
这里恰好有单调性~
ST表预处理一下 O(1)查询
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 500010
using namespace std;
int n,a[maxn],f[maxn][],p[maxn],mxx,mxl,sum,Ans[maxn];
struct node{
int len,L;
}ans[maxn];
int init(){
int x=;char s=getchar();
while(s<''||s>'')s=getchar();
while(s>=''&&s<=''){x=x*+s-'';s=getchar();}
return x;
}
int cmp(const node &x,const node &y){
if(x.len==y.len)return x.L<y.L;
return x.len>y.len;
}
int Gcd(int a,int b){
return !b?a:Gcd(b,a%b);
}
int Get_pow(int x){
for(int i=;;i++)
if((<<i)>x)return i-;
}
void Get_ST(){
for(int i=;i<=n;i++)
f[i][]=a[i];
for(int j=;j<=;j++)
for(int i=;i+(<<j)-<=n;i++)
f[i][j]=Gcd(f[i][j-],f[i+(<<(j-))][j-]);
for(int i=;i<=n;i++)
p[i]=Get_pow(i);
}
int find(int l,int r){
if(l>r)return ;
int len=p[r-l+];
return Gcd(f[l][len],f[r-(<<len)+][len]);
}
int main()
{
freopen("pair.in","r",stdin);
freopen("pair.out","w",stdout);
n=init();
for(int i=;i<=n;i++)
a[i]=init();
Get_ST();
for(int k=;k<=n;k++){
int li=k,ri=k,gcd=a[k],l,r;
l=;r=n-k+;
while(l<=r){
int mid=l+r>>;
int x=find(k,k+mid-);
if(x==gcd){
ri=k+mid-;l=mid+;
}
else r=mid-;
}
l=;r=k;
while(l<=r){
int mid=l+r>>;
int x=find(k-mid+,k);
if(x==gcd){
li=k-mid+;l=mid+;
}
else r=mid-;
}
ans[k].len=ri-li;ans[k].L=li;
}
sort(ans+,ans++n,cmp);
mxx=ans[].len;mxl=ans[].L;
Ans[++sum]=mxl;
for(int k=;k<=n;k++)
if(ans[k].len!=mxx)break;
else if(ans[k].L!=mxl){
Ans[++sum]=ans[k].L;
mxl=ans[k].L;
}
printf("%d %d\n",sum,mxx);
for(int i=;i<=sum;i++)
printf("%d ",Ans[i]);
return ;
}

3.交换

【题目描述】

给定一个{0, 1, 2, 3, … , n - 1}的排列 p。一个{0, 1, 2 , … , n - 2}的排列q被认为是优美的排列,当且仅当q满足下列条件:

对排列s = {0, 1, 2, 3, ..., n - 1}进行n – 1次交换。

  1. 交换s[q0],s[q0 + 1]
  2. 交换s[q1],s[q1 + 1]

最后能使得排列s = p.

问有多少个优美的排列,答案对10^9+7取模。

【输入格式】

第一行一个正整数n.

第二行n个整数代表排列p.

【输出格式】

仅一行表示答案。

【样例输入】

3

1 2 0

【样例输出】

1

【样例解释】

q = {0,1}
{0,1,2} ->{1,0,2} -> {1, 2, 0}

q = {1,0}
{0,1,2} ->{0,2,1} -> {2, 0, 1}

【数据范围】

30%:  n <= 10

100%:  n <= 50

暴力:

/*直接暴力n!*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 51
using namespace std;
int n,p[maxn],q[maxn],s[maxn],ans;
int main()
{
freopen("swap.in","r",stdin);
freopen("swap.out","w",stdout);
scanf("%d",&n);
for(int i=;i<n;i++)q[i]=i;
for(int i=;i<=n;i++){
int x;scanf("%d",&x);
p[i]=x+;
}
do{
for(int i=;i<=n;i++)s[i]=i;
for(int i=;i<n;i++)swap(s[q[i]],s[q[i]+]);
int falg=;
for(int i=;i<=n;i++)
if(s[i]!=p[i]){
falg=;break;
}
if(falg==)ans++;
}while(next_permutation(q+,q+n));
printf("%d\n",ans);
}

9.2noip模拟试题的更多相关文章

  1. 模拟试题C

    模拟试题C 一.单项选择题(2′*14 =28′) 1.双线性法向插值法(Phong Shading)的优点是( ) A)法向计算精确 B)高光域准确 C)对光源和视点没有限制 D)速度较快 2.用编 ...

  2. 模拟试题B

    模拟试题B 一.单项选择题(2′*8 =16′) 1.灰度等级为256级,分辨率为2048*1024的显示器,至少需要的帧缓存容量为( ) A)512KB B)1MB C)2MB D)3MB 2.在多 ...

  3. 模拟试题A

    模拟试题A 一.单项选择题(2′*12=24′) 1.下面各种坐标变换中,会产生变换前后维度的改变的是( ) A)建模变换 B)观察变换 C)投影变换 D)视口变换 2.下列描述深度缓冲消隐算法的特点 ...

  4. CCF 模拟试题——出现次数最多的数 官方答案解析及自己写的正确答案

    前几天知道的CCF计算机职业资格认证考试,觉得好像比软考含金量高一些,就去了解了一下,做了模拟试题中的 “出现次数最多的数” 这道题,我的算法和官方答案算法不同,个人觉得觉得官方的好一点,没那么繁琐, ...

  5. 11.9 noip模拟试题

    NOIP2016 模拟赛——那些年,我们学过的文化课背单词(word.c/cpp/pas)[题目描述]fqk 退役后开始补习文化课啦, 于是他打开了英语必修一开始背单词. 看着满篇的单词非常头疼, 而 ...

  6. 10.26 noip模拟试题

    enc[问题背景]zhx 和他的妹子聊天.[问题描述]考虑一种简单的加密算法.假定所有句子都由小写英文字母构成,对于每一个字母,我们将它唯一地映射到另一个字母.例如考虑映射规则:a->b, b- ...

  7. 9.29noip模拟试题

    环上的游戏(cycle) 有一个取数的游戏.初始时,给出一个环,环上的每条边上都有一个非负整数.这些整数中至少有一个0.然后,将一枚硬币放在环上的一个节点上.两个玩家就是以这个放硬币的节点为起点开始这 ...

  8. 9.20 noip模拟试题

      Problem 1 双色球(ball.cpp/c/pas) [题目描述] 机房来了新一届的学弟学妹,邪恶的chenzeyu97发现一位学弟与他同名,于是他当起了善良的学长233 “来来来,学弟,我 ...

  9. 9.16noip模拟试题

    题目描述 在幻想乡,东风谷早苗是以高达控闻名的高中生宅巫女.某一天,早苗终于入手了最新款的钢达姆模型.作为最新的钢达姆,当然有了与以往不同的功能了,那就是它能够自动行走,厉害吧(好吧,我自重).早苗的 ...

随机推荐

  1. nodebb在阿里云主机部署过程

    1.在centos上安装nodejswget http://nodejs.org/dist/v0.8.9/node-v0.8.9.tar.gztar zxvf node-v0.8.9.tar.gzcd ...

  2. bzoj3191

    其实这是一个约瑟夫问题的变种首先我们先处理这样一个问题已知n个人,编号0~n-1,每k人干掉一个,问最后留下来的是谁当n,k非常大的时候,模拟是不行的,这时候我们考虑重编号第1次退出的肯定是肯定是编号 ...

  3. 2015 BJOI 0102 Secret

    传送门:http://oj.cnuschool.org.cn/oj/home/problem.htm?problemID=477 试题描述: 在 MagicLand 这片神奇的大陆上,有样一个古老传说 ...

  4. -_-#【RequireJS】Define a Module

    define({ color: 'black', size: 'unisize' }) define(function() { // Do setup work here return { color ...

  5. DIV+CSS布局问题:一个宽度不确定的DIV里面放三个水平对齐的DIV,左右两个DIV宽度固定为150px,中间那个DIV充满剩余的宽度

    一个入门的DIV+CSS布局问题:一个宽度不确定的DIV里面放三个水平对齐的DIV,左右两个DIV宽度固定为150px,中间那个DIV充满剩余的宽度. 说明:代码非真实情况下使用,所以直接简单. 没耐 ...

  6. iOS: 关于Certificate、Provisioning Profile、App ID的介绍及其之间的关系

    刚接触iOS开发的人难免会对苹果的各种证书.配置文件等不甚了解,可能你按照网上的教程一步一步的成功申请了真机调试,但是还是对其中的缘由一知半解.这篇文章就对Certificate.Provisioni ...

  7. JDK的下载和安装

    检查 检查是否已经安装了JRE,可以在命令行窗口输入"java –version",如果能看到下图所示的信息,则说明已经安装: 检查是否已经安装了JDK,暂时没有发现什么高大上的方 ...

  8. zoj 1586

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=1586 //zoj 1586 #include<iostream> ...

  9. 301重定向与CNAME

    CNAME记录是域名指向另一个域名A记录是域名指向IP地址URL跳转是打开这个网址的时候会跳转到另一个指定的网址,URL跳转分为301永久重定向和302临时重定向. 301永久重定向会使搜索引擎抓取新 ...

  10. 另一个有趣的Captcha 网站

    今天在一个网站注册时又发现了一个有趣的Captcha形式.给你一个翻转的图片,然后让你拽下面的slide bar让它回到正常的位置,很有趣.下面是提供这个Captcha的网站. minteye – s ...