9.2noip模拟试题
|
题目名称 |
改造二叉树 |
数字对 |
交换 |
|
英文名称 |
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次交换。
- 交换s[q0],s[q0 + 1]
- 交换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模拟试题的更多相关文章
- 模拟试题C
模拟试题C 一.单项选择题(2′*14 =28′) 1.双线性法向插值法(Phong Shading)的优点是( ) A)法向计算精确 B)高光域准确 C)对光源和视点没有限制 D)速度较快 2.用编 ...
- 模拟试题B
模拟试题B 一.单项选择题(2′*8 =16′) 1.灰度等级为256级,分辨率为2048*1024的显示器,至少需要的帧缓存容量为( ) A)512KB B)1MB C)2MB D)3MB 2.在多 ...
- 模拟试题A
模拟试题A 一.单项选择题(2′*12=24′) 1.下面各种坐标变换中,会产生变换前后维度的改变的是( ) A)建模变换 B)观察变换 C)投影变换 D)视口变换 2.下列描述深度缓冲消隐算法的特点 ...
- CCF 模拟试题——出现次数最多的数 官方答案解析及自己写的正确答案
前几天知道的CCF计算机职业资格认证考试,觉得好像比软考含金量高一些,就去了解了一下,做了模拟试题中的 “出现次数最多的数” 这道题,我的算法和官方答案算法不同,个人觉得觉得官方的好一点,没那么繁琐, ...
- 11.9 noip模拟试题
NOIP2016 模拟赛——那些年,我们学过的文化课背单词(word.c/cpp/pas)[题目描述]fqk 退役后开始补习文化课啦, 于是他打开了英语必修一开始背单词. 看着满篇的单词非常头疼, 而 ...
- 10.26 noip模拟试题
enc[问题背景]zhx 和他的妹子聊天.[问题描述]考虑一种简单的加密算法.假定所有句子都由小写英文字母构成,对于每一个字母,我们将它唯一地映射到另一个字母.例如考虑映射规则:a->b, b- ...
- 9.29noip模拟试题
环上的游戏(cycle) 有一个取数的游戏.初始时,给出一个环,环上的每条边上都有一个非负整数.这些整数中至少有一个0.然后,将一枚硬币放在环上的一个节点上.两个玩家就是以这个放硬币的节点为起点开始这 ...
- 9.20 noip模拟试题
Problem 1 双色球(ball.cpp/c/pas) [题目描述] 机房来了新一届的学弟学妹,邪恶的chenzeyu97发现一位学弟与他同名,于是他当起了善良的学长233 “来来来,学弟,我 ...
- 9.16noip模拟试题
题目描述 在幻想乡,东风谷早苗是以高达控闻名的高中生宅巫女.某一天,早苗终于入手了最新款的钢达姆模型.作为最新的钢达姆,当然有了与以往不同的功能了,那就是它能够自动行走,厉害吧(好吧,我自重).早苗的 ...
随机推荐
- bom type:Phantom
bom的类型 'type': fields.selection([('normal','Normal BoM'),('phantom','Sets / Phantom')], 'BoM Type', ...
- django笔记1
最近在博客园看来越来越多的关于python的文章,我看到时感觉特别的好,因为我也是一个特别喜欢python这门语言,喜欢python的简洁.干净,简洁而不失强大. 最近在学习django的Model模 ...
- 使用ToUpperInvariant避免使用ToUpper
ToUpperInvariant使用不依赖于区域性进行转换,而ToUpper则使用了当前线程的CultureInfo,进行转换,所以性能会有所影响,以下为测试: [Test] public void ...
- ios中关于delegate(委托)的使用心得
ios中关于delegate(委托)的使用心得 分类: iOS开发2012-05-15 10:54 34793人阅读 评论(9) 收藏 举报 iosuiviewtimerinterfaceprinti ...
- ios+oc面试题
ios+oc面试题 浅复制和深复制的区别?//浅拷贝和深拷贝答案:浅层复制(copy):只复制指向对象的指针,而不复制引用对象本身.//通过对象的指针来访问这个对象深层复制(mutableCo ...
- Canny边缘检测-Wiki
Canny edge dector 由 John F. Canny 在1986年提出. Canny 算法的发展 Canny算法的步骤 2.1 降噪 2.2 寻找图像的亮度梯度 2.3 非极大值抑制 2 ...
- Java吸收换行符
今天做题遇到的-- 由于读入的字符串可能包含空格,所以采用nextLine. int n = sc.nextInt(); for(int i=0; i<n; i+ ...
- 轮值CEO胡厚崑:到2025年所有的企业都将用到云(云的2.0时代,会有几千朵云几万朵云升起来,这将产生不同的技术模式、商业模式、思维模式)
2016年09月04日 07:38 中国经营报 李凡 在全国工商联“2016年中国民营企业500强”排行榜上夺得头把交椅的华为,向外界描绘了面向未来进一步做大做强的路径. 华为创始人任正非于201 ...
- Java实现二叉树的构建与遍历
转载:http://ocaicai.iteye.com/blog/1047397 目录: 1.把一个数组的值赋值给一颗二叉树 2.具体代码 1.树的构建方法 2.具体代码 package tree; ...
- Javascript面向对象编程(二):构造函数的继承 by 阮一峰
今天要介绍的是,对象之间的"继承"的五种方法. 比如,现在有一个"动物"对象的构造函数. function Animal(){ this.species = & ...