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

Getting input into dynamically allocated memory then returning the pointer

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

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

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

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

您之前的代码:

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

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

首先,动态输入的方法就像彩虹中的颜色一样多...它可以像使用 scanf%m 转换说明符一样简单和无知(旧版本和 windows 使用 %a)。它伴随着 scanf 带来的所有陷阱。但是,它可以工作:

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

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

    return ln;
}

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

您还可以使用 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对输入字符串的长度没有限制,所以由您来验证。

动态输入的不同方式没有数量限制。 (SO 上可能至少有 1000 个示例)但是,这一切都归结为 characterline 输入,因此请做出选择至于你想用什么方法,然后写你的代码。这是一个具有上述两个不同功能的小工作示例。查看 post 如果您有任何问题,请告诉我。

#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;
}

例子Use/Output

$ ./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!'