排位赛 09 题解

排位赛 09 题解

A - Fox and Number Game

CodeForces - 389A

题意

给出一列数,数之间可以相减(大数-小数),不限次数,求这列数可以得到的最小总和。

思路

观察数据加上思考不难发现,每个数最后都可以变成同一个数,就是他们的最大公约数。

代码

#include <cstdio>
#include <algorithm>
using namespace std;
int main(){
    int n, num[105];
    scanf("%d", &n);
    for (int i = 0; i < n; ++i)
        scanf("%d", num+i);
    int ans = num[0];
    for (int i = 1; i < n;++i)
        ans = __gcd(ans, num[i]);
    printf("%d\n", ans*n);
    return 0;
}

B - Fox and Cross

CodeForces - 389B

题意

给出一张由'.''#'组成的正方形图,需要判断是否图上的'#'可以不重叠地组成若干个十字。

思路

简单DFS。

代码

#include <cstdio>
#include <cstring>
#include <cmath>
#include <iostream>
#include <algorithm>

using namespace std;

int n;
char maps[105][105];
const int dir[4][2] = {1,0,-1,0,0,1,0,-1};

void check(int x, int y) {
    if (x == 0 || x == n-1 || y == 0 || y == n-1)
        return;
    for (int i = 0; i < 4; ++i)
        if (maps[x+dir[i][0]][y+dir[i][1]] == '.')
            return;
    for (int i = 0; i < 4; ++i)
        maps[x+dir[i][0]][y+dir[i][1]] = '.';
    maps[x][y] = '.';
}

int main(){
    bool flag = 1;
    scanf("%d", &n);
    for (int i = 0; i < n; ++i)
        scanf("%s", maps+i);
    for (int i = 0; i < n; ++i)
        for (int j = 0; j < n; ++j)
            if (maps[i][j] == '#')
                check(i, j);
    for (int i = 0; i < n; ++i){
        for (int j = 0; j < n; ++j)
            if (maps[i][j] == '#')
                flag = 0;
        //printf("%s\n", maps[i]);
    }
    if (flag) printf("YES\n");
    else printf("NO\n");
    return 0;
}

C - Fox and Box Accumulation

CodeForces - 389C

题意

给出一列数,让你排成若干列。每个数字的值表示这个数字前面允许存在的数字个数。求最少要排几列。

思路

贪心分堆。先排个序,从头开始遍历,取一个作为一排的第一个,然后向后遍历数列,挨个判断是否可以放进这一排,并且标记。

代码

#include <cstdio>
#include <cstring>
#include <algorithm>

using namespace std;

int main(){
    int n, num[105], ans = 0, vis[105];
    memset(vis, 0, sizeof vis);
    scanf("%d", &n);
    for (int i = 0; i < n; ++i)
        scanf("%d", num+i);
    sort(num, num+n);
    for (int i = 0, cnt = 0; i < n; ++i, cnt = 0) {
        if (vis[i]) continue;
        ++vis[i], ++cnt, ++ans;
        for (int j = i+1; j < n; ++j) {
            if (vis[j]) continue;
            if (num[j] >= cnt) ++vis[j], ++cnt;
        }
    }
    printf("%d\n", ans);
    return 0;
}

D - Fox and Minimal path

CodeForces - 389D

题意

给出一个数字n,求一张有着若干节点的图。输出这张图的邻接表。要求这张图从点1到点2的最短路的条数为n。输出时如果这条路存在则输出Y,否则为N

思路

因为没有限定要节点数最少,所以任意。

一开始想的是,只要找出最多需要多少点,然后用这张表来处理就可以了。但是没有弄清楚到底是怎么添加最短路的,所以WA了。

我们换一种方式。1是起点,2是终点。我们先确定一条主链,主链规定了这条路会经过多少个节点。然后从起点开始,按照主链的节点数给副链添加节点。这里规定副链的一层有两个节点,这样n层就是 $2^n$ 条路径,这样的原始图建成之后,是 $2^n+1$ 条路径。

然后考虑添加路径数。i+1层主链和i层副链相连一条,可以添加 $2^{i-1}$ 条路径。

到这里我们的解法就出来了。首先因为我们有主链,所以先n-=1;然后找出最大的二进制位,这就是我们主链的长度,然后根据每一个二进制位选择是否要进行添加。

代码

#include <cstdio>
#include <cstring>
#include <cmath>
#include <iostream>
#include <algorithm>

using namespace std;

int maps[107][107];

void addEdge(int a, int b) {
    maps[a][b] = maps[b][a] = 1;
}

int main(){
    int n, size = 0, t;
    scanf("%d", &n);
    t = --n;
    while (t) {
        ++size;
        t >>= 1;
    }
    if (!size) {
        printf("2\nNY\nYN\n");
        return 0;
    }
    addEdge(1, 3);
    addEdge(1, 4);
    addEdge(1, 5);
    addEdge(size*3, 2);
    for (int i = 1; i < size; ++i) {
        addEdge(i*3, (i+1)*3);
        addEdge(i*3+1, (i+1)*3+1);
        addEdge(i*3+2, (i+1)*3+2);
        addEdge(i*3+1, (i+1)*3+2);
        addEdge(i*3+2, (i+1)*3+1);
    }
    int now = 0;
    while (n) {
        ++now;
        if (n&1) {
            if (now == size)
                addEdge(2, 3*now + 1);
            else
                addEdge(3*(now+1), 3*now+1);
        }
        n >>= 1;
    }
    size = size * 3 + 2;
    printf("%d\n", size);
    for (int i = 1; i <= size; ++i) {
        for (int j = 1; j <= size; ++j)
            printf("%c", maps[i][j]?'Y':'N');
        printf("\n");
    }
    return 0;
}

F - Water problem

HDU - 5867

题意

给一个一千以内的数字n,求从1n的英文写法的字母总数。不计算空格和连字符。

思路

打表,1-100,然后预处理出1-1000的所有答案,O(1)输出。

代码

#include <cstdio>
using namespace std;

int a[1001] ={
    0,
    3, 3, 5, 4, 4, 3, 5, 5, 4, 3,
    6, 6, 8, 8, 7, 7, 9, 8, 8, 6,
    9, 9, 11, 10, 10, 9, 11, 11, 10, 6,
    9, 9, 11, 10, 10, 9, 11, 11, 10, 5,
    8, 8, 10, 9, 9, 8, 10, 10, 9, 5,
    8, 8, 10, 9, 9, 8, 10, 10, 9, 5,
    8, 8, 10, 9, 9, 8, 10, 10, 9, 7,
    10, 10, 12, 11, 11, 10, 12, 12, 11, 6,
    9, 9, 11, 10, 10, 9, 11, 11, 10, 6,
    9, 9, 11, 10, 10, 9, 11, 11, 10, 10};

int main(){
    for(int i = 20; i < 100; i += 10)
        for(int j = i+1; j < i+10; ++j)
            a[j] = a[i]+a[j-i];
    for(int i = 100; i < 1000; i += 100){
        a[i] = a[i/100] + 7;
        for(int j = i+1; j <= i+99; ++j)
            a[j]=a[i]+3+a[j-i];
    }
    a[1000] = 3+8;
    for(int i = 2; i <= 1000; ++i)
        a[i] += a[i-1];
    int t, n;
    scanf("%d", &t);
    while (t--) {
        scanf("%d", &n);
        printf("%d\n", a[n]);
    }
    return 0;
}

H - Max Sum

HDU - 1003

题意

给一列数,求最大字段和。

思路

递推DP。稍微说下思路。首先因为是有负数的,所以记录最大值初始值为负,避免全负数情况。然后遍历数列,都加进计数器中,如果计数器变为负数就归零。如果大于记录最大值就记录下来,并且记录当前的子段位置。

代码

#include <cstdio>
using namespace std;

int main(){
    int t, cas = 0;
    scanf("%d", &t);
    while (t--) {
        int n, a, left = 1, right = 1;
        int sum = 0, maxx = -1005, cur = 1;
        scanf("%d", &n);
        for (int i = 1; i <= n; ++i){
            scanf("%d", &a);
            sum += a;
            if (sum > maxx) {
                maxx = sum;
                left = cur, right = i;
            }
            if (sum < 0) {
                sum = 0;
                cur = i + 1;
            }
        }
        printf("Case %d:\n%d %d %d\n", ++cas, maxx, left, right);
        if (t) printf("\n");
    }
    return 0;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,723评论 0 33
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,454评论 25 707
  • 32位和64位的区别 区别在于一次存取数据的位数。 c/c++ 数据类型区分32和64,但java不区分,因为JV...
    Acorld阅读 122评论 0 0
  • 秋天是一个顽皮的姐姐,他每天都打一些果实拿镰刀割下来。人们总是说果实磊磊秋天这个果实累累的季节,说实在的我们的...
    吴怡霖阅读 239评论 0 0
  • 感受通常是几个形容词,以前我是从来不懂自己的感受的。后面去上过正面管教的课,才明白感受没有对错,会来也会走。任何感...
    邓美芳阅读 311评论 0 1