「算法」方阵打印数字问题

简介: 想当初我是真的暴力到了机制,面对一个vs的图标,一行一行找规律把它打印出来,简直佛了...🤺。这里先汇总两种题:方阵蛇形填数,矩阵上三角。

「算法」方阵打印数字问题


想当初我是真的暴力到了机制,面对一个vs的图标,一行一行找规律把它打印出来,简直佛了...🤺。这里先汇总两种题:


  • 方阵蛇形填数,
  • 矩阵上三角。


1.方阵蛇形填数


题目

1.png

观察数字顺序如下图所示:

1.png

思考


首先先观察所填的数的范围为1~n*n,

所以开始写循环

注意其中的!res[x + 1][y]很巧妙,意味着你只需要填写好四个方向即可,不需要为下一次的重复填数继续考虑。


在螺旋形蛇形填数中我们可以把填满数组一圈当作为一次循环,在这一次循环中我们又分成四步进行填充。


我们假设用户输入的数字为n

那么第一步就是从arr[0][n-1]依次向下填充到arr[n-2][n-1]]

第二步就是从arr[n-1][n-1]依次向左填充到arr[n-1][1]

第三步就是从arr[n-1][0]依次向上填充到arr[1][0]

第四步就是从arr[0][0]依次向右填充到arr[0][n-2]


每四步为一次循环,但是当填满一圈填下一圈时,圈会变小,而每次循环的每一步的起始元素与结束元素的行列坐标也会变化,那么我们将循环次数定为i,当第一次循环时i的值为0,每次循环i的值加一,那么我们只需将循环中四个步骤的起始与结束元素的坐标适当加上或者减去i即可。


即:

第一步: 从arr[i] [n-1-i]依次向下填充到arr[n-2-i][n-1-i]
第二步: 从arr[n-1-i][n-1-i]依次向左填充到arr[n-1-i][1+i]
第三步: 从arr[n-1-i][i]依次向上填充到arr[1+i][i]
第四步: 从arr[i][i]依次向右填充到arr[i][n-2-i]

而需要循环的次数便是n/2,但是当n是奇数时,你会发现数组的正中央的那个位置没有得到填充,所以如果用户输入的n为奇数,我们需要在循环填充完成后再填充那个正中央的数。


这就是while循环地伟大之处吧,需要转换一下思维,不需要想着怎么自己一步步把数填进去,我也许找到了自己不会模拟的根源,在于思维没有转化过来,单纯只想着怎么自己把他一个一个输出,其实写好条件,让他自己跑就完事了...


看代码👇:

//向下走 
while(x + 1 <= n && !res[x + 1][y]) {
        res[++x][y] = ++count;
}
//向左
while(y - 1 > 0 && !res[x][y - 1]) {
        res[x][--y] = ++count;
}
//向上
while(x - 1 > 0 && !res[x - 1][y]) {
        res[--x][y] = ++count;
}
//向右
while(y + 1 <= n && !res[x][y + 1]) {
        res[x][++y] = ++count;
} 

这个可以作为模板套用,面对所有打印有规律的题,还是很好用滴❣️


完整代码


c++版本:

#include <bits/stdc++.h> 
using namespace std;
int main() {
  int n;
  cin >> n;
  vector<vector<int>>  res(n + 2, vector<int>(n + 2,0));
  int count = 1;
  res[1][n] = 1;
  int x = 1;
  int y = n;
  while(count < n *n) {
    //向下走 
    while(x + 1 <= n && !res[x + 1][y]) {
      res[++x][y] = ++count;
    }
    //向左
    while(y - 1 > 0 && !res[x][y - 1]) {
      res[x][--y] = ++count;
    }
    //向上
    while(x - 1 > 0 && !res[x - 1][y]) {
      res[--x][y] = ++count;
    }
    //向右
    while(y + 1 <= n && !res[x][y + 1]) {
      res[x][++y] = ++count;
    } 
  }
    for(int i = 1; i <= n; i++) {
      for(int j = 1; j <= n; j++) {
        printf("\t%d",res[i][j]);
            }
    cout << endl;
  }
  return 0;
}

python版本:

n, m = map(int,input().split())
arr = [[0 for j in range(m)] for i in range(n)]
dx, dy = [-1,0,1,0], [0,1,0,-1]
x, y ,d = 0,0,1
for i in range(1, n * m + 1):
    arr[x][y] = i
    a, b = x + dx[d], y + dy[d]
    if a < 0 or a >= n or b < 0 or b >= m or arr[a][b]:
        d = (d + 1) % 4
        a,b =  x + dx[d], y + dy[d]
    x, y = a, b
for i in range(n):
    for j in range(m):
        print(arr[i][j], end=' ')
    print()

for循环版本:

#include<stdio.h>
int main()
{
  int n = 0;
  int arr[100][100] = { 0 };
  scanf("%d", &n);//输入需要填充的数组的大小,arr[n][n]
  //填充数组
  int i = 0;
  int num = 1;
  for (i = 0; i < n / 2; i++)//循环的次数
  {
    int x = i;
    int y = n - 1 - i;
    for (x = i; x <= n - 2 - i; x++)//第一步,列不变,行依次++
    {
      arr[x][y] = num;
      num++;
    }
    for (y = n - 1 - i; y >= i + 1; y--)//第二步,行不变,列依次--
    {
      arr[x][y] = num;
      num++;
    }
    for (x = n - 1 - i; x >= i + 1; x--)//第三步,列不变,行依次--
    {
      arr[x][y] = num;
      num++;
    }
    for (y = i; y <= n - 2 - i; y++)//第四步,行不变,列依次++
    {
      arr[x][y] = num;
      num++;
    }
  }
  if (n % 2 != 0)//判断n是否为奇数
    arr[n / 2][n / 2] = num;//填充正中央的数
  //打印数组
  int j = 0;
  for (i = 0; i < n; i++)
  {
    for (j = 0; j < n; j++)
    {
      printf("%-3d ", arr[i][j]);
    }
    printf("\n");
  }
  return 0;
}

2.矩阵上三角


题目


样例输入为5时,打印出如下三角形:

1.png

思考


观察数的规律,很显然每一个斜的都是从左下角往右上角依次递增,不妨定义两个变量x,y,一个增加,一个减小,就可以实现。


核心代码:

for(int i=1;i<=n;i++){
    for(int x=i,y=1; x--,y++){
        if(x == 0 && y == i+1) break;
        a[x][y]=ans++;
    }
}

代码

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
#define N 110
int a[N][N];
int main() {
    int n;
    while(~scanf("%d",&n)) {  // 多次输入
        memset(a,0,sizeof(a));
        int ans=1;
        for(int i=1;i<=n;i++)
            for(int st=i,ed=1; ;st--,ed++){
                if(st==0&&ed==i+1) break;
                a[st][ed]=ans++;
            }
        for(int i=1; i<=n; i++) {  // 打印上三角
            for(int j=1; j<=n; j++) {
                if(j!=1) printf(" ");  // 除了第一列,每个数字前面都有一个空格,因为末尾不能有多余的空格
                if(a[i][j]==0) printf(" ");
                else printf("%d",a[i][j]);
            }
            printf("\n");
        }
    }
    return 0;
}

本次的打印数字暂且到这儿了。

相关文章
|
7月前
|
存储 安全 算法
基于UWB和蓝牙Beacon:室内高精度蓝牙定位系统在工厂中的工作原理与应用场景(一)
本文探讨UWB与蓝牙Beacon融合的室内高精度定位方案,结合二者优势,实现低成本、低功耗、高精度的工厂人员与资产定位,助力企业数字化转型与安全生产管理。
|
10月前
|
JSON API 开发者
深度分析阿里妈妈API接口,用Python脚本实现
阿里妈妈是阿里巴巴旗下营销平台,提供淘宝联盟、直通车等服务,支持推广位管理、商品查询等API功能。本文详解其API调用方法,重点实现商品推广信息(佣金、优惠券)获取,并提供Python实现方案。
|
机器学习/深度学习 PyTorch API
MindIE Torch快速上手
MindIE Torch 是一款高效的深度学习推理优化工具,支持 PyTorch 模型在 NPU 上的高性能部署。其核心特性包括:1) 子图与单算子混合执行,配合 torch_npu 实现高效推理;2) 支持 C++ 和 Python 编程语言,灵活适配不同开发需求;3) 兼容多种模式(TorchScript、ExportedProgram、torch.compile),覆盖广泛场景;4) 支持静态与动态 Shape 模型编译,满足多样化输入需求。通过简单易用的 API,开发者可快速完成模型加载、编译优化、推理执行及离线模型导出等全流程操作,显著提升开发效率与性能表现。
|
JavaScript Linux 网络安全
Termux安卓终端美化与开发实战:从下载到插件优化,小白也能玩转Linux
Termux是一款安卓平台上的开源终端模拟器,支持apt包管理、SSH连接及Python/Node.js/C++开发环境搭建,被誉为“手机上的Linux系统”。其特点包括零ROOT权限、跨平台开发和强大扩展性。本文详细介绍其安装准备、基础与高级环境配置、必备插件推荐、常见问题解决方法以及延伸学习资源,帮助用户充分利用Termux进行开发与学习。适用于Android 7+设备,原创内容转载请注明来源。
4861 77
|
11月前
|
Web App开发 自然语言处理 机器人
微博自动发布工具,微博自动发帖机器人,自动发布微博脚本插件
这个微博自动发布工具包含四个主要模块:主控制程序、微博操作类、内容生成器和配置文件
|
数据采集 存储 JSON
【专栏】网络爬虫与数据抓取的基础知识,包括爬虫的工作原理、关键技术和不同类型
【4月更文挑战第27天】本文介绍了网络爬虫与数据抓取的基础知识,包括爬虫的工作原理、关键技术和不同类型。通过实例展示了如何构建简单爬虫,强调实战中的环境搭建、目标分析及异常处理。同时,文章探讨了法律、伦理考量,如尊重版权、隐私保护和合法用途,并分享了应对反爬策略。最后,倡导遵守数据抓取道德规范,以负责任的态度使用这项技术,促进数据科学的健康发展。
1789 2
|
Kubernetes 网络协议 Java
程序技术好文:记一次k8spod频繁重启的优化之旅
程序技术好文:记一次k8spod频繁重启的优化之旅
911 0
|
人工智能 自然语言处理 机器人
招商银行X通义大模型,2024年度AI最佳实践案例!
招商银行X通义大模型,2024年度AI最佳实践案例!
1721 2
|
存储 SQL 数据挖掘
虚拟化数据恢复—EXSI虚拟机误还原快照的数据恢复案例
虚拟化技术原理是将硬件虚拟化供不同的虚拟机使用,一台物理机上可以有多台虚拟机。人为误操作或者物理机故障会导致上层虚拟机不可用,甚至虚拟机里的重要数据丢失。下面给大家分享一个vmware虚拟化误操作还原快照的数据恢复案例。 虚拟化数据恢复环境: 一台由物理机迁移到EXSI上面的虚拟机,迁移完成后做了一个快照。该虚拟机上运行SQL Server数据库,记录了几年的数据。 EXSI虚拟化平台上一共有数十台虚拟机,EXSI连接了一台EVA存储,所有的虚拟机(包括故障虚拟机)都放在EVA存储上。
|
Rust Cloud Native 安全
哇塞!Rust 在云原生环境中搞大事啦!构建微服务竟如此酷炫,你还不来看看?
【8月更文挑战第31天】《构建微服务:Rust 在云原生环境中的实践》探讨了 Rust 语言凭借其内存安全、高性能及可靠性等特性,在快速发展的云计算领域构建微服务的优势。书中介绍了选择合适框架(如 Axum 和 Tide)、容器化部署、服务间通信及确保服务可靠性等方面的内容,并展示了 Rust 在云原生环境中的广泛应用前景。
928 1