c - 将输入获取到动态分配的内存中,然后返回指针

标签 c arrays pointers malloc

嘿伙计们,我一直遇到这个问题。我已经尝试了很多方法,从使用 stdin 循环并将每个字母保存到 scanf("%s") 到数组中,然后将数组声明为指针,然后返回指针。我已经接近了,但主要是在我输入的字符串中有空格时遇到问题。程序需要调用此 char* 两次来获取指向字符串的指针,然后逐行输出这两个字符串。非常感谢任何帮助。

char* get_input()
{
    char c = 'a';
    int counter = 1;
    while(c != '\0')
    {
        c = getc(stdin);
        char* d = (char)malloc(sizeof(char)*counter);
        *(d+counter) = c;
        counter++;
        return d;
    }
    return d;
}

所以我认为这样的事情是最好的方法,但我不知道如何将指针从 while 循环中返回,然后再次返回到 char* 中。我真正需要的是一些关于将一些输入直接存储到分配的数组中的最佳方法的指导,该方法不会浪费任何内存(它会知道您输入的字符串有多长并分配足够的内存)我也尝试过使用 scanf("%s"),但我不知道如何将 scanf 中的数据存储到动态内存块中。

抱歉,我的实际代码 atm 不在工作中,而我的程序在家里。任何想法请

最佳答案

您之前的代码:

char* get_input()
{
    char c = 'a';
    int counter = 1;
    while(c != '\0')
    {
        c = getc(stdin);
        char* d = (char)malloc(sizeof(char)*counter);
        *(d+counter) = c;
        counter++;
        return d;
    }
    return d;
}

噢,不,不,不!每次调用 malloc 时,您都会创建一个指向 counter 大小的内存块的单独指针地址,将其分配给 d,然后尝试将 c 分配给 *(d+counter) ,它会在当前 block 的末尾之外写入,并且绝对无法知道 d 是什么在上一次迭代期间 - 叹息...

首先,进行动态输入的方法有很多,就像彩虹中有多种颜色一样...它可以像使用 scanf%m 一样简单而明智。 code> 转换说明符(旧版本和 Windows 使用 %a)。它带有 scanf 带来的所有陷阱。但是,它可以工作:

char *get_input_scanf()
{
    char *ln = NULL;

    scanf ("%m[^\n]%*c", &ln);

    return ln;
}

但数据输入完全没有限制,控制也很少。 (注意:除非您将附加信息传递给任何输入函数或使用全局 #define 或变量,否则任何输入方法都将受到此限制。请考虑将最大长度传递给输入函数,或确保验证返回)

您还可以使用getchar()getc(fp)逐个字符读取,或者您可以使用行输入方法,例如fgetsgetline。使用任何这些方法(getline),您都需要分配一个临时行缓冲区来保存输入,然后为了将分配限制在所需的数量,根据输入的 strlen + 1(或仅使用 strdup)。与所有动态方法一样,您负责跟踪它,保留内存块的起始地址,并在不再需要内存时释放内存。除此之外,您可以使用该函数进行任何操作。

使用 getline 的动态输入例程示例如下所示:

char *get_input()
{
    char *ln = NULL;    /* line buffer, NULL - getline allocates    */
    size_t n = 0;       /* initial buff size, 0 - getline decides   */
    ssize_t nchr = 0;   /* getline return - actual no. of chars read*/

    if ((nchr = getline (&ln, &n, stdin)) != -1)
    {
        /* strip newline or carriage rtn    */
        while (nchr > 0 && (ln[nchr-1] == '\n' || ln[nchr-1] == '\r'))
            ln[--nchr] = 0;

        /* if (!nchr) {             // do not accept blank lines
            free (ln);
            return NULL;
        } */

        char *input = strdup (ln);  /* duplicate ln in input        */

        free (ln);                  /* free getline allocated mem   */

        return input;
    }

    return NULL;
}

注意:如上所述,getline 对输入字符串的长度没有限制,因此由您来验证。

动态输入的不同方式没有数量限制。 (这里可能至少有 1000 个示例)但是,这一切都归结为字符输入,因此请选择您想要的方法使用然后编写代码。这是一个具有上述两个不同功能的小工作示例。请查看该帖子,如果有疑问请告诉我。

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

char *get_input();
char *get_input_scanf();

int main (void) {

    char *line = NULL;

    printf ("\nEnter input below, [ctrl+d] to quit\n");

    for (;;)
    {
        printf ("\n input: ");
        line = get_input();

        if (line)
            printf (" value: '%s'\n", line);
        else {
            printf ("\n value: [ctrl+d] received\n");
            break;
        }
        free (line);
        line = NULL;
    }

    if (line) free (line);

    printf ("\n");

    line = NULL;

    printf ("\nEnter input below to read with get_input_scanf\n\n input: ");

    line = get_input_scanf();

    if (line) {
        printf (" value: '%s'\n", line);
        free (line);
    }

    return 0;
}

char *get_input()
{
    char *ln = NULL;    /* line buffer, NULL - getline allocates    */
    size_t n = 0;       /* initial , 0 - getline decides*/
    ssize_t nchr = 0;

    if ((nchr = getline (&ln, &n, stdin)) != -1)
    {
        /* strip newline or carriage rtn    */
        while (nchr > 0 && (ln[nchr-1] == '\n' || ln[nchr-1] == '\r'))
            ln[--nchr] = 0;

        /* if (!nchr) {             // do not accept blank lines
            free (ln);
            return NULL;
        } */

        char *input = strdup (ln);  /* duplicate ln in input        */

        free (ln);                  /* free getline allocated mem   */

        return input;
    }

    return NULL;
}

char *get_input_scanf()
{
    char *ln = NULL;

    scanf ("%m[^\n]%*c", &ln);

    return ln;
}

示例使用/输出

$ ./bin/getline_getinput

Enter input below, [ctrl+d] to quit

 input: some string of input.
 value: 'some string of input.'

 input: another string that can be any length ........ ........
 value: 'another string that can be any length ........ ........'

 input:
 value: ''

 input: a
 value: 'a'

 input:
 value: [ctrl+d] received


Enter input below to read with get_input_scanf

 input: some input to show scanf will allocate dynamically as well!
 value: 'some input to show scanf will allocate dynamically as well!'

关于c - 将输入获取到动态分配的内存中,然后返回指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30281512/

相关文章:

c - 未读取的 UDP 数据包会发生什么情况?

C 面试题

java - 如何将 int[] 数组传递到 main(String []args) 中?

c# - 如何获得具有不同数量元素的 n 个数组的所有可能组合?

c++ - 构造函数设置的 "const variables"用于表示 C++ 数组的边界?

c - 不带 printf 的指针地址 C

c - 指针 "char **strData"指向哪里?

c - 为什么声明变量后我得到不同的数字?

c - 关于指针和段错误

python - Unix 域套接字 C 服务器和 Python 客户端