atof coredump с getopt

Я пишу приложение на C++, которое конвертирует фаренгейты в градусы Цельсия и Кельвина, а кельвины в градусы Цельсия и Фаренгейта и т. Д. Поскольку писать здесь интерактивное приложение глупо, я решил ознакомиться с функцией getopt в unistd.h.

Формат:F2C -k 273.15

Выход:

FAHR CELSIUS KELVIN

32 0 273.15

Вот мой код:

#include <iostream>
#include <stdlib.h>
#include <unistd.h>

#define VERSION 0.1
#define HELP help(argv[0])

#define OPTS "vk:f:c:h"
float ver = (float)VERSION;

void help(char *s);
namespace Fahrenheit
{
    float FK(float F) {
        return ((5.0/9.0) * (F - 32.0) + 273.15);
    }

    float FC(float F) {
        return ((5.0/9.0) * (F - 32.0));
    }

    void printfahr(float F) {
        std::cout << "FAHR\t\tCELSIUS\t\tKELVIN" << std::endl;
        std::cout << F << "\t\t" << FC(F) << "\t\t" << FK(F) << std::endl;
    }      
}

namespace Celsius
{
    float CF(float C) {
        return ((C*(9/5)) + 32);
    }
    float CK(float C) {
        return (C+273.15);
    }
    void printc(float C) {
        std::cout << "FAHR\t\tCELSIUS\t\tKELVIN" << std::endl;
        std::cout << CF(C) << "\t\t" << C << "\t\t" << CK(C) << std::endl;
    }
}

namespace Kelvin
{
    float KF(float K) {
        return (((9.0/5.0) * (K-273.15)) + 32);
    }    
    float KC(float K) {
        return (K-273.15);
    }    
    void printk(float K) {
        std::cout << "FAHR\t\tCELSIUS\t\tKELVIN" << std::endl;
        std::cout << KF(K) << "\t\t" << KC(K) << "\t\t" << K << std::endl;
    }
}
int main(int argc, char *argv[])
{
    char arg = '\0';
    if(argc < 2 && argc == 1 && argc > 0) {
        help(argv[0]);
        exit(1);
    }
    /*** Use function getopt() defined in unistd.h to accept 5 arguments: -v, -h, -k, -f, and -c ***/
    while((arg=getopt(argc, argv, OPTS))!=-1)
    {
        float floatarg = atof(optarg);                                                      
        switch(arg)
        {

            case 'v':
                std::cout << "The current version is:" << ver << std::endl;
            break;

            case 'h':
                HELP;
            break;

            case 'k':
                Kelvin::printk(floatarg);
            break;

            case 'f':
                Fahrenheit::printfahr(floatarg);
            break;

            case 'c':
                Celsius::printc(floatarg);
            break;

            default:
                HELP;
            break;
        }
    }
    return 0;
}

void help(char *s) {
    std::cout << "Usage:\t"<< s << " [-option] [argument]" << std::endl;
    std::cout << "option:\t" << "-c [temperature]: convert a Celsius temperature to Fahrenheit and Kelvin" <<  std::endl;
    std::cout << "\t" << "-f [temperature]: convert a Fahrenheit temperature to Celsius and Kelvin" << std::endl;
    std::cout << "\t" << "-h: show help information" << std::endl;
    std::cout << "\t" << "-k [temperature]: convert a Kelvin temperature to Fahrenheit and Celsius" << std::endl;
    std::cout << "\t" << "-v: show version information" << std::endl;
}

Моя проблема в том, что всякий раз, когда я использую опцию, которая не принимает аргументов (например, -v), я получаю дамп ядра.

dbx показал мне, что SIGSEV происходит в строке 70 (float floatarg = atof(optarg);).

Когда я запускаю программу так:

./F2C -k 273.15

Математика выполнена правильно, и я получаю четкую распечатку. Это только когда я использую -v или же -h что моя программа СИГСЕВА.

Дополнительная информация:

Эта программа была скомпилирована с пакетом компиляторов Sun studio версии 5.12.

Я полностью сбит с толку, почему моя программа SIGSEV. Это противоречиво и не имеет смысла. Буду признателен за любую доступную помощь.

2 ответа

Решение

Должен был сделать некоторые optarg проверка. В конце концов, вы не можете конвертировать null на поплавок.

новый основной ():

#define FLOATARG atof(optarg)

int main(int argc, char *argv[])
{
    char arg = '\0';
    if(argc < 2 && argc == 1 && argc > 0) {
        help(argv[0]);
        exit(1);
    }
    /*** Use function getopt() defined in unistd.h to accept 5 arguments: -v, -h, -k, -f, and -c ***/
    while((arg=getopt(argc, argv, OPTS))!=-1)
    {                                                      
        switch(arg)
        {

            case 'v':
                std::cout << "The current version is:  << ver << std::endl;
            break;

            case 'h':
                HELP;
            break;

            case 'k':
                Kelvin::printk(FLOATARG);
            break;

            case 'f':
                Fahrenheit::printfahr(FLOATARG);
            break;

            case 'c':
                Celsius::printc(FLOATARG);
            break;

            default:
                HELP;
            break;
        }
    }
    return 0;
}

Самое короткое исправление:

        float floatarg = optarg ? atof(optarg) : 0.0;

Вы также можете переписать свой код как

    float floatarg = 0.0;
    switch(arg)
    {

        case 'v':
            std::cout << "The current version is:" << ver << std::endl;
        break;

        case 'h':
            HELP;
        break;

        case 'k':
            floatarg = atof(optarg);
            Kelvin::printk(floatarg);
        break;

        case 'f':
            floatarg = atof(optarg);
            Fahrenheit::printfahr(floatarg);
        break;
...

или же

    float floatarg = 0.0;
    if(optarg) {
        floatarg = atof(optarg);
    }
    switch(arg)
    {

        case 'v':
            std::cout << "The current version is:" << ver << std::endl;
        break;

        case 'h':
            HELP;
        break;

        case 'k':
            Kelvin::printk(floatarg);
        break;

        case 'f':
            Fahrenheit::printfahr(floatarg);
        break;
...
Другие вопросы по тегам