运用模逆运算(同余方程)来解决Matlab课上的一道思考题
一个篮子有K个鸡蛋:
2个2个拿剩1个;
3个3个全部拿完;
4个4个拿剩1;
5个5个拿剩4个;
6个6个拿剩3个;
7个7个拿全部拿完;
8个8个拿剩1个;
9个9个拿全部拿完;
求篮子里鸡蛋的个数K
虽然这是一道matlab拿来玩的题目,可是我觉得完全可以拿来做笔试题或者面试题。仔细想还是有点考算法能力的。
resultNum = [];
resultIndex = 0;
i = 1;
while 63*i < 1000000000
while 1
i = i + 2;
if max(abs(rem(63*i, inputDivisors) - modResult)) == 0
break
end
end
resultNum(resultIndex + 1) = 63*i;
resultIndex = resultIndex +1;
end
disp(resultIndex)
resultNum( resultIndex)
我们先来看下枚举本身为什么会慢的问题,我们可以看到,枚举每一次都要尝试和M个除数进行相除,然后得到余数,除了本身求余这种对CPU的ALU指令非常不友好导致速度有瓶颈以外,还有最致命的就是枚举要通过不断地 “试验” 来得到判断的目的,当然这个过程可以裁枝(就像上面所说的观察法),但是这个问题的裁枝也只是减少了复杂度MN前面的常数系数。
step1: 63k
step2: 63*(2k + i)
step3: 63*(2(2k + j) + i) = 63*(4k + 2j + i)
step4: 63*(4(5k + l) + 2j + i) = 63*(20k + 4l + 2j + i)
step5: 63*(20(k + m) + 20l + 2j + i) = 63*(20k + 20m + 4l + 2j + i)
step6: 63*(20(2k + n) + 20m + 4l + 2j + i) = 63*(40k + 20n + 20m + 4l + 2j + i)
设存在方程 ax + by = L
如果该方程的线性同余形式为 ax = L mod b
如果这个形式下该方程有解,那么L一定且仅满足gcd(a,b) == L时成立
定理一:如果d = gcd(a, b),则必能找到正的或负的整数k和l,使d = ax+ by。
定理二:若gcd(a, b) = 1,则方程ax ≡ c (mod b)在[0, b-1]上有唯一解。
定理三:若gcd(a, b) = d,则方程ax ≡ c (mod b)在[0, b/d - 1]上有唯一解。
%External Euclidean Algorithm
%xPtr is a pointer, x is p(1), y is p(2)
%warnning: the index is different from C\C++
function result = ExMyGcd(n,m, xPtr)
if n == 0
find= m;
xPtr.value(1) = 1;
xPtr.value(2) = 0;
else
num=[0,0];
numPtr=libpointer('int32Ptr', num);
find = ExMyGcd(rem(m,n), n, numPtr); xPtr.value(1) = numPtr.value(2);
xPtr.value(2) = numPtr.value(1) - numPtr.value(2)*floor(m/n);
end
result = find;
end %ExMyGcd
obj = SearchPolyFuntionHelper(inputDivisors, modResult, 1, 1, 0);
obj = DoSearch(obj);
n = 0;
if obj.m_isValid == 0
disp('invalid condition') %不满足同余定理的条件,直接舍弃
else
coefficient = obj.m_coefficient;
innerCofficient = obj.m_innerCofficient;
offset = obj.m_offset; resultNum = [];
resultIndex = 0;
while 1
r = coefficient *(innerCofficient * n + offset);
if (r >=1000000000)
break
end
resultNum(resultIndex + 1) = r;
resultIndex = resultIndex + 1;
n = n + 1;
end
disp(resultIndex)
resultNum( resultIndex)
end
classdef SearchPolyFuntionHelper
methods (Access = 'public')
function obj = SearchPolyFuntionHelper(divisors, modResult, coefficient, innerCofficient, offset)
obj.m_divisors = divisors;
obj.m_modResult = modResult;
obj.m_coefficient = coefficient;
obj.m_innerCofficient = innerCofficient;
obj.m_offset = offset;
obj.m_isValid = 1; end % SerchPolyFuntionHelper function obj = DoSearch(obj)
sum = size(obj.m_divisors);
for n = 1: sum(2)
if obj.m_modResult(n) == 0
mCoefficient = obj.m_coefficient;
mDivisors = obj.m_divisors(n);
gcdDivisor = gcd(mCoefficient, mDivisors);
if gcdDivisor ~= mDivisors
obj.m_coefficient = mCoefficient*mDivisors / gcdDivisor;
end
end
n = n + 1;
end
for n = 1: sum(2)
if obj.m_modResult(n) ~= 0
obj = Expand(obj, n); if obj.m_isValid ==0
return
end
end
n = n + 1;
end
end %DoSearch
end
methods(Access = 'private')
%Expand the cofficent
function dObj =Expand(dObj, index)
%先确定最小公倍数
mDivisors = dObj.m_divisors(index);
gcdDivisor = gcd(dObj.m_coefficient * dObj.m_innerCofficient, mDivisors); oldInnerCofficient = dObj.m_innerCofficient;
dObj.m_innerCofficient = oldInnerCofficient * mDivisors / gcdDivisor; %确定偏移量 coefficient*oldInnerCofficient*i + offset ( i>=0)
mod = dObj.m_modResult(index);
offset = dObj.m_offset; offsetMod = rem(offset * dObj.m_coefficient, mDivisors); %得到偏移量的余数
lastMod = rem(mod - offsetMod + mDivisors, mDivisors); if lastMod ~=0 %如果当前offset已经可以满足余数,就不需要再加了
num=[0,0];
xPtr=libpointer('int32Ptr',num);
mul = dObj.m_coefficient* oldInnerCofficient;
gcdDivisor = ExMyGcd(mDivisors, mul, xPtr); if rem(lastMod, gcdDivisor) ~=0
dObj.m_isValid = 0; %如果不满足同余定理,说明无法找到这样的多项式,直接退出
return;
end newCofficient= xPtr.value(1);
newCofficient = newCofficient * lastMod/ gcdDivisor;
dObj.m_offset = newCofficient* oldInnerCofficient + offset;
end
end%Expand
end
properties (Access = 'public')
m_coefficient;
m_innerCofficient;
m_offset;
m_divisors;
m_modResult;
m_isValid;
ElementIndex;
end
properties (Dependent)
Modulus
end
end
%External Euclidean Algorithm
%xPtr is a pointer, x is p(1), y is p(2)
%warnning: the index is different from C\C++
function result = ExMyGcd(n,m, xPtr)
if n == 0
find= m;
xPtr.value(1) = 1;
xPtr.value(2) = 0;
else
num=[0,0];
numPtr=libpointer('int32Ptr', num);
find = ExMyGcd(rem(m,n), n, numPtr); xPtr.value(1) = numPtr.value(2);
xPtr.value(2) = numPtr.value(1) - numPtr.value(2)*floor(m/n);
end
result = find;
end %ExMyGcd
也就是
暴力算法与用同余方程的算法在需要算出n以内鸡蛋数的条件下,进行时间复杂度的比较结果。
clear
clc
mainFun()
function mainFun()
modResult = [1,0,1,4,3,0,1,0];
inputDivisors = [2:9]; disp('-----------------Mytest-----------------')
tic;
obj = SearchPolyFuntionHelper(inputDivisors, modResult, 1, 1, 0);
obj = DoSearch(obj);
n = 0;
if obj.m_isValid == 0
disp('invalid condition') %不满足同余定理的条件,直接舍弃
else
coefficient = obj.m_coefficient;
innerCofficient = obj.m_innerCofficient;
offset = obj.m_offset; resultNum = [];
resultIndex = 0;
while 1
r = coefficient *(innerCofficient * n + offset);
if (r >=1000000000)
break
end
resultNum(resultIndex + 1) = r;
resultIndex = resultIndex + 1;
n = n + 1;
end
disp(resultIndex)
resultNum( resultIndex)
end toc;
vpa(tic - toc); disp('-----------------OridnaryTest-----------------')
tic;
resultNum = [];
resultIndex = 0;
i = 1;
while 63*i < 1000000000
while 1
i = i + 2;
if max(abs(rem(63*i, inputDivisors) - modResult)) == 0
break
end
end
resultNum(resultIndex + 1) = 63*i;
resultIndex = resultIndex +1;
end
disp(resultIndex)
resultNum( resultIndex)
toc;
vpa(tic - toc);
end
运用模逆运算(同余方程)来解决Matlab课上的一道思考题的更多相关文章
- 如何解决 Matlab 画图时中文显示乱码的问题?
使用的是win10系统,从前几个月某一天,我的matlab的figure里的中文都变成了口口.很是郁闷,还以为是动到了什么配置引起的. 前几天更新了matlab 2018b,发现还有这个问题.就觉得不 ...
- 20165223《信息安全系统设计基础》第九周学习总结 & 第八周课上测试
目录 [第九周学习总结] 教材内容总结 [第八周课上测试] (一)求命令行传入整数参数的和 (二)练习Y86-64模拟器汇编 (三)基于socket实现daytime(13)服务器和客户端 参考资料 ...
- 20175316盛茂淞 2018-2019-2 《Java程序设计》第2周课上测试总结
20175316 2018-2019-2 <Java程序设计>第2周课上测试总结 上周考试题目总结 题目1 题目要求: 在Ubuntu中用自己的有位学号建一个文件,教材p29 Exampl ...
- 20165305 苏振龙《Java程序设计》第四周课上测试补做
第一次测试 第二次测试 第三次测试 上传代码 第四次测试 总结 之前我一直在git bash进行程序设计,但是对于我来说操作起来有点困难,所以我改用了虚拟机,之后之前一直困扰我的问题在虚拟机下就没有了 ...
- # 2017-2018-1 20155302 课下实践IPC及课上补充
课上实践补交 题目二要求: 学习使用stat(1),并用C语言实现 提交学习stat(1)的截图 man -k ,grep -r的使用 伪代码 产品代码 mystate.c,提交码云链接 测试代码,m ...
- 2017-2018-1 20155232 《信息安全系统设计基础》第四周学习总结以及课上myod练习补充博客
2017-2018-1 20155232 <信息安全系统设计基础>第四周学习总结以及课上myod练习补充博客 课上myod练习 1 参考教材第十章内容 2 用Linux IO相关系统调用编 ...
- 2018-2019-1 20165330 《信息安全系统设计基础》第六周课上测试ch02&课下作业
课上测试 测试-3-ch02 任务详情 编写一个程序 "week0203学号.c",运行下面代码: 1 short int v = -学号后四位 2 unsigned short ...
- Java实验--关于课上找“水王”问题分析
问题的表述就是说有那么一个人,他在一个论坛上发帖,然后每贴必回,自己也发帖.那么这个人在发帖的数目上就超过了整个论坛的帖子数目的一半以上. 我对这个问题一开始的思路是,用SQL语句获取整个列表中的数据 ...
- 关于转入软件工程专业后第二次java课上作业的某些体会
今天是第二周的java课. 自从转入了软件工程专业后,在我没有学习c++的基础上,直接开始了学习java的过程.不得不说过程很艰辛.今天下午老师让编写一个随机产生作业的软件.而我的基础差到都不知道如何 ...
随机推荐
- Lightoj1015【基础题】
题意: 计算输入数>0的所有和: 思路: 直接干... #include<cstdio> #include<queue> #include<map> #inc ...
- beanutils包下载
- PostgreSQL - 怎么将时间转换成秒
保留原来的毫秒值 select extract(epoch from '03:21:06.678'::time); 这个extract(epoch from )函数得到的是时间是秒单位,如果需要毫秒值 ...
- JAVA列出某文件夹下的所有文件
import java.io.*; public class ListFiles { private static String s = ""; private static Bu ...
- C# 基础之索引器
当一个类有数组成员时,索引器将大大简化对类中数组成员的访问 索引器类似于属性有get与set访问器 列如: 使用: 总结:从以上代码可以看出索引器也是对私有字段进行访问的方式,但此时的私有字段是数组类 ...
- 292. Nim游戏
292. Nim游戏 class Solution(object): def canWinNim(self, n): """ :type n: int :rtype: b ...
- Qt 进程和线程之一:运行一个进程和进程间通信
Qt提供了对进程和线程的支持.本节讲述了怎样在Qt应用程序中启动一个进程,以及几种常用的进程间通信方法.如果对进程和线程的概念不是很了解,可以看我的另一篇博客:[多进程和多线程的概念. 设计应用程序时 ...
- 2017浙江工业大学-校赛决赛 小M和天平
Description 小M想知道某件物品的重量,但是摆在他面前的只有一个天平(没有游标)和一堆石子,石子可以放左边也可以放右边.他现在知道每个石子的重量.问能不能根据上述条件,能不能测出所问的重量. ...
- STM32的低功耗模式
一 待机模式standby和STOP模式的区别: 进入低功耗模式:都一样,都是先关闭相应时钟,关闭相应外设,配置相应所有IO口(浮动输入),然后配置相应的唤醒中断源,中断影响的O口,然后调用相应函数进 ...
- The Weakest Sith
http://codeforces.com/gym/101149/problem/F 题目要输出最丑陋的衣服.所以每件衣服都要和其他衣服比一次. 但是注意到,能赢一件衣服的衣服,就算是好衣服了. 那么 ...