洛谷P1008 三连击

洛谷P1008 三连击

题目链接戳这里

题目背景

本题为提交答案题,您可以写程序或手算在本机上算出答案后,直接提交答案文本,也可提交答案生成程序。

题目描述

1 1 1, 2 2 2, ⋯ \dotsm , 9 9 9 9 9 9 个数分为 3 3 3 组,分别组成 3 3 3 个三位数,且使这 3 3 3 个三位数构成 1 : 2 : 3 1:2:3 1:2:3 的比例,试求出所有满足条件的 3 3 3 个三位数。

输入格式

无输入

输出格式

若干行,每行 3 3 3 个数字。按照每行第 1 1 1 个数字升序排列。


分析

  • 思路 1:找出所有从 1 到 9 里面取三个,其和为 3 的倍数的组合,这样就找到了所有可能是最大数的那个数,除以 3 即得最小数,再乘以 2 即得最大数,过程中判断是否有重复的数,若有重复则舍去。
  • 思路 2:找出最小的那个数,显然它的范围在 123 与 329 之间,遍历其中的每一个整数,求出它的两倍以及三倍,过程中判断是否有重复的数,若有重复则舍去。

显然思路 2 更为简单,因为不涉及到组合问题,只是简单的遍历,由于数据量不是很大,预计不会 TLE。

求解

首先我们需要写判断重复数字的函数,这个是用来截去冗余数据的,能够避免不必要的运算。
注意到相乘的过程中可能出现 0 ,因此也要对数字内部的 0 做出判断。

inline int com(int a, int b) {
	int t1 = a / 100, t2 = a / 10 - 10 * (a / 100), t3 = a - 100 * t1 - 10 * t2;
	//利用int的截断特性直接取整,得到各个位置的数字
	int m1 = b / 100, m2 = b / 10 - 10 * (b / 100), m3 = b - 100 * m1 - 10 * m2;
	return !((t1 == m1) || (t1 == m2) || (t1 == m3) || (t2 == m1) || (t2 == m2) || (t2 == m3) || (t3 == m1) || (t3 == m2) || (t3 == m3));//若有相同的则返回0,否则返回1
}
inline int com(int a) {
	int h = a / 100, t = a / 10 - 10 * (h), s = a - h * 100 - t * 10;
	return !((h == t) || (h == s) || (t == s)||(h==0)||(t==0)||(s==0));//若数字内部有重复或者0则返回0,否则返回1
}

剩下的就很简单啦,直接贴上 AC 代码。

#include<stdio.h>
#include<iostream>
using namespace std;

inline int com(int a, int b);//判断两个三位数中是否存在相同的数字,若存在则返回0,不存在则返回1
inline int com(int a);//检验三位数内部有无重复数字或者0,若有则返回0,无则返回1
int main() {
	int res[999] = { 0 },m = -1;
	for (int i = 192; i < 333; ++i)
	{
		if (com(i) && com(2*i)&&com(i, 2 * i) && com(3 * i)&&com(i,3*i)&&com(2*i,3*i))
		{
			++m;
			res[m] = i;
		}
	}
	for (int j = 0; j < 999; ++j)//打表,遇到0截断
	{
		if (res[j] != 0)
			cout << res[j] << " " << 2 * res[j] << " " << 3 * res[j] << endl;
		else
			break;
	}
}
inline int com(int a, int b) {
	int t1 = a / 100, t2 = a / 10 - 10 * (a / 100), t3 = a - 100 * t1 - 10 * t2;
	int m1 = b / 100, m2 = b / 10 - 10 * (b / 100), m3 = b - 100 * m1 - 10 * m2;
	return !((t1 == m1) || (t1 == m2) || (t1 == m3) || (t2 == m1) || (t2 == m2) || (t2 == m3) || (t3 == m1) || (t3 == m2) || (t3 == m3));
}
inline int com(int a) {
	int h = a / 100, t = a / 10 - 10 * (h), s = a - h * 100 - t * 10;
	return !((h == t) || (h == s) || (t == s)||(h==0)||(t==0)||(s==0));
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值