UVa512 - Spreadsheet Tracking 题解(两种解法)

题目

题目链接

UVa512 - Spreadsheet Tracking

题目大意

现在有个r行c列(1<=r,c<=50)的表格,行从上到下编号为1r,列从左到右编号为1c。如图a,如果先删除第1,5行,再删除3,6,7,9列,结果如图b所示。接下来在第2,3,5行前各插入一个空行,然后在第3列前插入一个空列,会得到如图c所示结果。共有5种操作,具体来说是:
EX r1 c1 r2 c2 为交换单元格(r1,c1),(r2,c2)
A x1 x2 ... xA 为插入或删除A行或列(DC-删除列,DR-删除行,IC-插入列,IR-插入行,1<=A<=10)
在插入/删除指令后,各个x值不相同,且顺序任意。
接下来是q个查询,每个查询格式为“r c”,表示查询原始表格的单元格(r,c)。对于每个查询,输出操作执行完后该单元格的新位置
输入保证在任意时候行列数均不会超过50。

输入有多个表格。输入结束为0 0

输出时,每个表格前要输出序号(从1开始),查询时如果能查到就输出新位置,如果这个单元格没了,就输出"GONE"。每2个表格之间要空行间隔开来。

输入样例

7 9
5
DR 2 1 5
DC 4 3 6 7 9
IC 1 3
IR 2 2 4
EX 1 2 6 5
4
4 8
5 5
7 8
6 5
0 0

输出样例

Spreadsheet #1
Cell data in (4,8) moved to (4,6)
Cell data in (5,5) GONE
Cell data in (7,8) moved to (7,6)
Cell data in (6,5) moved to (1,2)

题解

最容易想到的就是模拟表格操作了,然后直接查询就好。

在代码中用了个小技巧。BIG=10000;ans数组用来表示初始的(r,c)的移动到了(ans[r][c]/BIG, ans[r][c]%BIG)。

Then show the code.

#include <stdio.h>
#include <string.h>

#define BIG 10000
#define maxn 100

//r c表示当前行列
int r, c, n;
//d数组表示当前状态 d2数组临时用
//ans数组用来表示初始的(r,c)的移动到了(ans[r][c]/BIG, ans[r][c]%BIG)
int d[maxn][maxn], d2[maxn][maxn], ans[maxn][maxn], cols[maxn];

void copy(char type, int p, int q);
void del(char type);
void ins(char type);

int main(){
    int kase = 0;
    while(scanf("%d%d%d", &r, &c, &n) == 3 && r){
        //给d数组的值刻上印记 以后无论这个值到了哪儿 他都明白他初始位置为 i, j  (滑稽.jpg
        memset(d, 0, sizeof(d));
         for(int i=1; i<=r; i++)
            for(int j=1; j<=c; j++)
                d[i][j] = i*BIG+j;
        
        //模拟操作
        char cmd[10];
        while(n--){
            memset(cmd, 0, sizeof(cmd));
            scanf("%s", cmd);
            //交换操作
            if(cmd[0] == 'E'){
                int r1, c1, r2, c2, temp;
                scanf("%d%d%d%d", &r1, &c1, &r2, &c2);
                temp = d[r1][c1]; d[r1][c1] = d[r2][c2]; d[r2][c2] = temp;
            } //交换或删除操作
            else{
                int a, x;
                //cols数组表示要操作的行或列
                memset(cols, 0, sizeof(cols));
                scanf("%d", &a);
                while(a--){
                    scanf("%d", &x);
                    cols[x] = 1;
                }
                if(cmd[0] == 'D') del(cmd[1]);
                else ins(cmd[1]);
            }
        }


        memset(ans, 0, sizeof(ans));
        for(int i=1; i<=r; i++)
            for(int j=1; j<=c; j++)
                ans[d[i][j]/BIG][d[i][j]%BIG] = i*BIG+j;

        int q, r1, c1;
        scanf("%d", &q);
        if(kase++) printf("\n");
        printf("Spreadsheet #%d\n", kase);
        while(q--){
            scanf("%d%d", &r1, &c1);
            printf("Cell data in (%d,%d) ", r1, c1);
            if(ans[r1][c1])
                printf("moved to (%d,%d)\n", ans[r1][c1]/BIG, ans[r1][c1]%BIG);
            else
                printf("GONE\n");
        }
    }

    return 0;
}

//将d2的q行/列复制到d的p行/列
void copy(char type, int p, int q){
    if(type == 'R')
        for(int i=1; i<=c; i++)
            d[p][i] = d2[q][i];
    else
        for(int i=1; i<=r; i++)
            d[i][p] = d2[i][q];
}

void del(char type){
    memcpy(d2, d, sizeof(d));
    int cnt1 = type == 'R' ? r : c, cnt2 = 0;
    for(int i=1; i<=cnt1; i++)
        //如果需要操作这行/列 则不复制这行/列
        if(!cols[i]) copy(type, ++cnt2, i);
    type == 'R' ? r = cnt2 : c = cnt2;
}


void ins(char type){
    memcpy(d2, d, sizeof(d));
    int cnt1 = type == 'R' ? r : c, cnt2 = 0;
    for(int i=1; i<=cnt1; i++){
        //如果需要操作这行/列 则在前面加一个空行/列(用0表示空行)
        if(cols[i]) copy(type, ++cnt2, 0);
        copy(type, ++cnt2, i);
    }
    type == 'R' ? r = cnt2 : c = cnt2;
}

其实还可以对于只关注于要查询的那个单元格进行模拟,这样更快。

//代码太简单 懒得注释
#include <stdio.h>

#define maxn 1000

//存放操作
struct {
    char c[5];
    int r1, c1, r2, c2;
    int a, x[20];
} cmds[maxn];
int r, c, n;


int simulate(int &r0, int &c0){
    for(int i=0; i<n; i++){
        if(cmds[i].c[0] == 'E'){
            if(cmds[i].r1 == r0 && cmds[i].c1 == c0){
                r0 = cmds[i].r2; c0 = cmds[i].c2;
            }
            else if(cmds[i].r2 == r0 && cmds[i].c2 == c0){
                r0 = cmds[i].r1; c0 = cmds[i].c1;
            }        
        } else{
            int r1, c1; r1 = c1 = 0;
            for(int j=0; j<cmds[i].a; j++){
                if(cmds[i].c[0] == 'I'){
                    if(cmds[i].c[1] == 'R'){
                        if(cmds[i].x[j] <= r0) r1++;
                    }else{
                        if(cmds[i].x[j] <= c0) c1++;
                    }
                }else if(cmds[i].c[0] == 'D'){
                    if(cmds[i].c[1] == 'R'){
                        if(cmds[i].x[j] == r0) return 0;
                        if(cmds[i].x[j] < r0) r1--;
                    } else{
                        if(cmds[i].x[j] == c0) return 0;
                        if(cmds[i].x[j] < c0) c1--;
                    }   
                }
            }
            r0 += r1; c0 += c1;
        }
    }
    return 1;
}

int main(){
    int r0, c0, q, kase=0;
    while(scanf("%d%d%d", &r, &c, &n) == 3 && r){
        if(kase++) printf("\n");
        printf("Spreadsheet #%d\n", kase);
        for(int i=0; i<n; i++){
            scanf("%s", cmds[i].c);
            if(cmds[i].c[0] == 'E'){
                scanf("%d%d%d%d", &cmds[i].r1, &cmds[i].c1, &cmds[i].r2, &cmds[i].c2);
            }else{
                scanf("%d", &cmds[i].a);
                for(int j=0; j<cmds[i].a; j++){
                    scanf("%d", &cmds[i].x[j]);
                }
            }
        }

        scanf("%d", &q);
        while(q--){
            scanf("%d%d", &r0, &c0);
            printf("Cell data in (%d,%d) ", r0, c0);
            if(!simulate(r0, c0)) printf("GONE\n");
            else printf("moved to (%d,%d)\n", r0, c0);
        }
    }
    return 0;
}
posted @ 2020-11-14 20:17  1v7w  阅读(326)  评论(0编辑  收藏  举报