<백준> 13251

진기명기·2026년 3월 24일

코딩테스트<C++>

목록 보기
201/212

조약돌 꺼내기

문제
효빈이의 비밀 박스에는 조약돌이 N개 들어있다. 조약돌의 색상은 1부터 M까지 중의 하나이다.
비밀 박스에서 조약돌을 랜덤하게 K개 뽑았을 때, 뽑은 조약돌이 모두 같은 색일 확률을 구하는 프로그램을 작성하시오

입력
첫째 줄에 M (1 ≤ M ≤ 50)이 주어진다.
둘째 줄에는 각 색상의 조약돌이 몇 개 있는지 주어진다. 각 색상의 조약돌 개수는 1보다 크거나 같고 50보다 작거나 같은 자연수이다.
셋째 줄에는 K가 주어진다. (1 ≤ K ≤ N)

출력
첫째 줄에 뽑은 조약돌이 모두 같은 색일 확률을 출력한다. 정답과의 절대/상대 오차는 10-9까지 허용한다.

첫번째 코드는 확률식을 세워서 계산한 것이고,
두번째 코드는 조합으로 dp 점화식을 세워서 구현한 것이다.

int main() {
    ios::sync_with_stdio(false);
    cin.tie(NULL);

    int m;
    cin >> m;

    int d[51];
    int t = 0;

    for (int i = 0; i < m; i++)
    {
        cin >> d[i];
        t += d[i];
    }

    int k;
    cin >> k;

    double answer = 0.0;

    for (int i = 0; i < m; i++)
    {
        if (d[i] < k) continue;   // 여기 중요

        double result = 1.0;

        for (int j = 0; j < k; j++)
        {
            result *= (double)(d[i] - j) / (t - j);
        }

        answer += result;
    }

    cout << fixed;
    cout.precision(9);
    cout << answer;
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(NULL);

	int m;
	cin >> m;

	vector<int> v(m);
	int total = 0;

	for (int i = 0; i < m; i++)
	{
		cin >> v[i];
		total += v[i];
	}

	int k;
	cin >> k;

	double dp[51][51];

	for (int i = 0; i < 51; i++)
	{
		dp[i][0] = 1;
		dp[i][i] = 1;
	}

	for (int i = 2; i < 51; i++)
	{
		for (int j = 1; j < i; j++)
		{
			dp[i][j] = dp[i - 1][j] + dp[i - 1][j - 1];
		}
	}

	double all = dp[total][k];

	double good = 0;

	for (int i = 0; i < m; i++)
	{
		if (v[i] >= k)
			good += dp[v[i]][k];
	}

	cout << fixed;
	cout.precision(9);
	cout << good / all << "\n";
}

0개의 댓글