RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1611070
Accepted
Mik
Mik
Asked:2025-04-30 23:47:35 +0000 UTC2025-04-30 23:47:35 +0000 UTC 2025-04-30 23:47:35 +0000 UTC

C语言、地址运算、指针和malloc

  • 772

各位同事,我实在想不通问题出在哪里。

存在一个问题:

接受int n; - 从控制台输入的序列中的实数个数。将所有数字显示为小数点后两位。例如:输入 - 3 4.123 7.011 5.1// 3 - 数字的数量(<100),其余的是数字本身。

结论 -4.12 7.01 5.10

这是我得到的代码:

#include <stdio.h>
#include <stdlib.h>

main(){
    int n=0,i=0;
    double *pn;

    scanf("%d",&n);                        // Принимаю кол-во чисел в последовательности
    pn=(double *)malloc(n*sizeof(double)); // Присваиваю указатель на первый элемент блока 
                                           //   из n-ячеек памяти размером double 
    while(i<n)                             // Гоняю цикл n-раз
        scanf("%f",pn+i++);                // После присваивания инкрементирую i для 
                                           //   смещения в следующем цикле, по блоку памяти 
                                           //   выделенным malloc
    while(i>0){
        printf("%.2f\n",*(pn + n - i--));  // Вывод чисел n-раз с 2-мя десятичными знаками
    }
    free(pn);                              // Освобождаю память, хотя, она и так 
                                           // освободится после выхода из программы,
                                           // если я правильно понимаю.
}

结果如下:

![结果

如果我使用类型指示器 int *并输入int数字,将格式从 更改为"%f","%d"那么它似乎可以工作:

似乎有效

我不明白——为什么它不适用于实数?

我想了解指针和地址算术,所以我选择了这个解决方案。

c
  • 2 2 个回答
  • 32 Views

2 个回答

  • Voted
  1. Best Answer
    Stanislav Volodarskiy
    2025-05-01T00:07:05Z2025-05-01T00:07:05Z

    编译代码:

    $ gcc -std=c11 -pedantic -Wall -Wextra -Werror -Wwrite-strings -Wconversion temp.c 
    temp.c:7:1: error: return type defaults to ‘int’ [-Werror=implicit-int]
        7 | main(){
          | ^~~~
    temp.c: In function ‘main’:
    temp.c:12:26: error: conversion to ‘long unsigned int’ from ‘int’ may change the sign of the result [-Werror=sign-conversion]
       12 |     pn=(double *)malloc(n*sizeof(double)); // Присваиваю указатель на первый элемент блока
          |                          ^
    temp.c:15:17: error: format ‘%f’ expects argument of type ‘float *’, but argument 2 has type ‘double *’ [-Werror=format=]
       15 |         scanf("%f",pn+i++);                // После присваивания инрементируюю i для
          |                ~^  ~~~~~~
          |                 |    |
          |                 |    double *
          |                 float *
          |                %lf
    cc1: all warnings being treated as errors
    
    1. int之前沒有main。不影响最大速度。
    2. 该工作n*sizeof(double)混合了有符号和无符号算术。不影响最大速度。
    3. scanf读取float并写入double。这是一个真正的问题,类型混淆了。

    如果你修复了所有三个错误,你将得到:

    #include <stdio.h>
    #include <stdlib.h>
    
    int main(){
        int n=0,i=0;
        double *pn;
    
        scanf("%d",&n);                        // Принимаю кол-во чисел в последовательности
        pn=(double *)malloc((unsigned)n*sizeof(double)); // Присваиваю указатель на первый элемент блока 
                                               //   из n-ячеек памяти размером double 
        while(i<n)                             // Гоняю цикл n-раз
            scanf("%lf",pn+i++);               // После присваивания инрементируюю i для 
                                               //   смещения в следующем цикле, по блоку памяти 
                                               //   выделенным malloc
        while(i>0){
            printf("%.2f\n",*(pn + n - i--));  // Вывод чисел n-раз с 2-мя десятичными знаками
        }
        free(pn);                              // Освобождаю память, хотя, она и так 
                                               // освободится после выхода из программы,
                                               // если я правильно понимаю.
    }
    
    $ gcc -std=c11 -pedantic -Wall -Wextra -Werror -Wwrite-strings -Wconversion temp.c 
    
    $ ./a.out
    3
    4.123 7.011 5.1
    4.12
    7.01
    5.10
    

    一个讽刺的问题:没有编译器编程难吗?

    现在说说小事:

    • 没malloc必要写sizeof(double)。写成sizeof(*pn),这样就不会在两个地方重复类型。
    • 结果malloc不需要给出,void *它会double *自动给出。
    • 表达式pn+i++和&pn[i++]是等价的。但第一个通常表示以地址开头的数组pn+i++,第二个表示一个元素的地址。如果遵循这个约定,阅读代码的程序员就更容易理解它是指向单个元素的指针还是指向数组的指针。
    • 您可以随意使用索引,但最好通过规范的方式对数组进行迭代for(int i = 0; i < n; ++i)。
    • n可以不签名,演员阵容就会变少。
    • 变量的声明尽可能晚。循环索引在循环内声明。

    以下是规范代码,标准 C 程序员更容易阅读和理解:

    #include <stdio.h>
    #include <stdlib.h>
    
    int main() {
        unsigned n;
        scanf("%u", &n);                       // Принимаю кол-во чисел в последовательности
    
        double *pn = malloc(n * sizeof(*pn));  // Присваиваю указатель на первый элемент блока 
                                               //   из n-ячеек памяти размером double 
    
        for (unsigned i = 0; i < n; ++i) {     // Гоняю цикл n-раз, тут вся работа с i
            scanf("%lf", &pn[i]);
        }
    
        for (unsigned i = 0; i < n; ++i) {     // Гоняю цикл n-раз, тут вся работа с i
            printf("%.2f\n", pn[i]);           // Вывод чисел n-раз с 2-мя десятичными знаками
        }
    
        free(pn);                              // Освобождаю память, хотя, она и так 
                                               // освободится после выхода из программы,
                                               // если я правильно понимаю.
    }
    
    • 4
  2. MBo
    2025-05-01T00:06:12Z2025-05-01T00:06:12Z

    格式错误(对于 scanf 来说很重要)

     while(i<n)                             // Гоняю цикл n-раз
            scanf("%lf",pn+i++);                // После присваивания инрементируюю i для 
                                               //   смещения в следующем цикле, по блоку памяти 
                                               //   выделенным malloc
        while(i>0){
            printf("%.2lf\n",*(pn + n - i--));  // Вывод чисел n-раз с 2-мя десятичными знаками
        }
    
    • 3

相关问题

  • free 出于某种原因不会从内存中删除数组

  • 请帮助代码

  • 为什么 masm 对字符串或文本文字太长发誓,为什么在结构中设置 db 或 dw?

  • 如何将数字拆分为位并将其写入 C 中的数组?

  • 如何以给定的角度移动物体?

  • 解决“子集和问题”的时效算法

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    我看不懂措辞

    • 1 个回答
  • Marko Smith

    请求的模块“del”不提供名为“default”的导出

    • 3 个回答
  • Marko Smith

    "!+tab" 在 HTML 的 vs 代码中不起作用

    • 5 个回答
  • Marko Smith

    我正在尝试解决“猜词”的问题。Python

    • 2 个回答
  • Marko Smith

    可以使用哪些命令将当前指针移动到指定的提交而不更改工作目录中的文件?

    • 1 个回答
  • Marko Smith

    Python解析野莓

    • 1 个回答
  • Marko Smith

    问题:“警告:检查最新版本的 pip 时出错。”

    • 2 个回答
  • Marko Smith

    帮助编写一个用值填充变量的循环。解决这个问题

    • 2 个回答
  • Marko Smith

    尽管依赖数组为空,但在渲染上调用了 2 次 useEffect

    • 2 个回答
  • Marko Smith

    数据不通过 Telegram.WebApp.sendData 发送

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5