基于C#实现五家共井

古代数学巨著《九章算数》中有这么一道题叫"五家共井,甲二绠(汲水用的井绳)不足,如(接上)乙一绠;乙三绠不足,如丙一绠;丙四绠不足,如丁一绠;丁五绠不足,如戊一绠;戊六绠不足,如甲一绠,皆及。

意思就是说五家人共用一口井,甲家的绳子用两条不够,还要再用乙家的绳子一条才能打到井水;乙家的绳子用三条不够,还要再用丙家的绳子一条才能打到井水;丙家的绳子用四条不够,还要再用丁家的绳子一条才能打到井水;丁家的绳子用五条不够,还要再用戊家的绳子一条才能打到井水;戊家的绳子用六条不够,还要再用甲家的绳子一条才能打到井水。

最后问:井有多深?每家的绳子各有多长?

分析:同样这套题也是属于不定方程,拿这个题目的目地就是让大家能够在不定方程组这种范畴问题上做到"举一反三",根据题意我们设井深为h,各家分别为a,b,c,d,e,则可以列出如下方程组:

2a+b=h ①

3b+c=h ②

4c+d=h ③

5d+e=h ④

6e+a=h ⑤

首先我们看下普通青年的想法,他们的想法是找 a,b,c,d,e 之间的对应关系。

依次将 ② 代入 ①,③ 代入 ②,④ 代入 ③,⑤ 代入 ④ 可得如下方程组:

a=b+c/2

b=c+d/3

c=d+e/4

d=e+a/5

从计算机的角度来说,我不希望有小数的出现,所以我可推断: c 一定是 2 的倍数,d 一定是 3 的倍数,e 一定是 4 的倍数,a 一定是 5 的倍数,根据这种关系我们就可以有如下代码:

csharp 复制代码
 namespace Test
 {
     class Program
     {
         static void Main(string[] args)
         {
             int a, b, c, d, e, h;
 
             a = b = c = d = e = h = 0;
 
             bool flag = true;
 
             while (flag)
             {
                 //4的倍数
                 e += 4;
 
                 a = 0;
 
                 while (flag)
                 {
                     //5的倍数
                     a += 5;
 
                     d = e + a / 5;
 
                     c = d + e / 4;
 
                     if (c % 2 != 0)
                         continue;
 
                     if (d % 3 != 0)
                         continue;
 
                     b = c + d / 3;
 
                     if (b + c / 2 < a)
                         break;
 
                     if (b + c / 2 == a)
                         flag = false;
                 }
             }
 
             h = 2 * a + b;
 
             Console.WriteLine("a={0},b={1},c={2},d={3},e={4} ------h={5}\n", a, b, c, d, e, h);
 
             Console.Read();
         }
     }
 }

同样我们的时间复杂度是 O(N2),急需优化。

我们再来看看文艺青年的想法,他们的想法是找 a,b,c,d,e 中的某个数与 h 的对应关系。

比如我就找 c 与 h 的对应关系,上面的 ①②③④⑤ 可写成如下方程组:

b=h-2a ⑥

c=h-3b ⑦

d=h-4c ⑧

e=h-5d ⑨

a=h-6e ⑩

将 ⑥,⑧,⑨,⑩ 分别代入 ⑦,一阵痉挛后可知:

c=(148/721)h

上面的公式也就表明了 c 和 h 的比例关系,我们令 h=721k,则 c=148k,将其代入 ⑥,⑦,⑧,⑨,⑩ 可得如下方程组

a=265k

b=191k

c=148k

d=129k

e=76k

x=721k

又因为 k>0,所以题目有无数个解。这里我就取 0<k<5,否则绳子已经到达极限了,需要用蛟龙号去深潜了。

csharp 复制代码
 namespace Test
 {
     class Program
     {
         static void Main(string[] args)
         {
             for (int k = 1; k < 5; k++)
             {
                 int h = 721 * k;
 
                 int a = 265 * k;
 
                 int b = 191 * k;
 
                 int c = 148 * k;
 
                 int d = 129 * k;
 
                 int e = 76 * k;
 
                 Console.WriteLine("a={0},b={1},c={2},d={3},e={4} ------h={5}\n", a, b, c, d, e, h);
             }
 
             Console.Read();
         }
     }
 }
相关推荐
Source.Liu2 分钟前
Rust 整数
开发语言·rust
Yanbin_Q6 分钟前
从 Rust 官方文档理解 Ownership
开发语言·后端·rust
原来是猿13 分钟前
类和对象(上)
c语言·开发语言·数据结构·c++·算法
码喽不秃头14 分钟前
java中BigInteger类和BigDecimal类
java·开发语言
码界领航18 分钟前
Dev C++ 无法使用to_string方法的解决
开发语言·c++
qq_4340859018 分钟前
Day 65 || SPFA、判断负权回路、bellman_ford之单源有限最短路
算法
醇醛酸醚酮酯21 分钟前
二叉树遍历的非递归实现和复杂度分析
数据结构·算法·ducker成长之路
无限大.22 分钟前
C++ STL -- 模版
开发语言·c++
OTWOL33 分钟前
C语言中操作符详解(中)
c语言·开发语言·c++
lllsure35 分钟前
二分搜索的三种方法
算法