Как добавить строки, используя sprintf? - PullRequest
56 голосов
/ 20 апреля 2010

У меня серьезная проблема с sprintf.

Предположим, мой фрагмент кода:

sprintf(Buffer,"Hello World");
sprintf(Buffer,"Good Morning");
sprintf(Buffer,"Good Afternoon");
.
.
.

Несколько сотен спринтов ....

Если мне так нравится, это перезаписывается.

Как мне избежать перезаписи с помощью sprintf. Если я дам printf в конце, я хочу увидеть все строки.

Ответы [ 15 ]

95 голосов
/ 20 апреля 2010

Вам нужно:

sprintf(Buffer,"Hello World");
sprintf(Buffer + strlen(Buffer),"Good Morning");
sprintf(Buffer + strlen(Buffer),"Good Afternoon");

и, конечно, вам нужен достаточно большой буфер

64 голосов
/ 20 апреля 2010
int length = 0;
length += sprintf(Buffer+length, "Hello World");
length += sprintf(Buffer+length, "Good Morning");
length += sprintf(Buffer+length, "Good Afternoon");

Вот версия с некоторой устойчивостью к ошибкам. Это полезно, если вам все равно, когда происходят ошибки, если вы можете продолжать идти своим путем, когда они появляются.

int bytes_added( int result_of_sprintf )
{
    return (result_of_sprintf > 0) ? result_of_sprintf : 0;
}

int length = 0;
length += bytes_added(sprintf(Buffer+length, "Hello World"));
length += bytes_added(sprintf(Buffer+length, "Good Morning"));
length += bytes_added(sprintf(Buffer+length, "Good Afternoon"));
29 голосов
/ 20 апреля 2010

В целях безопасности (переполнение буфера) я рекомендую использовать snprintf ()

const int MAX_BUF = 1000;
char* Buffer = malloc(MAX_BUF);

int length = 0;
length += snprintf(Buffer+length, MAX_BUF-length, "Hello World");
length += snprintf(Buffer+length, MAX_BUF-length, "Good Morning");
length += snprintf(Buffer+length, MAX_BUF-length, "Good Afternoon");
12 голосов
/ 20 апреля 2010

A snprintfcat() оболочка для snprintf():

size_t 
snprintfcat(
    char* buf,
    size_t bufSize,
    char const* fmt,
    ...)
{
    size_t result;
    va_list args;
    size_t len = strnlen( buf, bufSize);

    va_start( args, fmt);
    result = vsnprintf( buf + len, bufSize - len, fmt, args);
    va_end( args);

    return result + len;
}
7 голосов
/ 18 февраля 2016

Используйте возвращаемое значение sprintf()

Buffer += sprintf(Buffer,"Hello World");
Buffer += sprintf(Buffer,"Good Morning");
Buffer += sprintf(Buffer,"Good Afternoon");
6 голосов
/ 20 апреля 2010

Почему вы хотите использовать sprintf для конкатенации строк, когда существуют методы, специально предназначенные для того, что вам нужно, например strcat и strncat?

3 голосов
/ 17 февраля 2016

Я думаю, что вы ищете fmemopen(3):

#include <assert.h>
#include <stdio.h>

int main(void)
{
    char buf[128] = { 0 };
    FILE *fp = fmemopen(buf, sizeof(buf), "w");

    assert(fp);

    fprintf(fp, "Hello World!\n");
    fprintf(fp, "%s also work, of course.\n", "Format specifiers");
    fclose(fp);

    puts(buf);
    return 0;
}

Если динамическое хранилище больше подходит для вашего случая использования, вы можете последовать прекрасному предложению Лиама об использовании open_memstream(3):

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

int main(void)
{
    char *buf;
    size_t size;
    FILE *fp = open_memstream(&buf, &size);

    assert(fp);

    fprintf(fp, "Hello World!\n");
    fprintf(fp, "%s also work, of course.\n", "Format specifiers");
    fclose(fp);

    puts(buf);
    free(buf);
    return 0;
}
3 голосов
/ 20 апреля 2010

Вы просто добавляете строковые литералы? Или вы собираетесь добавлять различные типы данных (целые числа, числа с плавающей запятой и т. Д.)?

Может быть проще абстрагировать это в свою собственную функцию (в следующем предполагается C99):

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

int appendToStr(char *target, size_t targetSize, const char * restrict format, ...)
{
  va_list args;
  char temp[targetSize];
  int result;

  va_start(args, format);
  result = vsnprintf(temp, targetSize, format, args);
  if (result != EOF)
  {
    if (strlen(temp) + strlen(target) > targetSize)
    {
      fprintf(stderr, "appendToStr: target buffer not large enough to hold additional string");
      return 0;
    }
    strcat(target, temp);
  }
  va_end(args);
  return result;
}

И вы бы использовали это так:

char target[100] = {0};
...
appendToStr(target, sizeof target, "%s %d %f\n", "This is a test", 42, 3.14159);
appendToStr(target, sizeof target, "blah blah blah");

и т.д.

Функция возвращает значение из vsprintf, которое в большинстве реализаций является количеством байтов, записанных в место назначения. В этой реализации есть несколько дыр, но она должна дать вам некоторые идеи.

2 голосов
/ 28 марта 2013

Я считаю, что следующий метод прекрасно работает.

sprintf(Buffer,"Hello World");
sprintf(&Buffer[strlen[Buffer]],"Good Morning");
sprintf(&Buffer[strlen[Buffer]],"Good Afternoon");
1 голос
/ 29 октября 2016

Использовать strcat http://www.cplusplus.com/reference/cstring/strcat/

int main ()
    {
      char str[80];
      strcpy (str,"these ");
      strcat (str,"strings ");
      strcat (str,"are ");
      strcat (str,"concatenated.");
      puts (str);
      return 0;
    }




    Output:


    these strings are concatenated. 
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...