视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
JavaScript趣题:迷路的牛仔
2020-11-27 20:25:20 责编:小采
文档


很久以前,有一条通往西部无法地带的路......

一位年轻的牛仔,根据指示,得从一个地方前往另一个地点。指示像这样"NORTH", "SOUTH", "WEST", "EAST"。

很明显,"NORTH"和"SOUTH"是相反的方向, "WEST"和"EAST"也是相反的。

走一个方向,然后又往回走,无疑在做无用功。

在这人迹罕至的西部荒野,糟糕的天气,匮乏的水源,节约点体力很重要,否则很可能死于非命!

怎样走最明智的路线,这很重要!

给牛仔的指示,像下面这样:

plan = ["NORTH", "SOUTH", "SOUTH", "EAST", "WEST", "NORTH", "WEST"]

你能一眼看出,"NORTH"和"SOUTH",这样的路线显然是不合理的,最好呆在原地。
所以,你的任务是,怎样精简路线,以节约体力。

一个更好的方案应该像这样:

plan = ["WEST"]

另一个例子:

["NORTH", "SOUTH", "EAST", "WEST"]

在这里,"NORTH"和"SOUTH"做抵消,"EAST"和"WEST"做抵消,最后返回空数组。

再看个复杂点的例子:

["NORTH", "EAST", "WEST", "SOUTH", "WEST", "WEST"]

"EAST", "WEST"作抵消,得到 ["NORTH", "SOUTH", "WEST", "WEST"]

"NORTH", "SOUTH"作抵消,最终得到["WEST", "WEST"]。
但是,请注意,下面这种情况是没办法抵消的:

["NORTH", "WEST", "SOUTH", "EAST"]

因为"EAST", "WEST"或"NORTH", "SOUTH"并不相邻,而是隔开的。

咋们来看看怎么编写一个这样的减少路线函数吧。

它接受一个字符串数组作为参数,返回新的字符串数组。

像这样:

dirReduc(["NORTH", "SOUTH", "SOUTH", "EAST", "WEST", "NORTH", "WEST"]) // ["WEST"]
dirReduc(["NORTH", "SOUTH", "SOUTH", "EAST", "WEST", "NORTH"]) // []

第一步,建立方向的映射关系,哪些是相反的方向:

var opposite = {
 "NORTH":"SOUTH",
 "SOUTH":"NORTH",
 "EAST":"WEST",
 "WEST":"EAST"
};

接着,从后向前遍历字符串数组plan,遇到相反的方向,就从数组中移除,抵消。

然后,开始第二轮遍历,移除,抵消,第三轮,第N轮,直到找不到相反的方向,跳出循环。

此时的数组就是经过精简后的最佳路线。

function dirReduc(arr){
 var flag = false;
 while(!flag){
 for(var i=arr.length-2,flag=true;i>=0;i--){
 if(opposite[arr[i]] === arr[i+1]){
 arr.splice(i+1,1);
 arr.splice(i,1);
 i--;
 flag = false;
 }
 }
 }
 return arr;
}
下载本文
显示全文
专题