洛谷 P1036 选数 C语言实现

P1036 选数 - 洛谷

题目描述

已知 n 个整数 x1,x2,…,xn ,以及1个整数k(k<n)。从n个整数中任选k个整数相加,可分别得到一系列的和。例如当n=4,k=3,4个整数分别为3,7,12,19时,可得全部的组合与它们的和为:

3+7+12=22
3+7+19=29
7+12+19=38
3+12+19=34。

现在,要求你计算出和为素数共有多少种。
例如上例,只有一种的和为素数:3+7+19=29。

输入输出格式

输入格式:

键盘输入,格式为:
n,k(1 <= n <= 20, k < n)
x1,x2,…,xn (1 <= xi <= 5000000)

输出格式:
屏幕输出,格式为: 1个整数(满足条件的种数)。

输入样例#1输出样例#1
4 3
3 7 12 19
1

思考

问题的关键点是,如何实现在N个数字中将其中的M个数字随机组合并不出现重复的情况。(即,位置不同但取的M个数是一样的情况为重复。 eg. 在1, 2, 3, 4, 5中取3个数,1, 2, 3 与 3, 2, 1是一种情况)
对于如何在N个数字中取M个数字这个问题,在看了其他人博客(参考博客)学习后采用的方法是

后往前选取,选定位置i后,再在前i-1个里面选取m-1个。

如 1 2 3 4 5 中选取 3 个 1、选取5后,再在前4个里面选取2个,而前4个里面选取2个又是一个子问题,递归即可。
2、如果不包含5,直接选定4,那么再在前3个里面选取2个,而前三个里面选取2个又是一个子问题,递归即可。
3、如果也不包含4,直接选取3,那么再在前2个里面选取2个,刚好只有两个。 纵向看,1、2、3刚好是一个for循环,初值为5,终值为m
横向看,该问题为一个前i-1个中选m-1的递归。

然后当m为0后,表示已经取了K个数,将b中存储的K个数相见,然后判断sum是否为素数,若为素数则计数t+1。
最后输出t就是最后答案。


AC代码

#include <stdio.h>

int sushu(int n);
void fun(int n, int m);

int a[22], b[21], n, k, t = 0;

int main()
{
    int i;
    scanf("%d%d", &n, &k);
    int b[k];
    for(i = 1; i <= n; i++)
        scanf("%d", &a[i]);
    fun(n, k);
    printf("%d", t);
}

void fun(int n, int m)//从n里面选m个数字 
{
    int i, sum;
    if(m == 0)
    {
        sum = 0;
        for(i = 0; i < k; i++)
            sum += b[i];
        if(sushu(sum))
            t++;
        return;
    }
    for(i = n ; i >= m; i--)
    {
        b[m - 1] = a[i];
        fun(i - 1, m - 1); 
    }
}

int sushu(int n)
{
    int i;
    for(i = 2; i < n; i++)
        if(n % i == 0)
            break;
    if(i == n || n == 2)
        return 1;
    else
        return 0;
}
  • 21
    点赞
  • 43
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
### 回答1: 下面是一个简单的C语言实现,用于判断一个是否为回文: ```c #include <stdio.h> #include <string.h> int main() { char n[10]; scanf("%s", n); int len = strlen(n); int flag = 1; for (int i = 0; i < len / 2; i++) { if (n[i] != n[len - i - 1]) { flag = 0; break; } } if (flag) printf("yes"); else printf("no"); return 0; } ``` 其,使用字符组 `n` 读入输入的字,`len` 记录字的长度,`flag` 为判断标志,初始化为1。然后,通过循环判断字是否为回文,若不是,则将 `flag` 设为0,跳出循环。最后,根据 `flag` 输出判断结果。 需要注意的是,输入的字可能存在前导零,因此应当使用字符组进行输入和处理,避免出现错误。 ### 回答2: 洛谷p1015是一道题目要求编写C语言程序来判断一个整数是否为回文。 回文是指正读和反读都一样的。例如121、3223都是回文,而123、12321不是回文。回文的判断可以通过将字转为字符串的方法,将该字符串与其反转后的字符串进行比较来实现。 首先,我们需要定义一个来判断一个整数是否是回文。函的输入是一个整数,输出为1表示是回文,输出为0表示不是回文。 函实现步骤如下: 1. 将整数转为字符串。 2. 定义一个变量left表示字符串的起始位置,变量right表示字符串的结束位置。 3. 通过循环比较左右两边的字符,如果不相等,则可以判断该整数不是回文,返回0。 4. 如果循环结束后没有返回0,则表示该整数是回文,返回1。 在主函,我们需要读入一个整数n,并调用判断回文的函来判断n是否是回文。然后根据函的返回值输出判断结果。 以上就是洛谷p1015回文C语言实现。其,核心思想是将整数转为字符串,并通过比较左右两边的字符来判断是否是回文。 ### 回答3: 洛谷p1015是一个关于回文的问题,需要用C语言编写。 回文是指正序和逆序都相同的字,例如121、131、1221等。 要解决这个问题,首先我们需要将输入的字进行逆序排列,然后与原字进行比较,如果相等,则说明该字是回文。 具体的解题思路如下: 1. 首先,我们需要读取输入的字n,n是一个整数。 2. 我们需要编写一个reverse,用于将字进行逆序排列。可以通过取余和除以10的方式逐位取得字。 3. 在主函,调用reverse函将输入的字n进行逆序排列,得到逆序后的字rev。 4. 进行比较,如果n和rev相等,则说明该字是回文。 5. 如果n和rev不相等,则说明该字不是回文。 6. 输出结果,如果是回文,则输出"YES",否则输出"NO"。 下面是一个简单的实现示例: ```c #include<stdio.h> int reverse(int num){ int rev=0; while(num>0){ rev=rev*10+num%10; num/=10; } return rev; } int main(){ int n; scanf("%d",&n); int rev=reverse(n); if(n==rev){ printf("YES"); }else{ printf("NO"); } return 0; } ``` 这样,我们就使用C语言解决了洛谷p1015回文问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值