收藏 分销(赏)

士兵站队问题题解.doc

上传人:pc****0 文档编号:8349136 上传时间:2025-02-10 格式:DOC 页数:3 大小:24.50KB 下载积分:10 金币
下载 相关 举报
士兵站队问题题解.doc_第1页
第1页 / 共3页
士兵站队问题题解.doc_第2页
第2页 / 共3页


点击查看更多>>
资源描述
【问题描述】 在一个划分成网格的操场上,n个士兵散乱地站在网格点上。网格点用整数坐标(x,y)表示。士兵们可以沿网格边往上、下、左、右移动一步,但在同一时刻任一网格点上只能有一名士兵。按照军官的命令,士兵们要整齐地列成一个水平队列,即排列成(x,y),(x+1,y),…,(x+n-1,y)。如何选择x和y的值才能使士兵们以最少的总移动步数排成一行。编程计算使所有士兵排成一行需要的最少移动步数。 【输入格式】 第1行是士兵数n,1≤n≤10000。接下来n行是士兵的初始位置,每行有2个整数x和y,-10000≤x,y≤10000。 【输出格式】 一个数据,即士兵排成一行需要的最少移动步数。 【输入样例】sol.in 5 1 2 2 2 1 3 3 -2 3 3 【输出样例】sol.out 8   分析: 一 士兵有多种移动方式 通过适当的移动顺序和移动路线可以使得同一时刻不会有两名士兵站在同一点 二 题目要求最佳移动方式(即求移动的最少步数) 题目要求转化为求士兵站立的“最终位置”,即如何取“最终位置”使得士兵移动的步数最少(最优) Y轴方向上的考虑 设目标坐标为M,即n个士兵最终需要移动到的Y轴的坐标值为M n个士兵的Y轴坐标分别为: Y0,Y1,Y2 …… …… Yn-1 则最优步数S=|Y0-M|+|Y1-M|+|Y2-M|+ …… …… +|Yn-1-M| 结论:M取中间点的值使得S为最少(最优) 证明:…… 从最上和最下的两个士兵开始递推…… 最优位置 最优位置 归结到最后,处于中间位置的士兵的Y轴坐标值就是“最终位置”的Y轴坐标 可能有两种情况 士兵总数为双数情况:取中间两点间的任意一个位置 士兵总数为单数情况:取中间点的所在位置 解决办法:对所有的Y轴坐标进行排序(O(nlogn))或者进行线性时间选择(O(n)) 然后取“中间”点的Y轴坐标值作为最佳位置M的值 最后通过公式求出Y轴方向上移动的最优步数 X轴方向上的考虑 首先需要对所有士兵的X轴坐标值进行排序 然后,按从左至右的顺序依次移动到每个士兵所对应的“最终位置”(最优),所移动的步数总和就是X轴方向上需要移动的步数 例,最左的士兵移动到“最终位置”的最左那位,第二个士兵移动到“最终位置”的第二位 则总的步数为:士兵一移动步数+士兵二移动步数+ …… +士兵n移动步数 如何确定X轴方向上的最佳的“最终位置”? 共n个士兵 他们相应的X轴坐标为:X0,X1,X2 …… …… Xn-1 设,士兵需要移动到的“最终位置”的X轴坐标值为:k,k+1,k+2 …… …… k+(n-1) 则所求最优步数S=|X0-k|+|X1- (k+1) |+|X2-(k+2)|+ …… +|Xn-1-(k+(n-1))| 经过变形S=|X0-k|+|(X1-1)-k|+|(X2-2)-k|+ …… …… +|(Xn-1-(n-1))-k| 注意到公式的形式与Y轴方向上的考虑一样,同样是n个已知数分别减去一个待定数后取绝对值,然后求和 因此还是采用取中位数的办法求得k值,最后算出最优解。   参考程序: type   arr=array[-10001..10001]of longint; var   i,j,k,l,n,m,num:longint;   f:array[-10001..10001]of boolean;   a,b,c:arr; procedure  ok(l,r:longint);  var  i,j,x,y:longint; begin   i:=l;j:=r;x:=a[(l+r) div 2];   repeat     while a[i]<x do i:=i+1;    while x<a[j] do j:=j-1;     if i<=j then   begin       y:=a[i];a[i]:=a[j];a[j]:=y;       i:=i+1;j:=j-1;     end;   until i>j;   if l<j then ok(l,j);   if i<r then ok(i,r); end; procedure  ok1(l,r:longint);  var  i,j,x,y:longint; begin   i:=l;j:=r;x:=b[(l+r) div 2];   repeat     while b[i]<x do i:=i+1;    while x<b[j] do j:=j-1;     if i<=j then   begin       y:=b[i];b[i]:=b[j];b[j]:=y;       i:=i+1;j:=j-1;     end;   until i>j;   if l<j then ok1(l,j);   if i<r then ok1(i,r); end; begin  assign(input,'sol.in');  assign(output,'sol.out');  reset(input);  rewrite(output);  readln(n);  for i:=1 to n do  readln(a[i],b[i]);  ok(1,n);  ok1(1,n);  for i:=1 to n do  a[i]:=a[i]-i+1;  ok(1,n);  l:=0;  k:=a[(1+n) div 2];  for i:=1 to n do  l:=l+abs(a[i]-k);   k:=b[(1+n) div 2];   for i:=1 to n do   m:=m+abs(b[i]-k);   writeln(m+l);   close(input);close(output); end.  
展开阅读全文

开通  VIP会员、SVIP会员  优惠大
下载10份以上建议开通VIP会员
下载20份以上建议开通SVIP会员


开通VIP      成为共赢上传

当前位置:首页 > 百科休闲 > 其他

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        抽奖活动

©2010-2026 宁波自信网络信息技术有限公司  版权所有

客服电话:0574-28810668  投诉电话:18658249818

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :微信公众号    抖音    微博    LOFTER 

客服