C++-Z字扫描实现(Zigzag Scan)

Z字扫描(Zigzag Scan)

将二维矩阵压缩成行输出:

int index=0;
for(int i=0;i<rows+cols-1;i++){//i是第几条对角线
    if(i&1){//odd,向下扫描
        for(int j=max(0,i-cols+1);j<=min(i,row-1);j++){
            res[index++]=mtx[j][i-j];
        }//
    }else{//偶数,向上扫描
        for(int j=min(i,rows-1);j>=max(0,i-cols+1);j--){
            res[index++]=mtx[j][i-j];
        }
    }
}
//假如是正方形:
int n,index;
for(int i=0;i<2*n-1;i++){
    if(i&1){//odd
        for(int j=max(i,i-n+1);j<=min(i,n-1);j--){
            res[index++]=mtx[j][i-j];
        }
    }else{//偶数
    	for(int j=min(i,n-1);j>=max(i,i-n+1);j++){
            res[index++]=mtx[j][i-j];
        }
    }
}

那么反过来,输入一串数字,变成Z型矩阵:

只需要更改中间赋值代码:

res[index++]=mtx[j][i-j]->mtx[j][i-j]=res[index++]

根据CCF例题中读入扫描数据,将扫描数据按照这样的顺序写入矩阵 M:从左上角 M 0 , 0 M_{0,0} M0,0开始,接下来填充它的右侧相邻的元素 M 0 , 1 M_{0,1} M0,1,然后依次向左下方填充直至 M 1 , 0 M_{1,0} M1,0,接下来从它下侧相邻的元素 M 2 , 0 M_{2,0} M2,0开始,依次向右上方填充直至 M 0 , 2 M_{0,2} M0,2,依次类推,循环往复,直至填充满整个矩阵或用尽所有扫描数据。

由于 Z 字形扫描的路径是从左上角到右下角,数组结尾处可能存在着连续的 0,为了节省空间,可以不存储这些连续的 0。得到的数据被称为扫描数据。

输入的数组元素可能是只有部分,针对它的实现是:

int index=1,n;//n是数组的元素个数=8
for(int i=0;i<15;i++){//2*8-1=15
	if(i&1){//odd
    	for(int j=max(0,i-7);j<=min(i,7);j++){
       		int num=0;
            if(index<=n){
                cin>>num;
                index++;
            }
            mtx[j][i-j]=num;
        }
    }else{
        for(int j=min(i,7);j>=max(0,i-7);j--){
            int num=0;
            if(index<=n){
                cin>>num;
                index++;
            }
            mtx[j][i-j]=num;
        }
    }
}

上面又有可以进行优化的地方。在初始化的时候我们把每个矩阵元素都置为0.所以可以在满足条件之后跳出。

int n;//输入数组元素的个数
int index=1;
bool flag=false;
for(int i=0;i<15;i++){
	if(i&1){
        for(int j=max(0,i-7);j<=min(i,7);j++){
            if(index==n){
                flag=true;
                break;
            }
            index++;
            cin>>mtx[j][i-j];
        }
    }else{
        for(int j=min(i,7);j>=max(0,i-7);j--){
            if(index==n){
                flag=true;
                break;
            }
            index++;
            cin>>mtx[j][i-j];
        }
    }
    if(flag)break;
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值