не использует strcmp должным образом в C - PullRequest
0 голосов
/ 18 февраля 2012

Я написал этот метод, чтобы найти наибольшую степень 2, которая будет соответствовать данному десятичному числу.Десятичное число в формате массива символов, чтобы избежать ошибки переполнения ввода при хранении числа.Степени 2 рассчитываются с использованием формата pow (2, power) с плавающей точкой, т.е.8.000000 Это число затем отправляется методу для удаления периода и нулей, которые следуют.то есть.8.000000 превращается в 8

1  #include <string.h>
2  #include <stdio.h>
3  #include <stdlib.h>
4  #include <memory.h>
5  #include <math.h>
6
7   int i;
16
17  void removeFloatZeros(char *floatvalue)
18  {
19      char *ptr = strchr(floatvalue, '.');
20      *ptr = '\0';
21  }
22
45
173 char *decimalToBinary(char *decimal)
174 {
176     int x;
177     double power = 0;
178     char *binary = malloc(sizeof(char *) * 1024);
179     char *twosPower = malloc(sizeof(char *) * 1024);
180
181     /* What is the greatest power of 2 that will fit into the decimal? */
182     for(x = 0; x <= 30; x++)
183     {
184         power = pow(2.0, x);
185         snprintf(twosPower, 1023, "%f", power);
186         removeFloatZeros(twosPower);
189         printf("strcmp(decimal, twosPower) = %d\n", strcmp(twosPower, decimal));  
190         memset(twosPower, '\0', 1023);
191     }
214 }
215
216 int main(int argc, char*argv[])
217 {
218     char *dec1 = argv[1];
219     decimalToBinary(dec1);
220     return 1;
221 }
222

Например, если я введу 20 в argv [1], он выдаст:

strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = -1  
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = 1
strcmp(decimal, twosPower) = -1

Где я ошибаюсь?Кроме того, игнорируйте конечное условие для цикла for.Предполагается выводить все 1 до 6-й итерации и все -1 на 6-й и после 6-й итерации.

Ответы [ 2 ]

2 голосов
/ 18 февраля 2012

strcmp возвращаемые значения:

Нулевое значение указывает, что обе строки равны.

Значение больше нуля указывает, что первый не соответствующий символ имеет большее значение в str1, чем в str2.

И значение меньше нуля указывает на обратное.

Ваш вклад: 20 Ваша первая итерация цикла: twosPower = "1" strcmp ("20", "1")

Первый символ не совпадает и имеет меньшее значение в str2 ("1"), чем в str1 ("2") -> возвращает отрицательное значение.

Остальные итерации должны объясниться ...

Кроме того, Изменить:

printf("strcmp(decimal, twosPower) = %d\n", strcmp(twosPower, decimal)); 

В вашей строке формата printf указано противоположное тому, что вы делаете в параметре.

Edit:

str1    str2    
1       20  First char that differs is '1' vs. '2'. '1' (ASCII 49) is smaller than '2' (ASCII 50), 49 - 50 = -1 = return value
2       20  First char that differs is '\0' vs. '0'. '\0' (ASCII 0) is smaller than '0' (ASCII 48), 0 - 48 = -48 = return value
4       20  First char that differs is '4' vs. '2'. '4' (ASCII 52) is greather than '2' (ASCII 50), 52 - 50 = 2 = return value
8       20  First char that differs is '8' vs. '2'. '4' (ASCII 56) is greather than '2' (ASCII 50), 56 - 50 = 6 = return value
16      20  First char that differs is '1' vs. '2'. '1' (ASCII 49) is smaller than '2' (ASCII 50), 49 - 50 = -1 = return value

... and so on ...

Может быть, этот вывод поможет немного больше

Кроме того, ваш метод нахождения наибольшей степени 2 в числе ошибочен, так как возвращаемое значение strcmp просто зависит от первого символа, который отличается. Поэтому strcmp ("2", "16") и strcmp ("200000000", "16") всегда возвращают одно и то же.

1 голос
/ 18 февраля 2012

Эта очищенная версия вашего кода производит вывод:

$ ./xx 20
strcmp(1, 20) = -1
strcmp(2, 20) = -48
strcmp(4, 20) = 2
strcmp(8, 20) = 6
strcmp(16, 20) = -1
strcmp(32, 20) = 1
strcmp(64, 20) = 4
strcmp(128, 20) = -1
strcmp(256, 20) = 5
strcmp(512, 20) = 3
strcmp(1024, 20) = -1
strcmp(2048, 20) = 52
strcmp(4096, 20) = 2
strcmp(8192, 20) = 6
strcmp(16384, 20) = -1
strcmp(32768, 20) = 1
strcmp(65536, 20) = 4
strcmp(131072, 20) = -1
strcmp(262144, 20) = 6
strcmp(524288, 20) = 3
strcmp(1048576, 20) = -1
strcmp(2097152, 20) = 57
strcmp(4194304, 20) = 2
strcmp(8388608, 20) = 6
strcmp(16777216, 20) = -1
strcmp(33554432, 20) = 1
strcmp(67108864, 20) = 4
strcmp(134217728, 20) = -1
strcmp(268435456, 20) = 6
strcmp(536870912, 20) = 3
strcmp(1073741824, 20) = -1
$

Достаточно сказать, что есть много небольших изменений.

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

static void removeFloatZeros(char *floatvalue)
{
    char *ptr = strchr(floatvalue, '.');
    *ptr = '\0';
}

static void decimalToBinary(char *decimal)
{
    int x;
    double power = 0;
    char *twosPower = malloc(sizeof(char *) * 1024);

    /* What is the greatest power of 2 that will fit into the decimal? */
    for(x = 0; x <= 30; x++)
    {
        power = pow(2.0, x);
        snprintf(twosPower, 1023, "%f", power);
        removeFloatZeros(twosPower);
        printf("strcmp(%s, %s) = %d\n", twosPower, decimal, strcmp(twosPower, decimal));  
        //printf("strcmp(decimal, twosPower) = %d\n", strcmp(twosPower, decimal));  
        memset(twosPower, '\0', 1023);
    }
    free(twosPower);
}

int main(int argc, char*argv[])
{
    for (int i = 1; i < argc; i++)
        decimalToBinary(argv[i]);
    return 0;
}

Отображение сравниваемых значений делаетвещи намного легче понять.Вам необходимо освободить память (или использовать автоматические массивы).Вам нужно использовать заголовки.Декларации static не являются необходимыми на 100%, но означают, что я не получаю никаких предупреждений при компиляции с настройками компилятора über-fussy.

Но ключевое изменение - это распечатка значений, которыесравнение - имеет смысл цифры от strcmp().

(рассмотрите возможность добавления подходящего числа ведущих нулей перед выполнением сравнения.)

...