2017-8-31 NOIP模拟赛(校内测)

可接受序列

【题目描述】

让计算机这样读入一列非负整数:

1、读入数T

2、接着往下读入T个数。

3、如果数列读完了,则停止,否则,转到1。

但是,往往会出现这样的问题:执行第2步时,数列已经没有T个数了。如果这样,我们称这个数列是“不可接受的”,否则,称它是“可接受的”。我们需要用最少的步数把一个数列变成“可接受的”,一步是指:

1、把数列中的某一个数加1。

2、把数列中的某一个数减1。

【输入格式】

第一行有一个数N (1<=N<=1000000),表示数列的长度,接下来有n行,描述这个数列,每一行有一个非负整数(不超过1000000)。

【输出格式】

仅一个数,表示最少的步数。

【输入输出样例】


sequence.in


sequence.out


7

3

1

2

3

4

5

6


1

【数据规模】

对于50%的数据,N≤1,000;

对于80%的数据,N≤100,000;

对于100%的数据,N≤1,000,000。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
using namespace std;
int n,a[1000010];
long long ans=9999999999;
void dfs(int pos,long long cnt){
    if(cnt>=ans)return;
    //if(pos>n)return;
    if(pos==n+1){
        ans=min(ans,cnt);
        return;
    }
    if(pos==n){
        dfs(pos+1,cnt+a[pos]);
        return;
    }
    for(int i=0;i+pos<=n+1;i++){
        int step=abs(a[pos]-i);
        dfs(pos+i+1,cnt+step);
    }
}
int main(){
    //freopen("Cola.txt","r",stdin);
    freopen("sequence.in","r",stdin);
    freopen("sequence.out","w",stdout);
    scanf("%d",&n);
    for(int i=1;i<=n;i++)scanf("%d",&a[i]);
    dfs(1,0);
    cout<<ans;
    return 0;
}

20分 暴力深搜

 

促销

【题目描述】

某某商场搞了一个促销活动,促销活动以以下方式进行:

  1. 一个想要参加活动的顾客将他的发票扔进抽奖箱里。
  2. 在每天的最后,从抽奖箱里抽出两张发票:

a)       金额最大的发票a

b)       金额最小的发票b

  1. 金额最大的发票的持有者得到a-b的奖金。

每天被抽出的发票都不会再被放回抽奖箱里。

你想知道促销活动结束时一共付出了多少奖金。

【输入格式】

第一行一个N,促销进行的天数;

接下来N行,第一个数是一个ki,表示第i天收到的发票数;接下来ki个数,每个数是一个发票的金额。

【输出格式】

一个数,整个促销活动过程中一共付出了多少奖金。

【输入输出样例】


promotion.in


promotion.out


5

3 1 2 3

2 1 1

4 10 5 5 1

0

1 2


19

【数据规模】

对于30%的数据,发票总数M不超过2000;

对于另外20%的数据,每张发票的金额不超过2000;

对于100%的数据,发票总数M不超过1000000,每张发票的金额不超过1000000。

#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
priority_queue<int>heap1;
priority_queue<int>heap2;
int n;
long long ans;
int main(){
    //freopen("Cola.txt","r",stdin);
    freopen("promotion.in","r",stdin);
    freopen("promotion.out","w",stdout);
    scanf("%d",&n);
    int x,y;
    for(int i=1;i<=n;i++){
        scanf("%d",&x);
        for(int j=1;j<=x;j++){
            scanf("%d",&y);
            heap1.push(y);
            heap2.push(-y);
        }
        int a,b,a1,b1;
        a=heap1.top();heap1.pop();
        b=heap2.top();heap2.pop();
        if(a!=-b){
            ans+=a+b;
            continue;
        }
        a1=heap1.top();heap1.pop();
        b1=heap2.top();heap2.pop();
        if(a+b1>b+a1){
            heap1.push(a1);
            heap2.push(b);
            ans+=a+b1;
            continue;
        }
        else {
            heap1.push(a);
            heap2.push(b1);
            ans+=a1+b;
            continue;
        }
    }
    cout<<ans;
    return 0;
}

100分 两个堆

亲和数

【题目描述】

某一天,老徐看了一本趣味数学书,上面提到了亲和数:定义数对 (x,y) 为亲和数对当且仅仅当x、y为不同正整数,且x、y各自的所有非自身正因子之和等于另一个数。例如 (220,284) 和 (284,220) 都是亲和数对,因为:220的所有非自身正因子之和为:1 + 2 + 4 + 5 + 10 + 11 + 20 + 22 + 44 + 55 + 110 = 284。284的所有非自身正因子之和为:1 + 2 + 4 + 71 + 142 = 220

数对 (x,y ) 跟 (y,x) 被认为是同一数对,所以我们只考虑 x<y 的情况。

老徐对某个范围内的亲和数对的数量非常感兴趣,所以希望你能帮她编写一个程序计算给定范围内的亲和数对的数量。给定一个范围A到B,如果A≤ x ≤ B,则我们称 (x,y)在范围[A,B]内。

【输入格式】

从文件的第一行分别读入正整数A和B,其中A、B满足

     1 ≤ A ≤ B ≤ 108  且 B-A ≤ 105

【输出格式】

输出文件只有一行,就是[A,B]内亲和数对的数量

【输入输出样例】


amicable.in


amicable.out


200 1200


2

注:[200,1200] 内的数对只有两个,分别是(220,284)和(1184 1210)

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int a,b,bit[10],len,q[10];
bool pan(int num){
    int sum=1;
    for(int i=2;i*i<=num;i++){
        if(num%i==0&&i*i!=num)sum+=i+num/i;
        if(i*i==num)sum+=i;
    }
    if(sum<=num)return 0;
    for(int i=2;i*i<=sum;i++){
        if(sum%i==0&&i*i!=sum)num-=i+sum/i;
        if(i*i==sum)num-=i;
        if(num<0)return 0;
    }
    if(num==1)return 1;
}
int shu(int pos,int num,bool limit){
    if(pos==len+1){
        if(pan(num))return 1;
        return 0;
    }
    int end=limit?bit[pos]:9;
    int ans=0;
    for(int i=0;i<=end;i++){
        ans+=shu(pos+1,num*10+i,limit&&i==end);
    }
    return ans;
}
int dp(int x){
    memset(q,0,sizeof(q));
    memset(bit,0,sizeof(bit));
    len=0;
    while(x){
        q[++len]=x%10;
        x/=10;
    }
    for(int i=len,j=1;i>=1;i--,j++)bit[i]=q[j];
    return shu(1,0,1);
}
int main(){
    //freopen("Cola.txt","r",stdin);
    freopen("amicable.in","r",stdin);
    freopen("amicable.out","w",stdout);
    scanf("%d%d",&a,&b);
    cout<<dp(b)-dp(a-1);
    return 0;
}

50分 数位dp(记忆化没写成)

时间: 2024-11-07 02:46:16

2017-8-31 NOIP模拟赛(校内测)的相关文章

2017 9 11 noip模拟赛T2

#include<cstdio> #include<cstring> #include<algorithm> using namespace std; const int N=205; int map[N][N]; int d[N],tag[N],book[N],f[N]; int n,m; void work(int x) { memset(d,63,sizeof(d)); memset(book,0,sizeof(book)); memset(f,0,sizeof(

2017 11.6 NOIP模拟赛

1.数学老师的报复(attack.pas/c/cpp)[问题描述]11 班数学大佬 YXN 又在上数学课的时候把班主任 MP6 的错误当众挑出来了,MP6 再一次感到很难堪,于是决定报复 YXNMP6 对 YXN 说:给你一个函数 f(x),定义如下:f ( 1 ) = 1f ( 2 ) = 1f ( n ) = ( A * f ( n - 1 ) + B * f ( n - 2 ) ) mod 7.YXN 说这还不简单,可以秒杀!MP6 微微笑了笑说:n 等于 100 你算得出来,那 n 等于

2017 9 15 noip模拟赛

t1: #include<cstdio> #include<cstring> const int N=1000010; int a[N],b[N]; int sum=1,maxx=0; int main() { freopen("sequence.in","r",stdin); freopen("sequence.out","w",stdout); int n; scanf("%d"

2017.5.27 NOIP模拟赛(hzwer2014-5-16 NOIP模拟赛)

期望得分:100+100+60+30=290 实际得分:100+20+60+0=180 当务之急:提高一次正确率 Problem 1 双色球(ball.cpp/c/pas) [题目描述] 机房来了新一届的学弟学妹,邪恶的chenzeyu97发现一位学弟与他同名,于是他当起了善良的学长233 “来来来,学弟,我考你道水题检验一下你的水平……” 一个栈内初始有n个红色和蓝色的小球,请你按照以下规则进行操作 只要栈顶的小球是红色的,将其取出,直到栈顶的球是蓝色 然后将栈顶的蓝球变成红色 最后放入若干个

2017.9.24 noip模拟赛 day2—组合数

简化版题意: 给定n,m(n<=m),求C(n,m)的末尾有几个0 输入格式: 第一行一个整数t,代表数据组数. 接下来t行,每行两个整数n,m 输出格式: t行,每行一个整数,代表C(n,m)的末尾0的个数. 样例输入: 310 111 720 4 样例输出: 110 数据范围: 1<=m<=n<=1000000,t<=1000 首先这道题需要知道一个公式:C(n,m)=n!/(n-m)!*m! 然后10=2*5,也就是说,对于一个数,我们将其因数分解,2和5中较少的数的个

NOIP模拟赛

#1[Nescafé 31]杯NOIP模拟赛 t1 题意:n*m的棋盘上从(1,1)走到(n,m),只能向下或向右,一些格子有老鼠,每个老鼠互不相同,当处于与老鼠有重边的格子时,视为看见了这只老鼠,求到终点看到最少的不同老鼠数. 分析:DP 由于求得是看到的不同的老鼠数目,不能直接用过河卒做,因为同一个位置的老鼠可能会统计多次,我们还需要增加一维即方向. f[i,j,0]表示到从上面一个格子走到(i,j)时最少老鼠数,f[i,j,1]表示左边. f[i,j,0]:=min(f[i-1,j,0]+

NOIP模拟赛 6.29

2017-6-29 NOIP模拟赛 Problem 1 机器人(robot.cpp/c/pas) [题目描述] 早苗入手了最新的Gundam模型.最新款自然有着与以往不同的功能,那就是它能够自动行走,厉害吧. 早苗的新模型可以按照输入的命令进行移动,命令包括‘E’.‘S’.‘W’.‘N’四种,分别对应东南西北.执行某个命令时,它会向对应方向移动一个单位.作为新型机器人,它可以执行命令串.对于输入的命令串,每一秒它会按命令行动一次.执行完命令串的最后一个命令后,会自动从头开始循环.在0时刻时机器人

【简单思考】noip模拟赛 NTR酋长

NTR酋长 (ntr.pas/.c/.cpp) 黄巨大终于如愿以偿的进入了czy的后宫中……但是czy很生气……他要在黄巨大走到他面前的必经之路上放上几个NTR酋长来阻挡黄巨大. 众所周知,NTR酋长有一个技能是沟壑(F).它会在地图上产生一条长长的障碍物阻挡人前进.Czy打算在一个n*m的矩形(必经之路?)中放上NTR酋长.NTR酋长要一个一个放下去,而且每放一个都会向四角倾斜的方向放出无限长的沟壑,而已经被沟壑挡住的地方就不能再放NTR酋长了. 请注意:不会出现沟壑的路径挡住另一个沟壑的情况

【noip模拟赛】 射击

这题似乎是什么安阳一中的模拟题,不管了,反正是学长出的noip模拟赛里面的题目.... 射击(shoot.pas/.c/.cpp) 时间限制:1s,内存限制128MB 题目描述: 据史书记载,对越反击战时期,有位中国侦察兵,他的代号叫814.一天他执行狙击任务,他的任务地区是n座恰巧在一条直线上的山.这些山所在直线恰巧为东西走向,山从东到西依次编号为1~n.一天814隐藏在编号为k的山上,每座山上都有1个目标. 814也非常的厉害,任务结束时杀了很多人,可是史书中只记载了两点: 1:814一定攻

2017.6.11 校内模拟赛

题面及数据及std(有本人的也有原来的) :2017.6.11 校内模拟赛 T1 自己在纸上模拟一下后就会发现 可以用栈来搞一搞事情 受了上次zsq 讲的双栈排序的启发.. 具体就是将原盘子大小copy一下排个序 用两个指针维护两个数组(原数据 和 排序后的数据), 即分为1数据和2数组 将小于1指针指向的数据的2数组中的数据全部压入栈中 后进行消除, 将栈栈顶元素与当前1数组中的1指针指向的元素进行比较 相同则消除 后重复过程 直至指针超过N 后判断一下是否两个指针都超过了N... #incl