• 人工智能之地形导航系统


    项目需求

    2018 年 11 月 26 日,“洞察”号火星探测器在火星上成功着陆,执行人类首次探究火星“内 心深处”的任务。火星上地形比较复杂,高低起伏,有山峰和低谷,稍有不慎,就会引起“翻 车”。因此我们必须要识别地形上的各个最高点(峰点)和最低点(谷点),以便为探测器 提供导航数据。

    042badabcc9b496a910487a31f897e79.png

    峰点就是一个其周围所有点的海拔都低于它的点

     

    203

    109

    181

    289

    300

    264

    190

    250

    188

    现在我们要做的就是分析来自地图上的海拔数据,以确定地形中峰点的数目和位置

     

    二维数组

    数组,就是多个同类型的元素的有序“组合”。如下的一组女兵,注:同类型是指都是女兵,不能混入男兵 -:)

    90853cd767d34bd29b5367b8a1cbca51.png

     

    二维数组,就是指含有多个数组的数组!

    0b5aa524d52c4710886ceeb2c6781460.png

    如果把一维数组理解为一行数据,那么,二维数组可形象地表示为行列结构。

    二维数组的定义

    和数组一样,需要先定义,再使用。

    int a[25] ; //一行女兵实例:

    int      a[5][25];     //五行女兵

    //定义了一个二维数组,

    //数组名是“a”,

    //包含 5 行 25 列,共 125 元素,

    //每个元素是 int 类型的变量

     

    二维数组的初始化

    int a[3][4];  //二维数组元素的值可能是随机的(全局变量会初始化为 0,局部变量值随机)

    方式一 初始化时指定每行的值

    int a[3][4]={    //等效于 int a[][4]

    {1,     2,     3,     4},

    {5,     6,     7,     8},

    {9,    10,    11,   12}

    };

    注:最外围括号内部的每个括号相当于初始化一行,括号中可以省略某些元素的初始化

    方式二初始化时从头开始,依次序进行

    int a[3][4]={ 1,2,3,4,5,6,7,8,9,10,11,12};

     

    int a[3][4]={ 0}; //所有元素都初始化为 0

    注:后面的多个元素可以不指定,不指定全部初始化为 0

     

    #include <stdlib.h> #include

    int main(void){

    //int ages[5][25];//定义一个二维数组 int i=0, j=0;

    //初始化

    //第一种方式 初始化时指定每行的值

    int a[3][4]={

    {1},//省略掉得列会默认置零

    {5,6,7},

    {9,10,11,12}

    };

    //第二种方式 初始化时从头开始,依次序进行

    int a1[3][4]={1,2,3,4,5,6,7,8,9,10,11,12};

    int a2[3][4]={1};//只初始化第一个,其他得默认置零

    for(i=0; i<3; i++){ for(j=0; j<4; j++){ printf("%d ", a[i][j]);

    }

    printf("\n");

    }

    system("pause");

    return 0;

    }

    二维数组的访问

    a[3][4]={1,2,3,4,5,6,7,8,9,10,11,12};

    a[0][0]

    1

    a[0][1]

    2

    a[0][2]

    3

    a[0][3]

    4

    a[1][0]

    5

    a[1][1]

    6

    a[1][2]

    7

    a[1][3]

    8

    a[2][0]

    9

    a[2][1]

    10

    a[2][2]

    11

    a[2][3]

    12

    如下图表示,左侧表示的是一个大小为M+1的一维数组,右侧表示的是一个大小为(M+1)*(N+1)的二维数组。

    8d01823dd8094831becccc16239fc2c2.png

     

    int i=0, j=0; int a[3][4];

    //给数组成员赋值

    for(int i=0;i<12; i++){ a[i/4][i%4] = i+1;

    }

    //或

    for(int i=0;i<3; i++){ for(int j=0;j<4; j++){ a[i][j] = 4*i+j+1;

    }

    }

     

    //输出

    for(int i=0;i<3; i++){ for(int j=0;j<4; j++){ printf("%d ",a[i][j]);

    } printf("\n");

    }

    11

    二维数组的存储方式

    设置断点调试,可以看到二位数组中的所有元素在内存中的存储方式

    aaedec8366bd4899a0058da03c9d72f9.png

     307acf9ff9924b3f94d83e7da78af553.png

     

     多维数组

    上面讨论的二维数组的相关内容都适用于三维数组或更多维的数组。声明一个三维数组: int girl[3][8][5];

    可以把一维数组想象成一排女兵,把二维数组想象成一个女兵方阵,把三维数组想象成多个女兵方阵。这样,当你要找其中的一个女兵时,你只要知道她在哪个方阵(从 0、1、2 中选择),在哪一行(从 0-7)中选择,在哪一列(从 0-4 中选择)

    03edcc0528634a47b4914eb771b2c816.png

     

    二维数组作为函数的参数

    切记!数组作为函数的参数传递,不是单纯的值传递,传递的是数组本身。

    二维数组作为函数的参数:

    #include

    #include

    //版本1 指明参数

    void print1(int a[3][4]){ for(int i=0;i<3; i++){ for(int j=0;j<4; j++){ printf("%d ",a[i][j]);

    }

    printf("\n");

    }

    }

    //版本1 省略一个高维参数

    void print2(int a[][4], int lines){ for(int i=0;i

    }

    printf("\n");

    }

    }

    int main(void){

    //int arr[3][4]={{},{3,4}}; int a[3][4]={0}; int i=0; int j=0;

    //给数组成员赋值

    for(int i=0;i<3; i++){ for(int j=0;j<4; j++){ a[i][j] = 4*i+j;

    }

    }

    print1(a);

    print2(a);

    }

     

    常见错误总结

    数组定义时编译器不能确定数组的大小

    int a[3][] ; int a[][4] ;

    int a[3][]={{1,2},{3,4},{5,6}}

    一条原则:仅定义时(无初始化)不能省略,有初始化可以省略高维

     

    严禁数组越界

    在使用数组时,要防止数组下标超出边界。也就是说,必须确保下标是有效的值。

    int a[10] ;         a[10]=100;    //错误,访问越界,a[10] 的成员是 a[0] - a[9]

    int a[3][4];      a[3][0]=666; //错误, a[3][4] 包含了 3 行 4 列的数组,行下标也是从 0 开始,有效范围 0 - 2

    数组(无论几维)传参并不是整个数组的复制

    #include

    #include //int arr[3][4];

    void fun(int a[3][4]){

    for(int i=0;i<3;i++){

    for(int j=0;j<4;j++){ a[i][j]=0;

    }

    }

    }

    int main(void){

    //int arr[3][4]={{},{3,4}};

    int a[3][4]={0};

    int i=0; int j=0;

    //给数组成员赋值

    for(int i=0;i<3; i++){

    for(int j=0;j<4; j++){

    a[i][j] = 4*i+j;

    }

    }

     

    fun(a);

    //输出

    for(int i=0;i<3; i++){

    for(int j=0;j<4; j++){

    printf("%d ",a[i][j]);

    }

    printf("\n");

    }


    项目实现

    假设下面的数据代表一个 6 x 7 的网格, 加了下划线的网格即为峰点。

    5039    5127   5238    5259   5248    5310   5299

    5150    5392   5410    5401   5320    5820   5321

    5290    5560   5490    5421   5530    5831   5210

    5110    5429    5430    5411   5459    5630   5319

    4920    5129   4921    5821   4722    4921   5129

    5023    5129   4822    4872   4794    4862   4245

    为了描述峰点的位置,我们需要使用一个位置方案:使用二维数组描述

    假定左上角是[0][0],那么向下移动,则行号加 1;向右移动,则列号加 1, 那么这些峰点的位置就可以描述为:[2][1]   [2][5]    [4][3]。

    位置确定后,与周围 4 个邻节点比较即可确定峰点!(注:网格边界点缺乏 4 个相邻点不计算峰点)地形数据保存于文件中。

     

    49997b3f13ba49edb4ac1b2ea55c3709.png

     

    算法设计

    1. 将地形数据从文件读入二维数组;
    2. 逐行遍历二维数组的每个元素,确定是否峰值并打印结果。
    1. #include
    2. #include
    3. #include
    4. #include
    5. using namespace std;
    6. #define N 64
    7. bool isPeak(int map[N][N], int nrows, int ncols)
    8. {
    9. if ((map[nrows][ncols] > map[nrows + 1][ncols]) &&
    10. (map[nrows][ncols] > map[nrows - 1][ncols]) &&
    11. (map[nrows][ncols] > map[nrows][ncols + 1]) &&
    12. (map[nrows][ncols] > map[nrows][ncols] - 1))
    13. {
    14. return true;
    15. }
    16. else
    17. {
    18. return false;
    19. }
    20. }
    21. int main(void)
    22. {
    23. string filename;
    24. fstream file;
    25. int map[N][N];
    26. int ncols, nrows;
    27. cout << "请输入你要打开的文件:";
    28. cin >> filename;
    29. file.open(filename.c_str());
    30. if (file.fail())
    31. {
    32. cerr << "打开文件失败" << endl;
    33. exit(1);
    34. }
    35. file >> nrows >> ncols;
    36. if (nrows > N || ncols > N)
    37. {
    38. cerr << "行或列超出指定范围" << endl;
    39. exit(1);
    40. }
    41. for (int i = 0; i < nrows; i++)
    42. {
    43. for (int j = 0; j < ncols; j++)
    44. {
    45. file >> map[i][j];
    46. }
    47. }
    48. for (int i = 1; i < nrows-1; i++)
    49. {
    50. for (int j = 1; j < ncols-1; j++)
    51. {
    52. if (isPeak(map, i, j))
    53. {
    54. cout << "行:" << i << "列" << j << "峰值" << map[i][j] << endl;
    55. }
    56. }
    57. }
    58. file.close();
    59. system("pause");
    60. return 0;
    61. }

     

     

     

     

  • 相关阅读:
    从Linux的tty_struct指针获取驱动上下文
    SQL Server外键约束
    C++数据结构——红黑树
    技术冲浪: 拥抱 WordPress 官网时代
    c4d和blender哪个简单?哪个好用?
    TRex学习之旅六
    linux netlink实现机制:通信 - 驱动与应用层数据交互(三)
    强大的JTAG边界扫描(5):FPGA边界扫描应用
    Go编译原理系列6(类型检查)
    Spring Boot中添加Thymeleaf模板
  • 原文地址:https://blog.csdn.net/m0_65635427/article/details/127586804