奇怪的分式|2014年蓝桥杯B组题解析第六题-fishers

奇怪的分式

上小学的时候,小明经常自己发明新算法。一次,老师出的题目是:
1/4 乘以 8/5
小明居然把分子拼接在一起,分母拼接在一起,答案是:18/45 (参见图1.png)
老师刚想批评他,转念一想,这个答案凑巧也对啊,真是见鬼!
对于分子、分母都是 1~9 中的一位数的情况,还有哪些算式可以这样计算呢?
请写出所有不同算式的个数(包括题中举例的)。
显然,交换分子分母后,例如:4/1 乘以 5/8 是满足要求的,这算做不同的算式。
但对于分子分母相同的情况,2/2 乘以 3/3 这样的类型太多了,不在计数之列!
注意:答案是个整数(考虑对称性,肯定是偶数)。请通过浏览器提交。不要书写多余的内容。

思路一:dfs全排列,最后筛选条件

思路二:也可以暴力枚举,四成循环。蓝桥杯这种暴力解题很常见!

代码一:

#include<iostream>
using namespace std;

int a[6];
int ans = 0;

//求最大公约数 
int gcd(int xx, int yy) {
	if (yy == 0) {
	    return xx;
	}
	return gcd(yy, xx%yy);
}

//测试条件是否满足 
bool test(){
	if(a[1] == a[2] && a[3] == a[4]){
		return false;
	}
	if(a[1] == a[3] && a[2] == a[4]){
		return false;
	}

	 
	int x = gcd(a[2]*a[4],a[1]*a[3]);
	int y = gcd(10*a[2]+a[4],10*a[1]+a[3]);
	//同分、化简后再比较两个分数 
	if((a[2]*a[4])/x == (10*a[2]+a[4])/y && (a[1]*a[3])/x == (10*a[1]+a[3])/y){
		return true;
	}
	return false;
}

/*
参数x表示第几个位置 
b/a c/d,这里 a是a[1]即第一个位置,b是a[2] 即第二个位置 d是a[3] c是a[4] 
*/ 
void dfs(int x){
	if(x == 5){
		if(test()){
			ans++;
//			for(int i=1;i<=4;i++){
//				cout<<a[i]<<" ";
//			}
//			cout<<endl;
			return;
		}
		return;
	}
	
	//每个位置都可以填9个数 
	for(int i=1;i<=9;i++){
		a[x] = i;
		dfs(x+1);//搜索下一个位置 
	}
}


int main(){
	dfs(1);//从a这个位置开始搜索,也就是第一个位置 
	cout<<ans<<endl;
}

代码二:

#include <iostream>
#include <cmath>
#include <cstdio>
using namespace std;

#define eps 10e-10

int main(){
    int ans = 0;
    for(int i = 1; i < 10; ++i){
        for(int j = 1; j < 10; ++j){
            for(int r = 1; r < 10; ++r){
                for(int k = 1; k < 10; k ++){
                    if(i == j || r == k)continue;
                    if(fabs( (i*10 + r)*1.0/(j*10+k) - (i*r*1.0)/(j*k)) < eps){
                        //printf("%d/%d : %d/%d\n", (i*10 + r),(j*10+k), i*r,(j*k));
                        ans++;
                    }
                }
            }
       }
    }
    cout<<ans<<endl;
	return 0;
}
posted @ 2019-01-18 10:32  fishers  阅读(348)  评论(0编辑  收藏  举报