将输入输入到动态分配的内存中,然后返回指针
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;
}
但绝对没有数据输入的限制和很少的控制。 (注意: 除非您将附加信息传递给任何输入函数或使用全局 #define
s 或变量,否则任何输入法都会受到此限制。考虑将最大长度传递给输入函数,或确保您验证 return)
您还可以使用 getchar()
或 getc(fp)
逐字符阅读 ,或者您可以使用 行输入 方法,如 fgets
或 getline
。使用这些方法中的任何一种(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 个示例)但是,这一切都归结为 character 或 line 输入,因此请做出选择至于你想用什么方法,然后写你的代码。这是一个具有上述两个不同功能的小工作示例。查看 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!'
大家好,我一直遇到这个问题。我已经尝试了很多方法,从使用 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;
}
但绝对没有数据输入的限制和很少的控制。 (注意: 除非您将附加信息传递给任何输入函数或使用全局 #define
s 或变量,否则任何输入法都会受到此限制。考虑将最大长度传递给输入函数,或确保您验证 return)
您还可以使用 getchar()
或 getc(fp)
逐字符阅读 ,或者您可以使用 行输入 方法,如 fgets
或 getline
。使用这些方法中的任何一种(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 个示例)但是,这一切都归结为 character 或 line 输入,因此请做出选择至于你想用什么方法,然后写你的代码。这是一个具有上述两个不同功能的小工作示例。查看 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!'