1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

AtCoderをC#で始めてみたよ -AtCoder Beginners Selection-

Last updated at Posted at 2024-06-16

こんにちはYamayouです.早稲田の機械専攻とかいうところに生息しております.
はてなブログでコードとかのブログを書こうと思ったけど絶対Qiitaの方がいいじゃんと思ってこちらにしました.いちおうはてなブログの方も載せとく

0.はじめに

競プロに興味あったのではじめてみました.初心者です.AtCoder Beginners Selectionから始めてみようと思います.研究でちょっとC#を使っていたので,C#で進めていきます.備忘録って感じで残していくので気楽に見ていってね.

第1問 ABC 086 A - Product

image.png

コメント

掛け算して2で割ったあまりで条件分岐でいいのかな.2つの数字それぞれの偶奇が分かれば掛け算はしなくても判断できるけどまあこのくらいの計算量ならかけちゃった方がいいっぽいな

using System;
class Program
{
    static void Main(String[] args){
        string[] input = Console.ReadLine().Split(' ');
        int a = int.Parse(input[0]);
        int b = int.Parse(input[1]);
        if(a*b%2 == 0){
            Console.WriteLine("Even");
        }else{
            Console.WriteLine("Odd");
        }
    }
}

第2問 ABC 081 A - Placing Marbles

image.png

コメント

要は,0と1からなる3桁の数(例:101)が与えられるから1が何個あるか数えてねって感じだな.なんかA問題はfor使わなくても解けるっぽいけど最初に思い付いたfor使うことにするわ.とりあえず一の位だけ判定します.10で割った余りが1かどうかを判定して,1だったらansを+1していく感じでやるか.

 次はa=a/10を挟んで数字を右にずらすとさっきまで十の位にあった数が一の位にあるのでこれを繰り返せばいいかな.

using System;
class Program
{
    static void Main(String[] args){
        int a = int.Parse(Console.ReadLine());
        int ans = 0;

        for(int i = 0;i<3;i++){
            if(a%10==1){
                ans++;
            }
            a = a/10;
        }  
        
        Console.WriteLine(ans);
    }
}

for使わない方も一応書いとく,こっちの方が無難か.

using System;
class project1
{
    static void Main(String[] args)
    {
        string input = Console.ReadLine();
        int ans = 0;

        if (input[0] == '1') ans++;
        if (input[1] == '1') ans++;
        if (input[2] == '1') ans++;

        Console.WriteLine(ans);
    }  
}

第3問 ABC 081 B - Shift Only

image.png

コメント

「すぬけ君」...?まあいいか...これはシンプルに全部2で割っていけばいいのかな,なんかもっといい方法ある気がするけど思いつかないのでごり押ししていく.途中で2で割った余りが1になったらflagをtrueにしてwhileから抜け出す感じで

using System;
using System.Linq;
class project1
{
    static int N = 0;
    static int ans = 0;
    static int[] A = new int[210]; //一応200より多めにとっておく

    static void Main(String[] args)
    {
        N = int.Parse(Console.ReadLine());
        A = Console.ReadLine().Split(' ').Select(int.Parse).ToArray();

        while (true)
        {
            bool flag = false;
            for(int i = 0; i < N; i++)
            {
                if (A[i] % 2 == 1) flag = true;
                A[i] = A[i] / 2;
            }

            if (flag == true) break;
            ans++;
        }

        Console.WriteLine(ans);
    }  
}

第4問 ABC 087 B - Coins

image.png

コメント

3重でfor文回して全部探索するだけ...?なんかもっといい方法ないのかな

using System;
class Program
{
    static int A,B,C,X,ans;

    static void Main(String[] args)
    {
        A = int.Parse(Console.ReadLine());
        B = int.Parse(Console.ReadLine());
        C = int.Parse(Console.ReadLine());
        X = int.Parse(Console.ReadLine());


        for(int i = 0; i <= A; i++)
        {
            for(int j = 0; j <= B; j++)
            {
                for(int k = 0; k <= C; k++)
                {
                    int sum = 500 * i + 100 * j + 50 * k;
                    if (sum == X) ans++;
                }
            }
        }

        Console.WriteLine(ans);

    }
}

第5問 ABC 083 B - Some Sums

image.png

コメント

各位の和を出すプログラムを書いてねってことね.CalculateSumOfDigits関数の中に処理を書いていく.totalにnumを10で割ったあまり(一の位)を代入して,numを10で割って一の位を消して各桁を右にずらす.これをnumが0になるまで繰り返したらいいのかな.

using System;
class Program
{
    static int N, A, B, ans;

    static int CalculateSumOfDigits(int num)
    {
        int total = 0;
        while (num > 0)
        {
            total += (num % 10);
            num /= 10;
        }
        return total;
    }

    static void Main(String[] args)
    {
        string[] input = Console.ReadLine().Split(' ');
        N = int.Parse(input[0]);
        A = int.Parse(input[1]);
        B = int.Parse(input[2]);
        ans = 0;

        for (int i = 1; i <= N; i++)
        {
            int total = CalculateSumOfDigits(i);
            if (A <= total & total <= B) ans += i;
        }

        Console.WriteLine(ans);

    }
}

第6問 ABC 088 B - Card Game for Two

image.png

コメント

ソートしてでかい順に,AさんとBさんにもっていってもらえばよさそう.ソートはArray.Sortを使ったけど昇順になっちゃうのでReverseしとく.

using System;
using System.Linq;
class Program
{
    static int N = 0;
    static int[] A = new int[110]; //一応100より多めにとっておく
    static int scoreA = 0;
    static int scoreB = 0;

    static void Main(String[] args)
    {
        N = int.Parse(Console.ReadLine());
        A = Console.ReadLine().Split(' ').Select(int.Parse).ToArray();

        Array.Sort(A);
        Array.Reverse(A);

        for(int i = 0; i < N; i++)
        {
            if (i % 2 == 0) scoreA += A[i];
            if (i % 2 == 1) scoreB += A[i];
        }

        Console.WriteLine(scoreA - scoreB);
        
    }
}

第7問 ABC 085 B -Kagami Mochi

image.png

コメント

大きい順に並べて前から数の要素数だけカウントしていけばいいかな.

using System;
class Program
{
    static int N, ans;
    static int[] d = new int[110];

    static void Main(String[] args)
    {
        N = int.Parse(Console.ReadLine());
        for(int i = 0; i < N; i++)
        {
            d[i] = int.Parse(Console.ReadLine());
        }

        Array.Sort(d);
        Array.Reverse(d);

        int nowD = d[0];
        int preD = 110; //少し大きめの数を入れとく

        for(int i = 0; i < N; i++)
        {
            nowD = d[i];
            if (nowD != preD) ans++;
            preD = nowD;
        }

        Console.WriteLine(ans);

    }
}

これでもうまくいったけど世の中にはバケット法なるものがあるらしい...書き直してみるか

using System;
class Program
{
    static int N, ans;
    static int[] d = new int[110];

    static void Main(String[] args)
    {
        N = int.Parse(Console.ReadLine());
        for(int i = 0; i < N; i++) d[i] = int.Parse(Console.ReadLine());
        ans = 0;

        int[] num = new int[110];
        for (int i = 0; i < N; i++) num[i] = 0;

        for (int i = 0; i < N; i++) num[d[i]]++;

        for(int i = 1; i <= 100; i++)
        {
            if (num[i] > 0) ans++;
        }

        Console.WriteLine(ans);

    }
}

Distinct().ToArray()を使うと重複した要素をすべて消してくれるらしい,これを使うと行数はもっと短くなるな

using System;
using System.Linq;
class Program
{
    static int N;
    static int[] d = new int[110];

    static void Main(String[] args)
    {
        N = int.Parse(Console.ReadLine());
        for(int i = 0; i < N; i++) d[i] = int.Parse(Console.ReadLine());
        int[] value = d.Distinct().ToArray();
        Console.WriteLine(value.Length-1);

    }
}

第8問 ABC 085 C -Otoshidama

image.png

コメント

初めてのC問題.お札をN枚使ってちょうどY円となるような組み合わせを見つける問題.10000円札をi枚,5000円をj枚つかったとすれば1000円札は自然とN-i-j枚となるので2回for文を回したらいけそうだな.

using System;
class Program
{
    static void Main(String[] args)
    {
        string[] input = Console.ReadLine().Split(' ');
        int N = int.Parse(input[0]);
        int Y = int.Parse(input[1]);

        int[] ans ={ -1, -1, -1 };

        for(int i = 0; i <= N; i++)
        {
            for(int j = 0; j <= N - i; j++)
            {
                int sum = 10000 * i + 5000 * j + 1000 * (N - i - j);
                if(sum == Y)
                {
                    ans[0] = i;
                    ans[1] = j;
                    ans[2] = N - i - j;
                }
            }
        }

        Console.WriteLine(ans[0] + " " + ans[1] + " " + ans[2]);

    }
}

第9問 ABC 049 C -Daydream

image.png

コメント

んーー難しい...結構悩んだけど分からなかったので解答見ながら微調整.dreamer含む4つの単語を後ろから見れば被ることがないっていうのは目から鱗!こういう発想も大事なのね...

using System;
using System.Linq;
class Program
{
    static string[] divide = {"dreamer","dream","erase","eraser"};

    static void Main(String[] args)
    {
        string S = Console.ReadLine();
        string revS = new string(S.Reverse().ToArray());
        string[] revDivide = new string[4];
        for (int i = 0; i < 4; i++) revDivide[i] = new string(divide[i].Reverse().ToArray());
        bool check1 = true;

        for(int i = 0; i < revS.Length;)
        {
            bool check2 = false;
            for(int j = 0; j < 4; j++)
            {
                string d = revDivide[j];
                if (i+d.Length<=revS.Length)
                {
                    if(revS.Substring(i, d.Length) == d)
                    {
                        i += d.Length;
                        check2 = true;
                    }
                }
            }

            if (!check2)
            {
                check1 = false;
                break;
            }

        }

        if (check1) Console.WriteLine("YES");
        else Console.WriteLine("NO");

    }
}

第10問 ABC 086C - Traveling

image.png

コメント

最後の問題.これは結構簡単だった.時刻tiからt(i+1)になったときに実行可能かどうかは次の2点を満たせばよい.

①時間の間隔>移動量

②時間の間隔―移動量が偶数

①移動量は(Xの差)+(Yの差)で出していて,最低でもこの数より時間の間隔の方が大きくないと移動できない.

②最速で次の予定地についた時あとはうろちょろして時間稼ぎをすればよい.そのとき偶数であれば行って戻ってこれるので偶数判定を行う.

using System;
using System.Linq;
class Program
{
    static int N;
    static int[] nowPos = new int[3];

    static void Main(String[] args)
    {
        N = int.Parse(Console.ReadLine());
        int[] prePos = new int[3] {0,0,0};
        bool check = true;

        for (int i = 0; i < N; i++)
        {
            nowPos = Console.ReadLine().Split(' ').Select(int.Parse).ToArray();
            int distance = Math.Abs(nowPos[1] - prePos[1]) + Math.Abs(nowPos[2] - prePos[2]);
            int interval = nowPos[0] - prePos[0];
            if (interval < distance | (interval - distance) % 2 == 1)
            {
                check = false;
                break;
            }
            prePos = nowPos;
        }

        if (check) Console.WriteLine("Yes");
        else Console.Write("No");

    }
}

最後に

とりあえずBeginners Selectionが終わったのでよかった.早速コンテスト的なのに参加していきたい.

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?