1. 程式人生 > >[數學模型]商人怎樣安全過河

[數學模型]商人怎樣安全過河

問題描述

三名商人各帶一個隨從乘船渡河,一隻小船隻能容納兩人,由於他們自己划行,隨從密約,在和的任一岸,一旦隨從的人數比商人多,就殺人越貨,但是如何安排乘船的大權安排在商人們的手中,商人們怎樣才能安全渡河?

模型假設

我們需要對問題做一些假設:
1. 每個商人和隨從都會划船
2. 只有一條船,且每條船上最多隻能乘坐兩個人
3. 所有商人與隨從之間沒有矛盾,不會出現兩人不願意坐一條船的現象
4. 船在渡河的過程中不受外界環境的影響

模型構造

記第 k 次過河前此岸的商人數為 xk , 隨從數為 yk , k = 1, 2, 3…, xk ,yk = 0, 1, 2, 3

定義狀態: 將二維向量 sk = ( xk , yk ) 定義為狀態

將安全渡河狀態下的狀態集合定義為允許狀態集合, 記為

S = {(x,y) | x=0,y=0,1,2,3; x=y=1; x=y=2; x=3,y=0,1,2,3}
記第 k 次渡河船上的商人數為 uk , 隨從數為 vk

定義決策: 將二維向量 dk = (uk , vk) 定義為決策

允許決策集合 記作

D = {(u,v) | 1 ≤ u+v ≤ 2, u,v = 0,1,2}
因為小船容量為2,所以船上人員不能超過2,而且至少要有一個人划船,由此得到上式。

由我們定義的狀態 sk 和決策 dk ,我們可以發現它們之間是存在聯絡的:

k 為奇數是表示船由此岸划向彼岸,k 為偶數時表示船由彼岸劃回此岸

狀態 sk 是隨著決策 dk 變化的,規律為:

sk+1 = sk + (-1)kdk
我們把上式稱為狀態轉移律,因此渡河方案可以抽象為如下的多步決策模型:

求決策 dk ∈ D(k = 1,2,…,n) , 使狀態 sk ∈ S 按照轉移率,初始狀態 s1 = (3,3) 經有限步 n 到達狀態 sn+1 = (0,0)

到這裡,整個數學模型就已經非常清晰了,接下來要做的就是求解模型得出結果。

模型求解

#include <cstdio>
#define maxn 101
int num;//number
of bus or fol int graph[maxn*maxn][maxn*maxn]; int state[maxn][maxn]; //when cross river int c_bus[5] = {2, 1, 0, 1, 0}; int c_fol[5] = {0, 1, 2, 0, 1}; int b_step[maxn*maxn]; int f_step[maxn*maxn]; bool flag = false; void DFS(int bus, int fol, int step, int dir) { b_step[step] = bus, f_step[step] = fol; if(bus == 0 && fol == 0) { for(int i = 0; i <= step; i++) { printf("(%d,%d)", b_step[i], f_step[i]); if(i != step ) printf(" -> "); } printf("\n"); flag = true; } int fa = bus * ( num + 1 ) + fol; for(int i = 0; i < 5; i++) { if(dir) { int b_next = bus - c_bus[i], f_next = fol - c_fol[i]; if(b_next >= 0 && b_next < num+1 && f_next >= 0 && f_next < num + 1 && state[b_next][f_next]) { int son = b_next * ( num + 1 ) + f_next; if(!graph[fa][son] && !graph[son][fa]) { graph[fa][son] = 1; graph[son][fa] = 1; DFS(b_next, f_next, step + 1, !dir); graph[fa][son] = 0; graph[fa][son] = 0; } } } else { int b_next = bus + c_bus[i], f_next = fol + c_fol[i]; if(b_next >= 0 && b_next < num + 1 && f_next >= 0 && f_next < num + 1 && state[b_next][f_next]) { int son = b_next * ( num + 1) + f_next; if(!graph[fa][son] && !graph[son][fa]) { graph[fa][son] = 1; graph[son][fa] = 1; DFS(b_next, f_next, step + 1, !dir); graph[fa][son] = 0; graph[fa][son] = 0; } } } } } int main() { printf("Please input the number of the businessman: "); scanf("%d",&num); for(int i = 0; i < num + 1; i++) { state[i][0] = 1; state[i][num] = 1; state[i][i] = 1; } DFS(num, num, 0, 1); if(!flag) printf("they can't cross the river."); }