html:
toc: true

写在前面:

  • 出于作者不了解C99以前标准中对二维数组的动态声明而导致的一场考场事故,作者写下这篇文章,,以便其他同学在遇到类似问题时不要犯同样的错误,同时作为对自己的警醒.
  • 本文主要是关于对于二维数组动态声明问题在不同C标准下方法的探讨,将给出一个简单的实例,以便读者能够快速理解.
  • 作者水平有限,若有错误,还请指正.

一个简单的实例

  • 题目描述:

    对输入的一个矩阵的偶数列分别求和并输出;

    输入说明:输入两个整数m,n代表矩阵的行数和列数,接下来输入m行,每行n个整数,代表矩阵的元素,数与数之间以空格隔开;

    输出说明:先输出偶数列的个数,然后输出每个偶数列所有元素的和,每个和后面有一个空格;

    输入示例:

    3 4

    3 5 6 7

    9 5 8 2

    6 5 8 7

    输出示例:

    2

    15 16
  • 要解决这个问题,我们可以声明一个二维数组,然后控制输入输出流程完成问题即可.但是注意到矩阵的行列数是未知的,因此我们需要使用动态数组来完成这个任务.

C99标准以及C99标准之后

在C99标准之后,要完成这个问题,只需要引入变长数组(variable length array)方法即可简单完成.

#include <stdio.h>
int main(void){
int m, n;
scanf("%d %d", &m, &n);
int matrix[m][n];
// 输入矩阵元素
for (int i = 0; i < m; i++)
for (int j = 0; j < n; j++)
scanf("%d", &matrix[i][j]);
// 计算偶数列的和
int sum[n / 2]; // 用于存储每个偶数列的和
int count = 0; // 偶数列的个数
for (int j = 1; j < n; j += 2){
int column_sum = 0;
for (int i = 0; i < m; i++){
column_sum += matrix[i][j];
}
sum[count++] = column_sum;
}
printf("%d\n", count);
for (int i = 0; i < count; i++){
printf("%d ", sum[i]);
}
return 0;
}

可以看到,对于二维数组的动态声明,使用变长数组方法可以直接使用类似于int matrix[m][n]这样的语句完成.

但是要注意的是,变长数组方法只能在C99标准之后使用,因为该方法是在C99标准中引入的,在C99之前,使用变长数组方法会报错.要动态分配二维数组,要借助指针传参.

现行部分学校旧OJ系统使用C89(C90)标准,使用变长数组将无法完成编译(血泪教训)

C99标准以前

在C99标准之前,要完成这个问题,我们需要使用指针来完成.

通常的步骤为:

  1. 声明一个指向指针的指针,用于保存动态分配的二维数组的首地址
  2. 使用malloc函数动态分配一个包含n行的指针数组,每个指针指向一个包含m个元素的一维数组
  3. 对分配的二维数组进行赋值或操作
  4. 在不需要使用该数组时,使用free函数释放分配的内存

    具体示例如下:
#include <stdio.h>
#include <stdlib.h>//使用malloc函数需要
int main(void){
int m, n;
scanf("%d %d", &m, &n);
//分配指向指针的指针数组
int **matrix = (int **)malloc(m * sizeof(int *));
//对每一行再分配空间并输入
for (int i = 0; i < m; i++){
matrix[i] = (int *)malloc(n * sizeof(int));
for (int j = 0; j < n; j++){
scanf("%d", &matrix[i][j]);
}
}
// 计算偶数列的和
int sum[n / 2]; // 用于存储每个偶数列的和
int count = 0; // 偶数列的个数
for (int j = 1; j < n; j += 2){
int column_sum = 0;
for (int i = 0; i < m; i++){
column_sum += matrix[i][j];
}
sum[count++] = column_sum;
}
// 输出结果
printf("%d\n", count);
for (int i = 0; i < count; i++){
printf("%d ", sum[i]);
}
// 释放内存
for (int i = 0; i < m; i++){
free(matrix[i]);
}
free(matrix);
matrix = NULL;
return 0;
}

C99标准前后对于二维数组的动态声明问题的更多相关文章

  1. C++二维数组的动态声明

    int **a  =  new int* [m]   //分配一个指针数组,将其首地址保存在a中   . for(int i = 0; i < m; i++)   //为指针数组的每个元素分配一 ...

  2. JAVASE(八) 数组: 一维数组、二维数组、动态数组、静态数组

    个人博客网:https://wushaopei.github.io/    (你想要这里多有) 1.一维数组 1.1 数组的声明和初始化声明方式: String str[]; //不建议使用 Stri ...

  3. C语言数组:C语言数组定义、二维数组、动态数组、字符串数组

    1.C语言数组的概念 在<更加优美的C语言输出>一节中我们举了一个例子,是输出一个 4×4 的整数矩阵,代码如下: #include <stdio.h> #include &l ...

  4. C语言学习笔记 (005) - 二维数组作为函数参数传递剖析

    前言 很多文章不外乎告诉你下面这几种标准的形式,你如果按照它们来用,准没错: //对于一个2行13列int元素的二维数组 //函数f的形参形式 f(int daytab[2][13]) {...} / ...

  5. c++ 二维数组传递

    c++ 二维数组传递 我们在传递二维数组时,对于新手来说,可能会存在某些问题,下面讲解几种传递方法 在讲解如何传递二维数组时,先看看如何动态new 二维数组 // 二维数组动态申请 int row , ...

  6. 【C语言入门教程】4.2 二维数组

    C 语言允许使用多维数组,即使用多组小标的数组,二维数组是最常用的多维数组.多维数组在内存中存放数据的顺序与一维数组相同,使用连续的存储单元. 4.2.1 二维数组的一般形式 二维数组的一般声明形式为 ...

  7. 《Java大学教程》—第16章 二维数组

    多维(Multi-dimensional)数组维数由索引个数决定.常用的数组:一维(one-dimensional)数组.二维(two-dimensional)数组 16.2    创建二维数组索引从 ...

  8. js一维数组转换为二维数组

    function arrTrans(num, arr) { // 一维数组转换为二维数组 const iconsArr = []; // 声明数组 arr.forEach((item, index) ...

  9. Java连载68-数组的拷贝、二维数组

    一.数组的拷贝 函数arraycopy(),参数为:源数组.源数组的开始下标.目标数组.目标数组的开始下标.拷贝长度 package com.bjpowernode.java_learning; ​ ...

  10. C# 二维数组 [,]与[][] 的区别 及特性

    arr[,] 用于声明等长的二维数组 Eg: //声明数组有3行 每行长度相等为2 var s = new int[3, 2] { { 1, 2 }, { 3, 4 }, { 1, 4 } }; 获取 ...

随机推荐

  1. AcWing 456. 车站分级

    原题链接AcWing 456. 车站分级 抽象出题意,停靠过的车站的等级一定严格大于为停靠过的车站的等级,且不存在环,例如车站\(A\)等级大于车站\(B\),则\(A >= B + 1\),不 ...

  2. Jdk_HashMap 源码 —— hash(Object)

    Jdk 源码 HashMap 的源码是在面试中考的算是比较多的,其中有很多高性能的经典写法,也值得多学习学习. 本文是本人在阅读和学习源码的过程中的笔记(不是教程),如有错误欢迎指正. Jdk Ver ...

  3. 关于Delphi

    # 关于Delphi ··Delphi中使用的面向对象pascal编程语言. ··Pascal语言最初由瑞士苏黎士理工学院的尼古拉斯-沃斯(Niklaus Wirth)教授在1971年设计. ··19 ...

  4. Modbus转PROFINET网关 TS-180

    TS-180实现 PROFINET 网络与串口网络之间的数据通信.三串口可分别连接具有 RS232 或 RS485 接口的设备到PROFINET 网络,三串口相同,全为 RS232 或RS485.即将 ...

  5. 环形缓冲区 Ring Buffer 的实现

    环形缓冲区(Circular Buffer 或 Ring Buffer)是一种数据结构,它在逻辑上形成一个闭环.这种结构非常适用于需要固定大小的缓冲区的情况,如音频处理.网络通信.实时数据传输等.环形 ...

  6. 掌握这些,轻松管理BusyBox:如何交叉编译和集成BusyBox

    在嵌入式系统中,由于设备的资源限制,需要开发人员寻找一种轻量.小型且使用广泛的工具集.而 BusyBox 就是这样一个在嵌入式系统中非常实用的工具集.本文将介绍如何在 Ubuntu 22.04 平台上 ...

  7. 每天5分钟复习OpenStack(十二)Ceph FileStore 和 BlueSotre

    一个最小化的Ceph集群需要三个组件MON MGR OSD.上一章我们部署了MON,本章节我们完成剩下MGR 和OSD 的部署.在文末我们将重点介绍下什么是FileStore和BlueStore,并详 ...

  8. MongoDB的聚合笔记

    1,聚合 聚合(aggregate)主要用于计算数据,类似sql中的sum().avg(). 常用的表达式如上图. 1.1,aggregate  语法 语法:     db.集合名称.aggregat ...

  9. MySQL索引命名规范

    [强制]主键索引名为 pk_字段名:唯一索引名为 uk_字段名:普通索引名则为 idx_字段名 说明:pk_ 即 primary key:uk_ 即 unique key:idx_ 即 index 的 ...

  10. 深入 K8s 网络原理(二)- Service iptables 模式分析

    目录 1. 概述 2. 准备 Service 和 Pods 资源 3. K8s 里 Service 的实现原理 3.1 kube-proxy 组件 3.2 iptables 简介 3.3 iptabl ...