2016-07-17 2 views
0
static char DEFAULT_DELIMITER = ','; 
char *calculator_retrieveFirstToken(char *numbers, const char * delimiters) { 
    //if (delimiters[strlen(delimiters)-1] == DEFAULT_DELIMITER) 
    if (delimiters[1] == DEFAULT_DELIMITER) { 
     return strtok(numbers, delimiters); 
    } 
    else { 
     char linefeed = '\n'; 
     strtok(numbers, &linefeed); 
     return strtok(NULL, delimiters); 
    } 
} 

char calculator_findDelimiter(const char *numbers) { 
    char delimiter = DEFAULT_DELIMITER; 
    if (strlen(numbers) >= 4 
     && numbers[0] == '/' 
     && numbers[1] == '/') { 
     delimiter = numbers[2]; 
    } 
    return delimiter; 
} 

double calculator_add(char *numbers) { 
    char delimiters[] = "\n"; 
    char *token = NULL; 

    char mainDelimiter = calculator_findDelimiter(numbers); 
    strcat(delimiters, &mainDelimiter); 
    token = calculator_retrieveFirstToken(numbers, delimiters); 
    return calculator_addEveryToken(token, delimiters); 
} 

Weirdest происходит, когда я отлаживаю функцию calculator_retrieveFirstToken.Функция не получает то, что я отправляю как const char *

Пример ввода чисел: ""

mainDelimiter равно ''

после вызова strcat: разделители равно: "\ п"

Когда я иду в Функция calculator_retrieveFirstToken Параметр разделителя равен: "\ n, \ n",

Я не понимаю, как это должно работать ... Что-то не так с моим вызовом strcat?

Вот главный

#include "calculator_kata.h" 
#include <assert.h> 
#include <string.h> 
#include <math.h> 
#include <stdbool.h> 

static char numbers[50]; 

static bool equals(const double expected, const double received) 
{ 
    double epsilon = 1e-7; 
    return fabs(received - expected) < epsilon; 
} 

static void test_addEmptyString_shouldReturnZero() 
{ 
    strcpy(numbers, ""); 
    const double sum = calculator_add(numbers); 
    assert(equals(0, sum)); 
} 

static void test_addOneNumber_shouldReturnThisNumber() 
{ 
    strcpy(numbers, "42"); 
    const double sum = calculator_add(numbers); 
    assert(equals(42, sum)); 
} 

static void test_addTwoNumbers_shouldReturnTheSum() 
{ 
    const double TWENTY_ONE = 9+10; 
    strcpy(numbers, "9,10"); 

    const double sum = calculator_add(numbers); 

    assert(equals(TWENTY_ONE, sum)); 
} 

static void test_addAnyQtyOfNumbers_shouldReturnTheSum() 
{ 
    strcpy(numbers, "1,2,3,4,5"); 
    const double sum = calculator_add(numbers); 
    assert(equals(15, sum)); 
} 

static void test_inputWithNewLines_insteadOfCommas_shouldStillBeParsed() 
{ 
    strcpy(numbers, "1\n2,3"); 
    const double sum = calculator_add(numbers); 
    assert(equals(6, sum)); 
} 

static void test_inputWithMultipleCommasBetweenNumbers_shouldReturnTheSum() 
{ 
    strcpy(numbers, "1\n,2,,,,,,,3"); 
    const double sum = calculator_add(numbers); 
    assert(equals(6, sum)); 
} 

static void test_inputWithCustomDelimiter_shouldChangeDelimiter() 
{ 
    strcpy(numbers, "//;\n1;2"); 
    const double sum = calculator_add(numbers); 
    assert(equals(3, sum)); 
} 

int main() 
{ 
    test_addEmptyString_shouldReturnZero(); 
    test_addOneNumber_shouldReturnThisNumber(); 
    test_addTwoNumbers_shouldReturnTheSum(); 
    test_addAnyQtyOfNumbers_shouldReturnTheSum(); 
    test_inputWithNewLines_insteadOfCommas_shouldStillBeParsed(); 
    test_inputWithMultipleCommasBetweenNumbers_shouldReturnTheSum(); 
    test_inputWithCustomDelimiter_shouldChangeDelimiter(); 
    return 0; 
} 
+0

Можете ли вы показать, как выглядит ваш главный объект с помощью некоторых примеров ввода/параметров, которые вызывают проблему? –

+0

@RetiredNinja Добавлены все основные –

+0

'strtok (numbers, & linefeed)' неверно. Параметр 'delim' является указателем * на строку с нулевым символом * NOT * указателем на char * (см.' Man 3 strtok') (** примечание: ** использование * string * в * man page * description 'delim') –

ответ

0

strtok и strcat принимают указатель на char массива (то есть строки) и не указатель на char переменной. То есть, им нужен ограничитель строки ('\0').

Кроме того, в calculator_add вы не определили достаточно места в , поэтому ваш strcat будет проходить мимо конца, вызывая неопределенное поведение.

Я аннотировал ваш источник с ошибками и предоставил исправления. Где это возможно, я сделал: // NOTE/BUG: ... #if 0 /* original code */ #else /* fixed code */ #endif:

static char DEFAULT_DELIMITER = ','; 
char *calculator_retrieveFirstToken(char *numbers, const char * delimiters) { 
    //if (delimiters[strlen(delimiters)-1] == DEFAULT_DELIMITER) 
    if (delimiters[1] == DEFAULT_DELIMITER) { 
     return strtok(numbers, delimiters); 
    } 
    else { 
     // NOTE/BUG: this is incorrect -- you must pass a string pointer _not_ 
     // the address of a [single] character 
#if 0 
     char linefeed = '\n'; 
     strtok(numbers, &linefeed); 
#else 
     char *linefeed = "\n"; 
     strtok(numbers, linefeed); 
#endif 

     return strtok(NULL, delimiters); 
    } 
} 

char calculator_findDelimiter(const char *numbers) { 
    char delimiter = DEFAULT_DELIMITER; 
    if (strlen(numbers) >= 4 
     && numbers[0] == '/' 
     && numbers[1] == '/') { 
     delimiter = numbers[2]; 
    } 
    return delimiter; 
} 

double calculator_add(char *numbers) { 
    // NOTE/BUG: this defines only enough space for a single character and the 
    // '\0' so when it is concatenated to, it is undefined behavior [you're 
    // running past the end of the string] 
#if 0 
    char delimiters[] = "\n"; 
#else 
    char delimiters[4] = "\n"; 
#endif 
    char *token = NULL; 

    // NOTE/BUG: this repeats the same mistake with the strtok above 
#if 0 
    char mainDelimiter = calculator_findDelimiter(numbers); 
    strcat(delimiters, &mainDelimiter); 
#endif 

    // NOTE: either of these will fix the strcat 
#if 0 
    char mainDelimiter = calculator_findDelimiter(numbers); 
    char temp[2]; 
    temp[0] = mainDelimiter; 
    temp[1] = 0; 
    strcat(delimiters, temp); 
#else 
    delimiters[1] = calculator_findDelimiter(numbers); 
    delimiters[2] = 0; 
#endif 

    token = calculator_retrieveFirstToken(numbers, delimiters); 
    return calculator_addEveryToken(token, delimiters); 
} 
Смежные вопросы