商人过河问题
商人过河问题数学建模
做业1、2:之阳早格格创做商人过河一、问题沉述问题一:4个商人戴着4个随从过河,过河的工具惟有一艘小船,只可共时载二部分过河,包罗划船的人.随从们稀约, 正在河的任一岸, 一朝随从的人数比商人多, 便杀人越货.乘船渡河的规划由商人决断.商人们何如才搞仄安过河?问题二:假若小船不妨容3人,请问最多不妨有几名商人各戴一名随从仄安过河.二、问题分解问题不妨瞅搞一个多步计划历程.每一步由此岸到此岸或者此岸到此岸船上的人员正在仄安的前提下(二岸的随从数没有比商人多),经有限步使部分人员过河.用状态变量表示某一岸的人员情景,计划变量表示船上的人员情况,不妨找出状态随计划变更的顺序.问题便变换为正在状态的允许变更范畴内(即仄安渡河条件),决定每一步的计划,达到仄安渡河的目标.三.问题假设1. 过河途中没有会出现没有成抗力的自然果素.2. 当随从人数大于商人数时,随从们没有会改变杀人的计划.3.船的品量很佳,正在多次谦载的情况下也能仄常运做.4. 随从会听从商人的调动.四、模型形成x(k)~第k 次渡河前此岸的商人数 x(k),y(k)=0,1,2,3,4; y(k)~第k 次渡河前此岸的随从数 k=1,2,…..s(k)=[ x(k), y(k)]~历程的状态 S~允许状态集中 S={(x,y)|x=0,y=0,1,2,3,4; x=4,y=0,1,2,3,4;x=y=1,2,3} u(k)~第k 次渡船上的商人数 u(k), v(k)=0,1,2;v(k)~ 第k 次渡船上的随从数 k=1,2…..d(k)=( u(k), v(k))~历程的计划 D~允许计划集中 D={u,v|u+v=1,2,u,v=0,1,2}状态果计划而改变s(k+1)=s(k)+(-1)^k*d(k)~状态变化律供d(k)ÎD(k=1,2,….n),使s(k)ÎS 并按变化律s(k+1)=s(k)+(-1)^k*d(k)由(4,4)到达(0,0)数教模型: k+1kS =S +k k D (-1)(1)'4k k x x += (2)'4k k y y +=(3)k.k x y ≥ (4)''k k x y ≥(5)模型分解:由(2)(3)(5)可得化简得概括(4)可得k k x y =战 {}(,)|0,0,1,2,3,4k k k k k S x y x y ===(6) 还要思量{}'(',')|'0,'0,1,2,3,4k k k k k S x y x y === (7)把(2)(3)戴进(7)可得化简得{}(,)|4,0,1,2,3,4k k k k k S x y x y === (8)概括(6)(7)(8)式可得谦脚条件的情况谦脚下式 {}(,)|0,4,0,1,2,3,4;k k k k k k k S x y x y x y ====(9) 所以咱们知讲谦脚条件的面如上图所示:面移动由{}(,)|4,0,1,2,3,4k k k k k S x y x y === (8)到达{}(,)|0,0,1,2,3,4k k k k k S x y x y ===(6)时,不妨认为完毕渡河.果为移动的格数小于等于2,惟有核心面(2,2)到(6)面战(8)面的距离为2,所以核心面(2,2)成为渡河的闭键面.当咱们移动到(2,2)面时,便无法举止下去.故4个商人,4个随从,船容量为2人时,无法仄安渡河. 对付于问题二,咱们不妨修坐模型为:k+1k S =S +k k D (-1)(10)'k k x x M += (11)'k k y y M += (12)k.k x y ≥(13)''k k x y ≥ (14)u(k), v(k)=0,1,2,3; (15)通过类似于问题一的步调不妨知讲:坐标上的闭键面是(3,3),最多不妨五名商人戴五名随从往日.需要决定五名商人戴五名随从的规划可止再决定六名商人戴六名随从的规划没有成止1、五名商人戴五名随从的情况:(1)最先没有成能有三名商人先过河,二名商人一名随从过河,一名商人二名随从过河(2)三个随从先过河(5,2),回去一个随从(5,3),往日二个随从(5,1)回去一个随从(5,2),再往日三个商人(2,2),回去一个商人一个随从(3,3),再往日三个商人(0,3),回去一个随从(0,4),往日三个随从(0,1),回去一个随从(0,2)再往日二个随从(0,0)综上可知:五名商人戴五名随从,小船不妨载三部分不妨过河2、六名商人戴六名随从的情况:(1)最先没有成能有三名商人先过河,二名商人一名随从过河,一名商人二名随从过河(2)三个随从先过河(6,3),回去一个随从(6,4),往日二个随从(6,2)回去一个随从(6,3),往日三个商人(3,3),此时二岸皆是(3,3),由坐标法分解知,那是最交近末面的临界面,然而是如果回去的时间一定是回去一个商人战一个随从,如果那一步可止,后里便举止没有去综上所述,六个商人戴六个随从,小船载三部分的情况下没有克没有及渡河分离1、2知,当小船最多载三部分的时间,最多五名商人各戴一个随从不妨过河.五、模型的考验取评介由少量人的过河问题推广到了更普遍人的过河问题,使得问题变得明白有顺序.六、参照文件[1]章胤,2014年燕山大教世界大教死数教修模竞赛训练ppt,2014年4月17日。
动态规划第7讲 - 商人过河问题
8.7 动态规划:商人过河问题问题提出三名商人各带一个随从乘船渡河(A岸到B岸)。
现此岸有一小船只能容纳两人,由他们自己划行。
若在河的任一岸随从人数比商人多,他们就可能杀人越货。
不过如何乘船渡河的大权由商人们掌握。
商人们怎样才能安全过河呢?模型建立此问题可视为一个多步决策模型建立决策变量:(a,b)a------船上的商人数b ------船上的随从数a,b的取值范围:{0,1,2}且满足a+b<=2,且均为整数。
允许决策集合:D={(a,b)|a+b=1,2}={(0,1), (0,2), (1,1), (1,0), (2,0)}模型求解这样问题要求由(3,3,1)变到(0,0,0)的一条道路。
根据题意,状态转移时要满足一定的规则:1. Z从1变为0与从0变为1交替进行(船在哪个岸);2. 当Z从1变为0时,即船从A岸到B岸,A岸人数减少1或2个;即(x,y,1)→(u,v,0)时, u≤x, v≤y, u+v=x+y-1 oru+v=x+y-23. 当Z从0变为1时,即船从B岸到A岸,A岸人数增加1或2个;即(x,y,0)→(u,v,1)时, u≥x, v≥y,u+v=x+y+1 oru+v=x+y+24. 不重复已出现过的状态,如(3,3,1)→(3,1,0)→(3,3,1);按照以上规则,求解过程如下从(3,2,0)只能到达(3,3,1)/*不必考虑*/从(3,3,1)出发(3,2,0)(3,1,0)如右图(2,2,0)(3,3,1)(3,2,0)(3,1,0)(2,2,0)从(3,1,0)出发(3,3,1) /*不必考虑*/(3,2,1)/*可取*/从(2,2,0)出发(3,3,1) /*不必考虑*/(3,2,1)/*可取*/模型求解如下图所示:逐步求解,可得:模型求解由此可得到渡河策略:(3,3,1) (3,2,1)→(3,0,0)→(3,1,1)→(1,1,0)→(2,2,1)→(0,2,0)→(0,3,1)→(0,1,0) (0,0,0)(2,2,0)(3,1,0)(1,1,1)(0,2,1)模型求解思考(1) 夫妻过河问题有三对夫妻要过河,船最多可载两人。
4名商人带4名随从安全过河
4名商人带4名随从安全过河一.问题提出:4名商人带4名随从乘一条小船过河,小船每次自能承载至多两人。
随从们密约, 在河的任一岸, 一旦随从的人数比商人多, 就杀人越货.乘船渡河的方案由商人决定,商人们如何才能安全渡河呢?二.模型假设:商人和随从都会划船。
三.问题分析:商随过河问题可以视为一个多步决策过程,通过多次优化,最后获取一个全局最优的决策方案。
对于每一步,即船由此岸驶向彼岸或由彼岸驶向此岸,都要对船上的人员作出决策,在保证两岸的商人数不少于随从数的前提下,在有限步内使全部人员过河。
用状态变量表示某一岸的人员状况,决策变量表示船上的人员状况,可以找出状态随决策变化的规律,问题转化为在状态的允许变化范围内(即安全渡河条件),确定每一步的决策,达到安全渡河的目标。
四.模型构成:xk~第k次渡河前此岸的商人数,yk~第k次渡河前此岸的随从数xk, yk=0,1,2,3,4; k=1,2,……Sk=(xk, yk)~过程的状态,S~允许状态集合,S={(x,y)| x=0, y=0,1,2,3,4; x=4 ,y=0,1,2,3,4; x=y=1,2,3} uk~第k次渡船上的商人数vk~第k次渡船上的随从数dk=(uk, vk)~决策,D={(u , v)| 1=<u+v=<2,uk, vk=0,1,2} ~允许决策集合 k=1,2,……因为k为奇数时船从此岸驶向彼岸,k为偶数时船从彼岸驶向此岸,所以状态Sk随决策dk的变化规律是Sk1=Sk+(-1)k dk~状态转移律求dk∈D(k=1,2, …n), 使Sk∈S, 并按转移律由S1=(4,4)到达状态Sn1=(0,0)。
五.模型求解:1.图解法:对于人数不多的情况,可以利用图解法来求解。
在xoy平面坐标系上画出如图所示的方格,方格点表示状态s=(x,y),允许状态集合S是圆点标出的13个格子点,允许决策dk是沿方格线移动1格或2格,k为奇数时向左、下方移动,k为偶数时向右、上方移动。
商人过河问题
商人过河问题一、三名商人各带一名随从的情况1.问题(略)2.模型假设①当一边岸满足随从数大于商人数,但商人数为0时仍为一种安全状态;②小船至多可容纳2人,且渡河时由随从(或者商人)来划船。
3.分析与建模商人过河需要一步一步实现,比如第一步:两个仆人过河,第二步:一个仆人驾船回来,第三步:又是两个仆人过河,第四步:……其中每一步都使当前状态发生变化,而且是从一种安全状态变为另一种安全状态。
如果我们把每一种安全状态看成一个点,又如果存在某种过河方式使状态a变到状态b,则在点a和点b之间连一条边,这样我们把商人过河问题和图联系起来,有可能用图论方法来解决商人过河问题。
建模步骤:⑴首先要确定过河过程中的所有安全状态,我们用二元数组(,)x y 表示一个安全状态(不管此岸还是彼岸),其中x表示留在此岸的主人数,y表示留在此岸的随从数。
两岸各有十种安全状态:(0,0),(0,1),(0,2),(0,3),(2,2),(1,1),(3,0),(3,1),(3,2),(3,3)⑵在两岸的安全状态之间,如存在一种渡河方法能使一种状态变为另一种安全状态,则在这两种状态之间连一条边。
这样,得到如下一个二部图(图1),其中下方顶点表示此岸状态,上方顶点表示彼岸状态。
我们的目的是要找出一条从此岸(3,3)到彼岸(0,0)的最短路。
⑶观察发现此岸的状态(0,0),(3,0)和彼岸的状态(0,3),(3,3)都是孤立点,在求最短路的过程中不涉及这些点,把它们删去。
两岸的点用1,2, (16)新标号。
(3,3)(3,2)(3,1)(3,0)(1,1)(2,2)(0,3)(0,2)(0,3)(0,0)○②④⑥⑧⑩○○12○14○16①③⑤○⑦⑨○11○13○15○(3,3)(3,2)(3,1)(3,0)(1,1)(2,2)(0,3)(0,2)(0,3)(0,0)(图1)4.模型求解求最短路程的matlab程序如下:function route=sroute(G,opt)%求图的最短路的Dijkstra算法程序,规定起点为1,顶点连续编号%G是给定图的邻接矩阵或弧表矩阵,程序能够自动识别%当opt=0(或缺省)时求无向图的最短路,当opt=1时求有向图的最短路%d——标记最短距离%route是一个矩阵,第一行标记顶点,第二行标记1到该点的最短路,第三行标记最短路上该点的先驱顶点while 1 %此循环自动识别或由弧表矩阵生成邻接矩阵if G(1,1)==0A=G;breakelsee=Gn=max([e(:,1);e(:,2)]); %顶点数m=size(e,1); %边数M=sum(e(:,3)); %代表无穷大A=M*ones(n,n);for k=1:mA(e(k,1),e(k,2))=e(k,3);if opt==0A(e(k,2),e(k,1))=e(k,3); %形成无向图的邻接矩阵endendA=A-M*eye(n) %形成图的邻接矩阵endbreakendpb(1:length(A))=0;pb(1)=1;index1=1;index2=ones(1,length(A));d(1:length(A))=M;d(1)=0; %标记距离temp=1;while sum(pb)<length(A)tb=find(pb==0);d(tb)=min(d(tb),d(temp)+A(temp,tb)); %更新距离temp=find(d(tb)==min(d(tb))); %确定新最小距离点temp=tb(temp(1));pb(temp)=1;index1=[index1,temp];index=index1(find(d(index1)==d(temp)-A(temp,index1)));if length(index)>=2index=index(1);endindex2(temp)=index; %记录前驱顶点endroute=[1:n;d;index2];在matlab的命令窗口输入图(1)的弧表矩阵e:e=[1 2;1 4;1 10;3 4;3 6;3 10;5 6;5 8;7 14;7 16;9 8;9 12;11 12;11 14;13 14;13 16;15 16];e=[e,ones(17,1)]; %边权都设为1调用程序:route=sroute(e,0)运行结果:e =1 2 11 4 11 10 13 4 13 6 13 10 15 6 15 8 17 14 17 16 19 8 19 12 111 12 111 14 113 14 113 16 115 16 1route =1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 160 1 2 1 4 3 10 5 6 1 8 7 10 9 12 111 1 4 1 6 3 14 5 8 1 12 9 14 11 16 7这表示存在一条从1到16的长度为11的路:1 4 3 6 5 8 912 11 14 7 16,此路对应商人成功渡河的一个方案:(3,3)变为(3,1)变为(3,2)变为(3,0)变为(3,1)变为(1,1)变为(2,2)变为(0,2)变为(0,3)变为(0,1)变为(1,1)变为(0,0)即:两个仆人过河,一个仆人回来;有两个仆人过河,一个仆人回来;两个主人过河,一主一仆回来;有两个主人过河,一个仆人回来;两个仆人过河,一个仆人回来;最后两个仆人过河。
日常生活中的数学建模
改进模型:
l1: 鱼的有效长度 A1:横截面积
V l1 A 1
l1 l
2 A s 1
W kls
2
W V
数学建模
模型检验
在钓鱼比赛期间收集了有关数据:
第i条鱼 长度li
腰围si
所钓鱼的长度、腰围与重量 cm, g
1 36.83
2 31.75
3
5 32.07
6
7
8 32.07
决策 ~ 每一步(此岸到彼岸或彼岸到此岸)船上的人员 要求 ~ 在商人安全的前提下(两岸的随从数都不比商人多), 经有限步使全体人员过河。
数学建模
模型建立及求解
xk~第k次渡河前此岸的商人数 xk, yk=0,1,2,3; 设 yk~第k次渡河前此岸的随从数 k=1,2, sk=(xk , yk)~过程的状态,S ~允许状态集合 S={(x , y) x=0, y=0,1,2,3; x=3, y=0,1,2,3; x=y=1,2}
态转方程,由 s1=(3,3)到达 sn+1=(0,0)。
数学建模
模型求解
穷举法 ~ 编程上机 图解法
状态s=(x,y) ~ 16个格点
3 2
y
s1
d1
S={(x , y) x=0, y=0,1,2,3;
x=3, y=0,1,2,3; x=y=1,2} 允许状态 ~ 10个 点
1
d11 0sn+1 1 2 3 x
sk+1=sk+(-1)k dk
~状态转移方程
uk~第k次渡船上的商人数 uk, vk=0,1,2; vk~第k次渡船上的随从数 k=1,2, D={(u , v) u+v=1, 2} ~允许决策集合
【数学模型】商人们怎样过河?
问题引出问题:三名商人各带一个随从过河,一只小船只能容纳两个人,随从们约定,只要在河的任何一岸,一旦随从人数多于商人人数就杀人越货,但是商人们知道了他们的约定,并且如何过河的大权掌握在商人们手中,商人们该采取怎样的策略才能安全过河呢?这次的问题是一个很经常遇到的过河问题,其实对于该类问题,我们经过逻辑思考就可以得到答案。
但是通过数学模型的建立,我们可以得到一个通用的解答,并且通过计算机的计算我们可以大大扩大问题的规模。
问题分析因为这个问题已经理想化了,所以我们无需对模型进行假设,该问题可以看作一个多步决策问题。
每一步,船由此岸划到彼岸或者由彼岸划回此岸,都要对船上的人员进行决策(此次渡河船上可以有几名商人和几名随从),在保证安全(两岸的随从都不比商人多)的前提下,在有限次的决策中使得所有人都到对岸去。
因此,我们要做的就是要确定每一步的决策,达到渡河的目标。
建立模型记第k 次过河前此岸的商人数为x k , 随从数为y k, k = 1, 2, 3…, x k ,yk = 0, 1, 2, 3定义状态:将二维向量s k = ( x k , y k ) 定义为状态将安全渡河状态下的状态集合定义为允许状态集合,记为S = {(x,y) | x=0,y=0,1,2,3; x=y=1; x=y=2; x=3,y=0,1,2,3}记第k 次渡河船上的商人数为u k,随从数为v k定义决策:将二维向量d k = (u k , v k) 定义为决策允许决策集合记作D = {(u,v) | 1 ≤ u+v ≤ 2, u,v = 0,1,2}因为小船容量为2,所以船上人员不能超过2,而且至少要有一个人划船,由此得到上式。
由我们定义的状态s k和决策d k,我们可以发现它们之间是存在联系的:•k 为奇数是表示船由此岸划向彼岸,k 为偶数时表示船由彼岸划回此岸••状态s k是随着决策d k变化的,规律为:•s k+1 = s k + (-1)k d k我们把上式称为状态转移律,因此渡河方案可以抽象为如下的多步决策模型:求决策d k∈D(k = 1,2,…,n) , 使状态s k∈S 按照转移率,初始状态s1 = (3,3) 经有限步n 到达状态s n+1= (0,0)到这里,整个数学模型就已经非常清晰了,接下来要做的就是求解模型得出结果。
商人渡河问题的有解性分析
商人渡河问题的有解性分析商人渡河问题是一个广为流传的游戏思考问题,也被称为“渡河问题”。
它是指在一个船上有一个商人(男)、一头牛、一头羊和一块麦子,河对岸有狼、鹿和白菜。
商人想要把这四个物品都带回河对岸,但是船上最多只能允许一个物品,且狼不能和羊一起,也不能和鹿一起,否则它们就会吃掉彼此。
道商人渡河问题可以追溯到以前的数学书籍中,它的解决方案给出的步骤也不同,但最终的结果总是一致的:商人可以成功地把这四个物品都带到河对岸。
从结果来看,商人渡河问题是可以解的,但是能够解决这个问题的步骤却不止一条路径。
这个问题可以从不同的角度来分析,包括搜索、优化等方法。
搜索方法是通过搜索,尝试不同的可能性,一步一步自顶而下地思考,最终找到合理的解决方案。
优化方法是在四个物品间做出选择,从几个有限的选择中,进行一系列排序,最终找到最优的解决方案。
此外,商人渡河问题也可以用数学方法来分析,它可以用约束条件来表示,这个问题的约束条件主要是船上只能携带一个物品,狼与羊不能在一起,狼与鹿也不能在一起,等等。
有了这些约束条件,就可以用线性规划或者深度优先搜索的方法来求解这个问题,找到最优解。
尽管商人渡河问题的最终结果是一致的,可以解出渡河的步骤,但是这个问题也启示我们要注意生活中对不同工作、学习和机会的选择。
在做出正确的选择之前,要考虑清楚自己的目标,然后仔细分析条件,尽可能把握住一些突然的机遇,选择出自己的最佳行动计划,最终在自己的座位上实现自己的目标。
从以上分析可以看出,尽管商人渡河问题分析起来似乎简单,但它实际上也可以帮助人们总结解决实际问题的步骤和方法,收集有价值的经验教训。
它可以给我们提供一个有效的解决问题的框架,让我们一步一步探索,把握机遇,最终实现自己的目标。
商人过河问题数学建模c语言
商人过河问题数学建模c语言商人过河问题是一个经典的数学建模问题,通过建立数学模型,我们可以更深入地理解问题的本质,并找到最优的解决方案。
本文将通过C语言来实现这个问题的数学建模。
一、问题描述假设有n个商人要过河,每艘船只能承载一定数量的货物,而过河需要消耗一定的时间。
为了在最短的时间内完成过河任务,我们需要考虑商人的数量、船只的承载量以及过河的时间等因素,建立相应的数学模型。
二、数学建模1. 变量定义我们需要定义一些变量来描述过河过程中的各种因素,如商人的数量、船只的数量、船只的承载量、过河的时间等。
2. 算法设计算法的核心思想是利用贪心策略,尽可能多地利用船只,以减少过河的时间。
具体步骤如下:(1) 分配船只:根据船只的承载量,将商人分配到不同的船只上;(2) 计算过河时间:根据当前船只的位置和目标河岸的位置,计算每艘船只的过河时间;(3) 更新船只位置:根据过河时间,更新每艘船只的位置;(4) 重复以上步骤,直到所有商人过河。
3. C语言实现以下是一个简单的C语言程序,实现了上述算法:```c#include <stdio.h>#include <stdlib.h>int main() {int n, m, t, i, j, k;scanf("%d%d", &n, &m); // 输入商人数量和船只数量int cargo[n], time[n]; // 定义变量数组,用于存储商人和船只的信息scanf("%d%d", &cargo[0], &time[0]); // 输入第一个商人和他的过河时间for (i = 1; i < n; i++) { // 输入剩余商人和他们的过河时间scanf("%d%d", &cargo[i], &time[i]);}int boat[m]; // 定义船只数组,用于存储船只的承载量和位置信息for (j = 0; j < m; j++) { // 输入船只的承载量和位置信息scanf("%d", &boat[j]);}for (k = 0; k < n; k++) { // 模拟过河过程for (j = 0; j < m; j++) { // 遍历所有船只if (boat[j] >= cargo[k]) { // 如果船只承载量足够承载当前商人time[k] += time[k] / boat[j]; // 根据过河时间和船只速度计算剩余时间boat[j] += cargo[k]; // 将商人转移到指定位置的船只上break; // 如果找到了足够承载商人的船只,跳出当前循环继续下一轮操作}}}printf("%d\n", time[n - 1]); // 输出最后一个商人的过河时间return 0;}```三、总结通过上述C语言程序,我们可以实现商人过河问题的数学建模。
商人过河的故事
《商人过河的故事》
从前,有一个商人要过河去做生意。
他找到了一条小船,于是带着货物坐上了船。
小船在河中缓缓前行,突然,一阵强风袭来,小船开始剧烈摇晃。
商人非常害怕,他担心自己和货物会掉进河里。
就在这时,一位渔夫划船经过,他看到了商人的困境,便主动提出帮助他过河。
渔夫让商人把货物搬到他的船上,然后带着商人安全地过了河。
商人非常感激渔夫的帮助,他想要给渔夫一些钱作为报答。
但是渔夫拒绝了,他说:“我帮助你不是为了钱,而是因为这是我应该做的。
”
商人听了非常感动,他决定以后也要像渔夫一样,帮助那些需要帮助的人。
这个故事告诉我们,在生活中,我们应该学会帮助他人,不要只考虑自己的利益。
当我们帮助别人时,我们也会得到别人的帮助和感激,这将使我们的生活更加美好。
商人过河问题
商人过河问题一、三名商人各带一名随从的情况1.问题(略)2.模型假设①当一边岸满足随从数大于商人数,但商人数为0时仍为一种安全状态;②小船至多可容纳2人,且渡河时由随从(或者商人)来划船。
3.分析与建模商人过河需要一步一步实现,比如第一步:两个仆人过河,第二步:一个仆人驾船回来,第三步:又是两个仆人过河,第四步:……其中每一步都使当前状态发生变化,而且是从一种安全状态变为另一种安全状态。
如果我们把每一种安全状态看成一个点,又如果存在某种过河方式使状态a变到状态b,则在点a和点b之间连一条边,这样我们把商人过河问题和图联系起来,有可能用图论方法来解决商人过河问题。
建模步骤:⑴首先要确定过河过程中的所有安全状态,我们用二元数组(,)x y 表示一个安全状态(不管此岸还是彼岸),其中x表示留在此岸的主人数,y表示留在此岸的随从数。
两岸各有十种安全状态:(0,0),(0,1),(0,2),(0,3),(2,2),(1,1),(3,0),(3,1),(3,2),(3,3)⑵在两岸的安全状态之间,如存在一种渡河方法能使一种状态变为另一种安全状态,则在这两种状态之间连一条边。
这样,得到如下一个二部图(图1),其中下方顶点表示此岸状态,上方顶点表示彼岸状态。
我们的目的是要找出一条从此岸(3,3)到彼岸(0,0)的最短路。
⑶观察发现此岸的状态(0,0),(3,0)和彼岸的状态(0,3),(3,3)都是孤立点,在求最短路的过程中不涉及这些点,把它们删去。
两岸的点用1,2, (16)新标号。
(3,3)(3,2)(3,1)(3,0)(1,1)(2,2)(0,3)(0,2)(0,3)(0,0)○②④⑥⑧⑩○○12○14○16①③⑤○⑦⑨○11○13○15○(3,3)(3,2)(3,1)(3,0)(1,1)(2,2)(0,3)(0,2)(0,3)(0,0)(图1)4.模型求解求最短路程的matlab程序如下:function route=sroute(G,opt)%求图的最短路的Dijkstra算法程序,规定起点为1,顶点连续编号%G是给定图的邻接矩阵或弧表矩阵,程序能够自动识别%当opt=0(或缺省)时求无向图的最短路,当opt=1时求有向图的最短路%d——标记最短距离%route是一个矩阵,第一行标记顶点,第二行标记1到该点的最短路,第三行标记最短路上该点的先驱顶点while 1 %此循环自动识别或由弧表矩阵生成邻接矩阵if G(1,1)==0A=G;breakelsee=Gn=max([e(:,1);e(:,2)]); %顶点数m=size(e,1); %边数M=sum(e(:,3)); %代表无穷大A=M*ones(n,n);for k=1:mA(e(k,1),e(k,2))=e(k,3);if opt==0A(e(k,2),e(k,1))=e(k,3); %形成无向图的邻接矩阵endendA=A-M*eye(n) %形成图的邻接矩阵endbreakendpb(1:length(A))=0;pb(1)=1;index1=1;index2=ones(1,length(A));d(1:length(A))=M;d(1)=0; %标记距离temp=1;while sum(pb)<length(A)tb=find(pb==0);d(tb)=min(d(tb),d(temp)+A(temp,tb)); %更新距离temp=find(d(tb)==min(d(tb))); %确定新最小距离点 temp=tb(temp(1));pb(temp)=1;index1=[index1,temp];index=index1(find(d(index1)==d(temp)-A(temp,index1)));if length(index)>=2index=index(1);endindex2(temp)=index; %记录前驱顶点endroute=[1:n;d;index2];在matlab的命令窗口输入图(1)的弧表矩阵e:e=[1 2;1 4;1 10;3 4;3 6;3 10;5 6;5 8;7 14;7 16;9 8;9 12;11 12;11 14;13 14;13 16;15 16];e=[e,ones(17,1)]; %边权都设为1调用程序:route=sroute(e,0)运行结果:e =1 2 11 4 11 10 13 4 13 6 13 10 15 6 15 8 17 14 17 16 19 8 19 12 111 12 111 14 113 14 113 16 115 16 1route =1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 160 1 2 1 4 3 10 5 6 1 8 7 10 9 12 111 1 4 1 6 3 14 5 8 1 12 9 14 11 16 7这表示存在一条从1到16的长度为11的路:1 4 3 6 5 8 912 11 14 7 16,此路对应商人成功渡河的一个方案:(3,3)变为(3,1)变为(3,2)变为(3,0)变为(3,1)变为(1,1)变为(2,2)变为(0,2)变为(0,3)变为(0,1)变为(1,1)变为(0,0)即:两个仆人过河,一个仆人回来;有两个仆人过河,一个仆人回来;两个主人过河,一主一仆回来;有两个主人过河,一个仆人回来;两个仆人过河,一个仆人回来;最后两个仆人过河。
商人过河问题.
A (aij )nn 为 G 的邻接距阵,其中
1 aij 0
viv j E(G) viv j E(G)
i, j 1, 2, , n
定理 1:设 A(G) 为图 G 的邻接距阵,则 G 中从顶点 vi 到顶点 v j ,长度为 k 的道路的条
数为 Ak 中的 i 行 j 列元素.
证: 对 k 用数学归纳法 k 1 时,显然结论成立; 假设 k 时,定理成立, 考虑 k 1的情形.
其中 A 表示从南岸到北岸渡河的图的邻接距阵, B AT 表示从北岸到南岸渡河的图的
邻接距阵。
由定理 1、我们应考虑最小的 k ,s t ( AB)k A 中 1 行 10 列的元素不为 0,此时 2k 1
即为最少的渡河次数,而矩阵 ( AB)k A 中 1 行 10 列的元素为最佳的路径数目。
商人过河问题
三名商人各带一个随从乘船渡河,现有一只小船只能容纳两个人,由他们自己划行,若 在河的任一岸的随从人数多于商人,他们就可能抢劫财物。但如何乘船渡河由商人决定,试 给出一个商人安全渡河的方案。
首先介绍图论中的一个定理
G 是一个图,V(G)为 G 的顶点集,E(G)为 G 的边集。 设 G 中有 n 个顶点 v1, v2 , , vn ;
条,所以长为 k 1的从 vi 经 k 步到 vl 再一步到 v j 的道路共有 ai(lk) alj 条,故从 vi 经 k 1
n
步到 v j 的路径共有 ai(jk1)
a(k) il
alj
条.
l 1
下面分析及求解 假设渡河是从南岸到北岸,(m,n)表示南岸有 m 个商人,n 个随从,全部的允许状 态共有 10 个
v1 (3,3) v2 (3,2) v3 (3,1) v4 (3,(0,3) v8 (0,2) v9 (0,1) v10 (0,0)
商人们怎样安全过河的数学模型
商人们怎样安全过河的数学模型示例文章篇一:话说啊,商人们遇到了一个棘手的问题:他们得带着随从们一起过河,但随从们可不是省油的灯,一有机会就想着害商人抢货。
这河又不宽不窄,一只小船每次只能载两个人,怎么过河才能确保安全呢?咱们来聊聊这个问题吧。
首先,商人们得明白,随从们人多势众,要是他们比商人多了,那可就危险了。
所以,商人们得想个法子,让随从们没法儿耍花招。
其实啊,这个问题可以变成一个数学模型。
想象一下,我们把每次过河的人都看成是一个状态,就像打游戏一样,每过一次河就是进入了一个新的关卡。
在这个关卡里,商人们得保证自己的人数不能少于随从们。
那具体怎么做呢?咱们得先设定一些规则。
比如说,每次过河的人数只能是两个,这是小船的容量决定的。
然后,商人们得选择让哪些人过河,这就得靠他们的智慧和策略了。
想象一下这个场景:商人们先让两个随从过河,然后一个商人再带一个随从回来。
这样,河对岸的随从人数虽然多了,但商人这边还有足够的人手可以应对。
接下来,两个商人再过河,这样河对岸的商人数就比随从数多了,安全就得到了保障。
然后,再让一个商人带一个随从回来,这样河这边也有足够的商人保护随从不敢造次。
最后,两个随从再过河,问题就解决了。
这个数学模型虽然简单,但却非常实用。
它告诉我们,在面对困难和挑战时,只要我们善于运用智慧和策略,就一定能够找到解决问题的方法。
所以,商人们要想安全过河,就得靠他们的智慧和勇气了。
示例文章篇二:话说啊,有这么一个古老的谜题,叫做“商人过河”。
话说有三名聪明的商人,他们各自带着一个狡猾的随从,准备乘船过河。
这船啊,一次只能载两个人,问题就在于,这些随从们心里都有个小九九,他们密谋着,只要到了河的对岸,随从人数多于商人人数,就立马动手抢货。
这商人们也不是吃素的,他们知道随从们的阴谋,但他们毕竟都是聪明人,于是就想出了一个绝妙的策略。
咱们来想想啊,这过河其实就是一个多步决策的过程。
每次渡河,船上的人员选择都至关重要。
商人渡河数学模型
商人渡河数学模型
1、商人渡河数学模型
商人渡河是一类有趣的动态规划问题,其本质是一类路径规划问题,用数学模型可以描述为:
假设有n种物品,体积大小分别为W1,W2,……,Wn,以及一艘能承重V的船,每次船只能装载一些物品,要求在尽可能少的船次内,将物品搬运到对岸。
令Xij表示第i次船运载物品的状态,其中0≤Xij≤1,Xij=1表示船上装有第j个物品,Xij=0表示船上没有第j个物品,那么商人渡河问题就可以用下面的数学模型表示:
目标函数:
(1) Min Z=X11+X12+ (Xi)
约束条件:
(2) W1X11+W2X12+……+WnXin≤V
(3) X11+X12+……+Xin=1
(4) 0≤Xij≤1
其中,约束条件(2)表示第i次船运的负载不超过容量V,约束条件(3)表示每次船运必须装一些物品,约束条件(4)表示每次船运的物品的数量限制在0与1之间。
商人过河问题
商人过河一、问题重述和分析随从们密约, 在河的任一岸, 一旦随从的人数比商人多, 就杀人越货。
现有4名商人各带一个随从一起渡河一只船只能容纳两个人,但如何乘船渡河的大权掌握在商人的手里,商人怎样安排才能在有限步内安全渡河?二、模型假设1、在商人人数多于随从时乘船渡河的大权掌握在商人的手里;2、商人和随从都会划船;三.符号说明x表示商人人数;y表示随从人数;z表示划船到河的此岸与彼岸。
四、模型的建立与求解本题为多步决策模型,每一次过河都是状态量的转移过程。
此岸四个商人用x=0、1、2、3、4表示,此岸四个随从用y=0、1、2、3、4表示,z=0时表示划船到河的此岸时,z=1时表示划船到河的彼岸时,用有序数对(x,y,z)表示每次转移的状态量。
解决此问题就是状态量(4,4,0)转移至(0,0,1),以下就是状态量转移的全部情况(其中“!”表示不能再转移下去或与前面步骤重复):(4,4,0)→(3,3,1)↓↓(4,2,1)→(4,3,0)→(4,1,1)→(4,2,0)→(4,0,1)→(4,1,0)→!↓(2,2,1)↓!由以上关系可知,一只船只能容纳两个人的情况下,四名商人各带一个随从无法过河。
此外,如果船的容量增加到3人,那么商人就能以几种方式安全过河,以下是其中一种方案:(4,4,0)→(4,2,1)→(4,3,0)→(4,1,1,)→(4,2,0)→(2,2,1)↓(0,1,1)←(0,3,0)←(0,2,1)←(0,4,0)←(0,3,1)←(3,3,0)↓(0,2,0)→(0,0,1)五、模型推广通过以上模型的建立,若商人和随从人数增加或小船容量加大,考虑n名商人各带一随从的情况。
安全渡河问题
上图给出了一种移动方案, 上图给出了一种移动方案,经过决策 d1, d2 , d3 ⋯d11, 最终 这个方案很容易写成渡河的方案。 有 S12 = (0,0)。这个方案很容易写成渡河的方案。
分析:在问题中每一步,即船由此岸驶向彼岸或从彼 分析:在问题中每一步,
岸驶向此岸,都要对船上的人员(商人、随从各几人) 岸驶向此岸,都要对船上的人员(商人、随从各几人) 作出决策,保证安全的前提下( 作出决策,保证安全的前提下(两岸的商人数都不比随 从数少),在有限的步内使人员全部过河。用状态( ),在有限的步内使人员全部过河 从数少),在有限的步内使人员全部过河。用状态(变 来表示某一岸的人员状况,决策(变量) 量)来表示某一岸的人员状况,决策(变量)表示船上 人员的状况,可以找出状态随决策变化的规律。问题转 人员的状况,可以找出状态随决策变化的规律。 化为在状态的允许范围内(即安全渡河条件), ),确定每 化为在状态的允许范围内(即安全渡河条件),确定每 一步的决策,达到渡河的目标。 一步的决策,达到渡河的目标。
这样的话,制订安全渡河的方案归结为如下的多步决策问题: 这样的话,制订安全渡河的方案归结为如下的多步决策问题:
d k ∈ D ( k = 1, 2, ⋯ n ), 使状态 S k ∈ S 按照转移规律, 由初始状态 S1 = (3, 3)经过有限步 n到达状态 S n +1 = 0, ( 0)。
y
D = {( u , v ) | u + v = 1, 2}
(2) )
因为k为奇数时船从次岸驶向彼岸, 为偶数时由彼岸驶回此 因为 为奇数时船从次岸驶向彼岸,k为偶数时由彼岸驶回此 为奇数时船从次岸驶向彼岸 岸,所以状态 Sk 随决策 d k 变化的规律是 k (3) ) k +1 k k
商人过河数学模型
商人过河数学模型专业信息与计算科学班级113010102姓名罗彪学号11301010229一、问题重述3名商人各带一名随从乘船渡河,一只小船只能容纳二人,由他们自己划行。
随从们密约,在河的任一岸,一旦随从的人数比商人多,就杀人越货。
但是如何乘船渡河的大权掌握在商人们手中。
商人们怎样才能安全过河呢?二、问题分析商随过河问题可以视为一个多步决策过程,通过多次优化,最后获取一个全局最优的决策方案。
对于每一步,即船由此岸驶向彼岸或由彼岸驶向此岸,都要对船上的人员作出决策,在保证两岸的商人数不少于随从数的前提下,在有限步内使全部人员过河。
用状态变量表示某一岸的人员状况,决策变量表示船上的人员状况,可以找出状态随决策变化的规律,问题转化为在状态的允许变化范围内(即安全渡河条件),确定每一步的决策,达到安全渡河的目标。
三、模型假设1.每个商人和随从都会划船;2.只有一条船,且每条船上最多只能乘坐两个人;3.所有商人与随从之间没有矛盾,不会出现两人不愿意坐一条船的现象;4.船在渡河的过程中不受外界环境的影响。
四、模型的建立与求解 1.模型建立k x ~第k 次渡河前此岸的商人数,k y ~第k 次渡河前此岸的随从数k x , k y =0,1,2,3; k =1,2,… …k S =(k x , k y , c k )~过程的状态,其中k x , k y , c k 分别表示对应时刻此岸的商人,仆人数以及船的行进方向,其中c 取值1表示即将向彼岸运行,为0表示即将向此岸运行S ~ 允许状态集合,S={(x , y )| x =0, y =0,1,2,3; x =3 ,y =0,1,2,3; x =y =1,2}k u ~第k 次渡船上的商人数 k v ~第k 次渡船上的随从数k d =(k u , k v )~决策,D={(u , v )| 21≤+≤v u ,k u , k v =0,1,2} ~允许决策集合k =1,2,… …因为k 为奇数时船从此岸驶向彼岸,k 为偶数时船从彼岸驶向此岸,所以状态k S 随决策k d 的变化规律是1+k S =k S +k )1(-k d ~状态转移律求k d ∈D(k =1,2, …n), 使k S ∈S, 并按转移律由1S =(3,3,1)到达状态1+n S =(0,0,0(1))。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
商人过河问题/****************************************************M个商人与每人各带的一个仆人过河问题*船每次至多运N个人,至少要有一人划船*在任一岸,当商人数<仆人数时,仆人就杀人越货*过河由商人安排运送人员及数目*找出安全渡河的全部方法并打印(原问题中M=3,N=2)*2010-10-10 20时许(纪念伟大的双十)* LYP***************************************************//*******************************************************************本题为多步决策*若考虑只针对人数为 M = 3 对,每次过河人数最多 N = 2*可以证明路径中必须经坐标中(3,1)过至(1,1)点(过诃时),*后返回至(2,2)点,再过诃至(0,2)点(只剩2个仆人)*可以先考虑(3,3)到(3,1)点*再经(0,2)至(0,0),完成过诃(由图形的对称性关系,可以直接将(3,1)至(3,1)路径翻转,更改对应标号即可)*当然也可以用动态规划求解*本代码不限定M,N值,可通过修改宏M,N的值,求其他商人(仆人)数与最大过河人数的全部路径*******************************************************************//********************************************************************* **商人数x < 仆人数y时遭杀人越货,过河失败*对应可行域为:*x = 0, y = 0…M; elements[]中编号0…M*0 < x < M, y = x; elements[]中编号M+1…2M-1*x = M, y = 0…M; elements[]中编号2M…3M*图像上表示如下:(共 3*M+1 个点),过河即从3M点到0点*过河为左下方1/4圆区域*返回为右上方1/4圆区域*点成中心对称分布,对称的点过河、返回时点的分布情况互为相同/\ y:仆人|| 编号:M* * M\ \3M| | .* * * .* * * 2\M+2\2M+2* * * 1\M+1\2M+1-*------------------*--------> 0\ \2M|0 M x:商人|**********************************************************************/#include <stdio.h>#include <stdlib.h>#include <time.h>#define M 3/* M为商人仆人对数*/#define N 2/* N为渡河时船的最大容量*/#define STACK_Vol 30/* 栈的容量*/#define DEBUG 0/* 调试时用的宏,用于查看unit中可到达的各点编号*/enum status{ NO, YES};/* 标记elements中的go_flag(back_flag)*//* 以NO=0标记过河(返回)时没到过该点,YES=1标记过河(返回)到过该点*/struct Element{int* go; /* 存储过河时可以到达的点的动态空间指针,以-1标记结束*/int* back; /* 存储返回时可以到达的点的动态空间指针,以-1标记结束*/enum status go_flag; /* 以NO=0标记为未曾从该点过河,YES=1为曾从该点过河*/enum status back_flag; /* 以NO=0标记为未曾从该点返回,YES=1为曾从该点返回*/}elements[3 * M + 1];/*********************************************************************** *******************对可行域的点对行编号,全局变量,已被自动初始化为NULL和0*考虑到N一般较小,每点到达的点也较少,所以进行初始化并存储在动态分配的空间中*只考虑较小的N,对较大的N( >= 4),解题为平凡的,程序空间时间复杂度较大,不用本程序求解*********************************************************************** *****************/struct Stack{int array[ STACK_Vol + 1]; /* 存储经过的路径,注意:最多为30步*/int mark[ STACK_Vol + 1]; /* 存储经过对应点的链表中下次要走的元素标号*/ int top; /* 栈顶指针,指向使用的空间*/}stack;/*************************** 找出路径要用到的运算栈***************************/int path_ok; /* 记录找到路径数量*/int up_bound = 3 * M; /* 数组中最大点的下标(上界*/void unit( void); /* 初始化各点的可到达的点*/void path( void); /* 找出无环路的过河路径*/void print_path( int *); /* 打印出过河的路径*/void delete_link( void); /* 释放动态分配的内存*/int main(){clock_t start, finish;double duration; /* 测量一个事件持续的时间*/start = clock(); /* 取初始时刻时间*/unit();path();if( path_ok < 1)printf(" 没有可行路径!!!\n");delete_link();finish = clock(); /* 取完成时刻时间*/duration = (double)(finish - start) / CLOCKS_PER_SEC;printf( "\n\nProgramme Execute Time:\t%.3f seconds\n", duration );getchar();return 0;}void unit( void){int *p1, *p2;int head, rear;int vert, level, diag, sum; /* 记录垂直、水平、斜向方向上所能到的点的个数*/ int i; /* 循环变量*//* 0为目标点,不进行初始化*//* up_bound为出发点,只考虑过河情况*/level = N - M;/* level < 0,商人无法全部过河;* = 0,商人刚好全部过河,后只能从M点原样返回(无其他返回点),所以不考虑该点;* > 0,商人全部过河外可以有仆人过河*/vert = M > N ? N : M;diag = N >> 1;if( level < 0)level = 0;sum = vert + diag + level + 1;if( sum > 1){elements[up_bound].go = p1 = ( int *) malloc( sum * sizeof( int));for( i = 1; i <= level; i++)*p1++ = M - i;for( i = 1; i <= vert; i++)*p1++ = up_bound - i;for( i = 1; i <= diag; i++)*p1++ = (M << 1) - i;*p1 = -1;}//iffor( head = 1, rear = up_bound - 1; head < M; head++, rear--){/* 先计算head过河时的情况:只有仆人过河;对应的rear为返回时的情况:只商人返回*/vert = head > N ? N : head;sum = vert + 1;if( sum > 1){elements[head].go = p1 = ( int *) malloc( sum * sizeof( int));elements[rear].back = p2 = ( int *) malloc( sum * sizeof( int));if( p1 == NULL || p2 == NULL){fprintf( stderr, " 动态分配内存失败!\n\n");delete_link();exit( EXIT_FAILURE);}//iffor( i = 1; i <= vert; i++){*p1++ = head - i;*p2++ = rear + i;}//for(i)*p1 = *p2 = -1;}//if/* 再计算head返回时的情况*//* 只仆人返回*/vert = M - head;if( vert > N)vert = N;/* 商人返回一个或全部时的情况*/if( N >= M){ /* 商人可以全部返回*/level = 2 + N - M;if( head + N - M > M)level = 2 + M - head;}else if( N >= head)level = 1;elselevel = 0;/* 仆人商人返回至等数量,但不包括开始时的情况*/diag = ( N - head) >> 1;if( diag < 0)diag = 0;else if( head + diag >= M)diag = M - head - 1;sum = vert + level + diag + 1;if( sum > 1){elements[head].back = p1 = ( int *) malloc( sum * sizeof( int)); elements[rear].go = p2 = ( int *) malloc( sum * sizeof( int));if( p1 == NULL || p2 == NULL){fprintf( stderr, " 动态分配内存失败!\n\n");delete_link();exit( EXIT_FAILURE);}//iffor( i = 1; i <=vert; i++){*p1++ = head + i;*p2++ = rear - i;}if( level == 1){*p1++ = head + M;*p2++ = rear - M;}else if( level > 1){*p1++ = head + M;*p2++ = rear - M;vert = (M << 1) - 1; /* 临时挪用存储数值*/for( i = 1; i < level; i++){*p1++ = head + vert + i;*p2++ = rear - vert - i ;}}for( i = 1; i <= diag; i++){*p1++ = head + M + i;*p2++ = rear - M - i;}*p1 = *p2 = -1;}//if}//for(head)/* 单独计算 head = M 的情况*//* 返回时,head = M 的点只能返回出发点(且只可能从出发点过河到此点)。