競プロをしながら、節約と株式投資でセミリタイアを目指す東大卒のブログ

東大卒でメーカー勤務の私がセミリタイアするために投資や競プロを頑張っていこうという趣旨で始めたブログです。独身男性です。火木土日に更新予定です。お金について考えています。

ABC169で書いたコード

using System;
using System.Numerics;
using System.Linq;
using System.Collections.Generic;
using System.Text;
using System.Collections;

namespace debug
{
    class main
    {
        static void Main(string args)
        {
            //問題クラスを展開
            ProgramF a = new ProgramF();
            a.main();//実行する

        }

    }

    class ProgramA
    {
        public void main()
        {
            //入力
            string s = Console.ReadLine().Split(' ');
            int a = int.Parse(s[0]);
            int b = int.Parse(s[1]);

            //a*bを出す
            Console.WriteLine(a*b);

        }
    }

    class ProgramB
    {
        public void main()
        {
            //入力
            int n = int.Parse(Console.ReadLine());
            string s = Console.ReadLine().Split(' ');
            long ans = 1;
            long a = new long[n];
            long INF = 1000000000000000000;

            for (int i = 0i < ni++)
                a[i] = long.Parse(s[i]);

            //小さい順に並べる
            Array.Sort(a);
            //試しに小さい順に割ってみてもし0になるなら10の18乗超えるんで-1を出す。
            for (int i = 0i < ni++)
            {
                if (a[i] == 0)
                    break;

                INF /= a[i];
                if(INF == 0)
                {
                    Console.WriteLine("-1");
                    return;
                }

            }

            //普通にかけていく
            for (int i = 0;i < n;i++)
            {
                ans *= a[i];
            }

            //答え出力
            Console.WriteLine(ans);
            

        }
    }

    class ProgramC
    {
        public void main()
        {
            
            //入力(decimal型だと桁精度が27桁ぐらいまである
            string s = Console.ReadLine().Split(' ');
            decimal a = decimal.Parse(s[0]);
            decimal b = decimal.Parse(s[1]);

            //かけてlong型で出せば切り捨て
            Console.WriteLine((long)(a * b));
        }
    }

    class ProgramD
    {
        public void main()
        {

            //入力
            long n = long.Parse(Console.ReadLine());
            long temp = n;
            long ans = 0;

            //平方根をとって素因数分解
            for(int i = 2i <= Math.Sqrt(n);i++)
            {
                //もし割り切れるならその回数を考える
                if(temp % i == 0)
                {
                    int count = 0;
                    int count2 = 1;
                    int ans_temp = 1;
                    //割り切れなくなるまで続ける
                    while (temp % i == 0)
                    {
                        count++;//割った回数
                        temp /= i;
                        if (ans_temp == count)
                        {
                            count2++;//ここに入ってきた回数
                            ans_temp += count2;//入ってきた数を足す(すると次に入るべき数になる1→3→6→10...)
                            ans++;//ここに入れば答えなので足す
                        }
                    }
                }
            }

            //あまったのが平方根以上であればそれも素因数なので加算する
            if (temp > Math.Sqrt(n))
                ans++;

            //答え出力
            Console.WriteLine(ans);
        }
    }

    class ProgramE
    {
        public void main()
        {
            //入力
            long n = long.Parse(Console.ReadLine());
            long a = new long[n];
            long b = new long[n];

            //下限をa,上限をbとする
            for (int i = 0i < ni++)
            {
                string s = Console.ReadLine().Split(' ');
                a[i] = long.Parse(s[0]);
                b[i] = long.Parse(s[1]);
            }

            //a,bを小さい順に並べる
            Array.Sort(a);
            Array.Sort(b);

            long min = 0;
            long max = 0;

            //偶数個のときは中央二つの和を考える、奇数個は中央を考える
            if (n % 2 == 0)
            {
                min = a[n / 2] + a[n / 2 - 1];
                max = b[n / 2] + b[n / 2 - 1];

            }
            else
            {
                min = a[n / 2];
                max = b[n / 2];
            }
           
            //minからmaxまでは全部中央値として取れるのでそれで出力する
            Console.WriteLine(max - min + 1);
        }
    }

    class ProgramF
    {
        public void main()
        {

            string s = Console.ReadLine().Split(' ');
            int n = int.Parse(s[0]);
            int num = int.Parse(s[1]);
            long MOD = 998244353;

            string a = Console.ReadLine().Split(' ');
            long[,] dp = new long[n + 1,num + 1];
            //初項設定
            dp[00] = 1;

            //i番目までで、選び方によって出てくる和(j)を求める
            for (int i = 0;i < n;i++)
                for(int j = 0;j < num + 1;j++)
                {
                    long temp = j + long.Parse(a[i]);

                    //3001以上は答にならないので省く
                    //入れる場合は、Tとして選んでかつその和に組み込むのでそのまま足す。
                    if (temp <= num)
                    {
                        dp[i + 1temp] += dp[ij];
                        dp[i + 1temp] %= MOD;

                    }
                    //いれない場合は集合として成立しているのでそのi番目を選ぼうが選ばなかろうが部分として考えなければよい。よって、2倍して拡張
                    dp[i + 1j] += (dp[ij] * 2) %MOD;
                    dp[i + 1j] %= MOD;
                }

            //答はdp[n,num]で出てくる
            Console.WriteLine(dp[nnum]);

        }
    }
}