任意变换顺序,a[i]*a[i+1)是4的倍数

本篇介绍了一个趣味算法题目,名为“GJJ的日常之暴富梦”。题目要求判断一组评委给出的分数是否可以通过调整顺序使任意相邻两数的乘积成为4的倍数,进而决定GJJ是否能够通过选秀比赛。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Contest - 河南省多校连萌(四)

Problem D: GJJ的日常之暴富梦

Time Limit: 1 Sec   Memory Limit: 128 MB
Submit: 646   Solved: 100

Submit Web Board

Description

GJJ有个暴富梦。
他梦想着有一天,他突然有了很多很多钱,他梦想着等他有钱了,要在自己的家里挖两个游泳池,
一个用来洗脸,一个用来洗脚。他梦想着等他有钱了,他要买两辆劳斯莱斯,每次出门时候自己
开一辆,车后面再拖一辆。
GJJ每天都沉浸在他的暴富梦中。这天他突然打听到,有个选秀节目冠军奖金100亿,他知道他的暴富梦
马上就要实现了。他去参加选秀了。
GJJ表演了一套出神入化的天鹅舞,迷倒了一大片观众和评委,赢来了一阵阵掌声。每个评委都
给他打了一个很高的分数,并且每个评委都有一个编号。但是这个选秀节目通过的条件不是比分数高低。这个奇葩的条件是:
在所有评委给选手打的分数中,如果可以经过任意的变换评委的顺序,使得每相邻的两个评委打的
分数的乘积是4的倍数,则通过,否则就失败。GJJ 现在急切的想要知道他是否通过了,你能帮帮他么。

Input

输入有多组样例,每组样例首先是一个n,代表评委的数量(2<=n<=1000000)
接下来有n个数,每个数a[i]表示评委给GJJ打的分数(0<a[i]<=1000000000)

Output

对于每组样例,如果GJJ能通过比赛(即经过任意变换顺序后,对于每个分数,如果a[i]*a[i+1)是4的倍数),输出"Pass",否则的话,输出"Not Pass"。

Sample Input

4
1 2 3 4

Sample Output

Not Pass


题解:要使得每相邻的两个数的乘积是4的倍数,那么相邻的两个数一定为以下两种情况:
1、两个偶数
2、一个奇数和一个4的倍数
(因为两个奇数乘积一定不是4的倍数,一个奇数一个偶数乘积未必是4的倍数)
所以问题就转换为统计奇数的个数和4的倍数的个数。如果4的倍数的个数大于等于
奇数的个数,那么一定可以满足条件。(可以使出现的每一个奇数都能搭配上一个4的倍数)
另外还需要考虑一个特殊情况,例如 “1 4 1 ”这种,恰好两个奇数公用一个4的倍数。
所以特判,当奇数个数 + 4的倍数个数 = n 并且 奇数个数比4的倍数个数多1,则也满足条件。
其余情况均不满足条件。

#include<bits/stdc++.h>
#define N 1000000+10
using namespace std;
int a[N];
int main()
{
    int n,cnt_4,cnt_ji;	//分别记录4的倍数,奇数个数 
    while(cin>>n)
    {
        int flag=0;
        cnt_4=cnt_ji=0;
        for(int i=0; i<n; i++)
        {
            cin>>a[i];
            if(a[i]%4==0)
                cnt_4++;
            if(a[i]%2!=0)
                cnt_ji++;
        }
        if(cnt_ji==cnt_4+1&&cnt_ji+cnt_4==n)	//只有奇数和4的倍数 ,1 4 1这种情况 
            flag=1;
        if(cnt_4>=cnt_ji)	//4的倍数大于奇数个数 
            flag=1;
        if(flag==1)
            cout<<"Pass"<<endl;
        else
            cout<<"Not Pass"<<endl;
    }
    return 0;
}


``` #include <bits/stdc++.h> using namespace std; const int N = 2e4 + 10, M = 1e9 + 7; long long n, k, s; long long dp[110][N]; void solve() { cin >> n >> s >> k; // 预处理:减去每个人最少捐赠的总金额 s -= (((n - 1) * n) / 2) * k; // 初始化:0个人捐赠0元的方案数是1 dp[0][0] = 1; // 动态规划求解 for (int i = 1; i <= n; ++i) { for (int j = 0; j <= s; ++j) { if (j < i) { dp[i][j] = dp[i - 1][j]; // 无法捐赠,继承之前状态 } else { dp[i][j] = (dp[i - 1][j] + dp[i][j - i]) % M; // 状态转移 } } } // 输出结果 cout << dp[n][s]; } int main() { int _ = 1; while (_--) solve(); return 0; }```# T525611 募捐 ## 题目描述 慈善晚宴开始了,一共有 $n$个人,总共要募捐 $s$元。 募捐从第一个人开始,按照顺序依次进行,直到第 $n$ 个人。由于这个顺序是按照社会地位来排的,后面的人总是要比前面的人多捐至少 $k$ 元。第一个人没有限制,可以捐任意多钱,甚至可以捐 $0$ 元。 你很好奇,想知道一共有多少种可能性,使得 $n$ 个人正好捐了 $s$ 元。 ## 输入格式 一行,三个整数 $n,s,k$ 。 ## 输出格式 一个整数,表示 $n$个人正好捐了 $k$元的方案数。由于答案可能很大,你只需要输出答案 mod $10^9+7$的结果。 ## 输入输出样例 #1 ### 输入 #1 ``` 3 10 1 ``` ### 输出 #1 ``` 8 ``` ## 输入输出样例 #2 ### 输入 #2 ``` 20 1000 10 ``` ### 输出 #2 ``` 0 ``` ## 输入输出样例 #3 ### 输入 #3 ``` 55 2555 1 ``` ### 输出 #3 ``` 966857668 ``` ## 说明/提示 ![](https://cdn.luogu.com.cn/upload/image_hosting/6yhimwrm.png) ![](https://cdn.luogu.com.cn/upload/image_hosting/vm5r2efx.png) $1\le k,n \le 100$ $1\le s\le 2\times 10^4$ 帮我写注释
03-13
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值