题意是在一个有 400 个房间的走廊中搬动房间里的椅子,如果两次的路线重叠,就要分两次搬动,如果不重叠,就可以一次搬动。

开始的时候直接当成求线段重叠条数的题,发现这种思路完全是错的,比如 1 - 3,2 - 4,3 - 5 这一组,只需搬动两次即可,但找重叠线段的话就会找到 3 条重叠线段。

然后打算直接模拟做,加入一点贪心的思路,用结构体数组存搬动要求,按椅子搬动的距离从小到大排序,扫描一边整个结构体数组,将搬动的路径标记,

若已经标记过,则 ++ans,注意 ans 的初值为 1,因为第一次搬动要算作一次搬动,但这一次没有与任何路径重叠。

模拟做的直接超时了......

代码如下:

 int t,n,ans,mp[];
bool f;
struct node
{
int from,to,len;
}p[];
bool cmp(node a,node b)
{
return a.len!=b.len?a.len<b.len:a.from<b.from;
}
int main()
{
scanf("%d",&t);
while(t--)
{
scanf("%d",&n);
ans = ;
memset(mp,,sizeof(mp));
for(int i = ; i < n; ++i)
{
scanf("%d%d",&p[i].from,&p[i].to);
p[i].len = p[i].to-p[i].from;
}
sort(p,p+n,cmp);
for(int i = ; i < n; ++i)
{
f = true;
for(int j = p[i].from; j <= p[i].to; ++j)
{
if(mp[j])
{
ans++;
fill(mp+p[i].from,mp+p[i].to,-);
f = false;
break;
}
}
if(f) fill(mp+p[i].from,mp+p[i].to,-);
}
printf("%d\n",ans*);
}
return ;
}

看了其他人的做法,才知道不应该把房间看作排列在一条线段上的点,而是如题目中描述的那样排在一对平行线上:

HDU 1050(搬椅子 数学)-LMLPHP

这样将房间前的位置用一个长 200 的一维数组来存,在每次的路径的每一个位置上增加 1,这个数组中的最大值即为搬动次数。如图示的情况即为:1,2,2,0,...

HDU 1050(搬椅子 数学)-LMLPHP

搬动次数即为 2 次。

代码如下:

 #include <bits/stdc++.h>
using namespace std;
int main()
{
int t,n,from,to,ans;
int cnt[],tmp;
scanf("%d",&t);
while(t--)
{
scanf("%d",&n);
memset(cnt,,sizeof(cnt));
ans = ;
for(int i = ; i <= n; ++i)
{
scanf("%d%d",&from,&to);
if(from > to)
{
tmp = from;
from = to;
to = tmp;
}
for(int j = (from+)/; j <= (to+)/; ++j)
cnt[j]++;
}
for(int i = ; i <= ; ++i)
if(ans<cnt[i]) ans = cnt[i];
printf("%d\n",ans*);
}
return ;
}

另外还知道了一点,在交换两变量的值时,用位运算的方法更快:

如 temp = a; a = b; b = temp;

用位运算实现:a = a^b; b = a^b; a = a^b;

本题中由于没注意到有可能搬动是从房间号较大的房间搬向房间号较小的房间,还 wa 了......

解决办法就是再加一步如果是这种情况就交换两房间的号码,即还是从小到大的顺序,这里直接交换的结果是 15ms,用位运算交换的结果是 0ms.

以后就要改用这种位运算交换的方法了。

特向这篇博客的作者致谢:https://www.cnblogs.com/cchun/archive/2011/05/14/2520076.html

05-07 12:39