2017-2018 ACM-ICPC, Asia Daejeon Regional Contest F(递推)

F题

Problem F Philosopher’s Walk

题意:给你n,m,n代表一个长宽都为2的n次方的格子里,m代表走了从左下角开始走了m米,求最后的坐标。

 

思路:

看上图很容易便可以看出规律,每幅图都是由上一幅图在自身的四个区域进行不同的变化的到的,如第二幅图中,第一区域是第一副图向左转90度得到的,第二第三区域都是横纵坐标偏移,第四区域是向右转90度且向右移动了一个区域。

那么就可以得出转移的规律:

第一区域:向左转90度只要每个点的横纵坐标互换即可如:2(2,1) 变成 2 (1,2);

第二区域:向上移动一个区域的长度;

第三区域:向上移动一个区域,再向左移动一个区域。

第四区域,向右转90度x1,y1坐标变化为: x = k(一个区域的长度) - y1(原坐标)+ 1; y = k - x1 + 1;(先向左转90度,然后用k - 当前坐标就得到了向右移动90度的公式),再向右移动一个区域的长度就可以了

按照这个规律一直递归推就可以了。

推的头皮发麻。。

实现代码:

#include<bits/stdc++.h>
using namespace std;
struct node {
 int x,y;
};
node t;
node solve(int n,int m){
    if(n == 2){
        if(m==0){
            t.x = 1;t.y = 1;return t;
        }
        if(m == 1){
            t.x = 1;t.y = 2;return t;
        }
        if(m == 2){
            t.x = 2;t.y = 2;return t;
        }
        else{
            t.x = 2;t.y = 1;return t;
        }
    }
    int k = n/2;
    int x = m/(k*k);  //判断在上一幅图的哪一个区域
    if(x == 0){
        t = solve(k,m%(k*k));
        swap(t.x,t.y);
        return t;
    }
    if(x == 1){
        t = solve(k,m%(k*k));
        t.y += k;
        return t;
    }
    if(x == 2){
        t = solve(k,m%(k*k));
        t.x += k;t.y += k;
        return t;
    }
    if(x == 3){
        t = solve(k,m%(k*k));
        node t1 = t;
        t1.x = 2*k - t.y + 1;
        t1.y = k - t.x + 1;
        return t1;
    }
}

int main()
{
    int n,m;
    cin>>n>>m;
    m--;
    node p = solve(n,m);
    cout<<p.x<<" "<<p.y<<endl;
}

 

posted @ 2018-02-10 16:22  冥想选手  阅读(673)  评论(0编辑  收藏  举报