1~n的之间的k个数组成和为n的方案数(动态规划)

绯色的子弹

Description

众所周知,夏季奥林匹克运动会时隔56年第二次在东京举办,紧接着出来的《名侦探柯南 M24绯色的子弹》竟也是有奥运会的背景,最重要的是重归主线!!!(赤井秀一好帅)

讲个笑话:太让我失望了,这次剧场版居然不是讲赤井秀一参加奥运射击项目吊打各路高手包揽所有金牌的,千载难逢的机会啊[蛆音娘_滑稽]

阿笠博士带着包括柯南在内的少年侦探团在奥运会现场,大家都知道,阿笠博士有个猜谜语的习惯,今天阿笠博士给出了这样一个题目:

给定两个整数nnnkkk,含义为用恰好kkk个正整数数字组合为和为nnn的方案数,对于n=4,k=2n=4,k=2n=4,k=2,{1,31,313}和{3,13,13,1}是两种方案,还有一点就是得保证,组合成答案的这kkk个数各不相同,所以{2,22,22,2}不能算是一种方案,那么下面就由聪明的你来代替侦探团来解答这个问题吧。

Input

输入包括一行,两个整数,分别是n,k(1≤ n≤200,1≤k≤10)n,k(1 \leq  n \leq 200,1 \leq k \leq 10)n,k1 n200,1k10

Output

输出只有一个整数,也就是答案。

Sample Input 1

10 2

Sample Output 1

8

Sample Input 2

100 3

Sample Output 2

4704

Source

nuoyanli

思路

这题说实话,到现在我还不是太理解这 dp 的思路

代码

#include<iostream>
using namespace std;

#define ll long long 
ll dp[15][205];             //dp[i][j] 表示由i个数的和组成的值 j

int main()
{
    /* freopen("A.txt","r",stdin); */
    int n, k;
    scanf("%d %d", &n, &k);
    //初始(只有从dp[0][0] 转化过来的方案才可行)
    dp[0][0] = 1;
    
    for(int i = 1; i <= n; i ++)                //这次我们选择加的数是 i
        for(int j = k; j >= 1; j --)            //假设组成的数的数量为 j 个
            for(int u = n; u >= i; u --)        //u是我们要假设的数
                dp[j][u] += dp[j - 1][u - i];

    ll ans = dp[k][n];

    //乘以全排
    for(ll i = 1; i <= k; i ++)
        ans *= i;
    printf("%lld\n", ans);

    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 以下是一个简单的C语言程序,可以实现这个功能: ```c #include <stdio.h> #include <string.h> int main() { char a[1000]; int k, n, i, j; scanf("%s %d", a, &k); n = strlen(a); // 从左往右删除字,直到删除k个 for (i = 0; i < k; i++) { // 找到第一个比右边字大的字,删除它 for (j = 0; j < n - 1; j++) { if (a[j] > a[j+1]) { // 删除a[j] memmove(a+j, a+j+1, n-j-1); n--; break; } } } // 输出剩余字 printf("%s\n", a); return 0; } ``` 程序首先读入一个字符串a和一个整k,然后计算字符串长度n。接下来,程序从左往右删除字,直到删除k个为止。具体地,程序从左往右遍历字符串,找到第一个比右边字大的字,删除它。删除操作使用了C标准库函`memmove`,它可以将字符串中从给定位置开始的若干个字符向左移动若干个位置,从而删除一个字符。最后,程序输出剩余字。 ### 回答2: 你可以使用C语言编程来实现这个功能。具体步骤如下: 1. 首先,创建一个函来实现删除字的功能。这个函将接受两个参:原始的正整a和要删除的个数k。函应该返回一个新的正整作为结果。 2. 在函内部,将正整a转换为一个字符串。你可以使用sprintf函来实现这一点。 3. 创建一个循环来删除k个数字。在每一次循环中,找到当前字符串中最小的字,并将其删除。你可以使用strchr函来找到字符串中的特定字符,并使用strcpy函来删除它。 4. 循环结束后,将剩余的字符串转换回一个正整。你可以使用atoi函来实现这一点。 5. 最后,将新的正整作为结果返回。 以下是一个示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> int deleteDigits(int a, int k) { char str[11]; sprintf(str, "%d", a); int len = strlen(str); while (k > 0) { int min_index = 0; for (int i = 1; i < len && i <= k; i++) { if (str[i] < str[min_index]) { min_index = i; } } strcpy(&str[min_index], &str[min_index + 1]); len--; k--; } return atoi(str); } int main() { int a, k; printf("请输入一个整:"); scanf("%d", &a); printf("请输入要删除的个数:"); scanf("%d", &k); int result = deleteDigits(a, k); printf("删除字后的结果为:%d\n", result); return 0; } ``` 请注意,这只是一个简单的示例代码,可能存在一些边界情况没有考虑到,比如当k大于等于字位时的情况。你可以根据实际需求对代码进行修改和优化。 ### 回答3: 问题描述: 给定一个n位的正整a,删除其中任意k(k<n)个数字后,剩余字按原次序组成一个新的正整。要求使用C语言实现。 解决方案: 首先,我们可以将正整a转化为字符串形式,以便进行操作。然后,我们可以通过比较相邻字符的大小,删除字,直到删除k个数字为止。具体步骤如下: 1. 将正整a转化为字符串形式,命名为str_a。 2. 创建一个新的字符串,命名为ans,用于存储最终结果。 3. 初始化一个变量count为0,用于记录已删除的个数。 4. 使用循环遍历str_a,判断相邻字符的大小。 5. 如果当前字符大于下一个字符,则删除当前字符,并将count加1。如果count等于k,跳出循环。 6. 如果count小于k,说明还需要继续删除字。将剩余的字符添加到ans中。 7. 如果count等于k,说明已经删除了足够的字。将剩余的字符添加到ans中,并结束循环。 8. 输出ans作为最终结果。 以下是C语言代码的实现: #include<stdio.h> #include<string.h> void delete_digits(char* str_a, int k) { int n = strlen(str_a); char ans[100]; int count = 0; int i; for(i = 0; i < n; i++) { while(count < k) { if(str_a[i] > str_a[i+1]) { count++; break; } else { ans[i-count] = str_a[i]; } i++; } ans[i-count] = '\0'; } printf("The new number after deleting %d digits is: %s\n", k, ans); } int main() { char str_a[100]; int k; printf("Enter the number: "); scanf("%s", str_a); printf("Enter the number of digits to be deleted: "); scanf("%d", &k); delete_digits(str_a, k); return 0; } 运行示例: 输入: Enter the number: 123456789 Enter the number of digits to be deleted: 3 输出: The new number after deleting 3 digits is: 1236789
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值