Ответы пользователя по тегу C++
  • Как расширить размер динамического массива типа string в с++?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Как расширить размер с сохранением данных(realloc) с массивом string в с++

    Думаю, что по честному (т.е. не завязываясь на конкретное внутреннее представление string) -- никак, только выделив новый массив и скопировав/переместив объекты из старого массива в новый. Имеет смысл эту работу делегировать стандартному контейнеру vector, где эта логика уже должна быть реализована.
    Ответ написан
    Комментировать
  • Может ли прерывание прервать выполнение конструктора / деструктора в С++?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Может ли прерывание прервать выполнение конструктора / деструктора в С++?

    Да, конечно может, как и любой другой код, явно не защищённый от прерываний.

    нужно ли в конструкторах / деструкторах защищать код критическими секциями?

    Нужно защищать любое состояние, к которому может обращаться код из разных контекстов выполнения -- например из разных потоков или из потока и обработчика прерывания. В ядре linux (где разных типов контекста много и разных способов синхронизации тоже много) есть руководство, что и когда использовать.
    Ответ написан
    Комментировать
  • Очень странная ошибка MPI?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    в чем ее суть

    в том, что процесс с rank == 0 посылает сообщение самому себе, и это становится для него неожиданностью.

    как с ней бороться

    как вариаент -- не посылать сообщение самому себе из процесса с rank == 0. Например так:
    if(rank == 0){
            for(size_t i = 1; i < size; i++){

    надо будет, конечно, пересмотреть все места где работа разделяется на size частей, и скорректировать для size - 1.
    Ответ написан
    Комментировать
  • Почему компилятор не генерирует конструктор по умолчанию в отдельной единице трансляции?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Почему компилятор не генерирует конструктор

    потому что нигде не написано, что он должен? Я вижу, что в коде генерируемом gcc инициализация таблицы виртуальных функций инлайнится по месту создания объекта, вне зависимости от опций оптимизации.
    Ответ написан
    1 комментарий
  • Почему можно не указывать virtual в файле реализации?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Почему можно не указывать virtual в файле реализации?

    Не то чтобы "можно не указывать". Нужно не указывать. Потому что так говорит стандарт.
    И потому что виртуальность функции -- это не свойство функции, а свойство класса которому она принадлежит. Код функции не зависит от того, виртуальная она или нет.
    Ответ написан
    4 комментария
  • Почему виснит программа на MPI?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Почему виснет программа на MPI?

    Потому что процесс с rank = 0 хочет послать VEC_SIZE сообщений, а процессы с rank > 0 обрабатывают каждый только по одному сообщению, после чего завершаются. Тебе нужно как-то объяснить исполнителям, сколько запросов они должны обработать перед завершением. Например так:

    if(rank == 0){
                    for(int i = 0; i < VEC_SIZE; i++){
                            …
                    }
                    for (int i = 1; i < size; ++i) {
                            int q = -1;
                            MPI_Send(&q, 1, MPI_INT, i, 0, MPI_COMM_WORLD);
                    }
            }else{
                    for (;;) {
                            int index;
                            MPI_Status status;
                            MPI_Recv(&index, 1, MPI_INT, 0, 0, MPI_COMM_WORLD, &status);
                            if (index < 0)
                                    break;
                            …
                    }
            };


    Отдельно стоит сказать о том, что цикл который посылает задачу процессу и тут же ждёт результата от этого же самого процесса не имеет смысла с точки зрения улучшения производительности. Имеет смысл сначала разослать задания свободным исполнителям, а только после этого начинать ждать от них ответов.

    И ещё по мелочи:
    for(size_t i = 0; i < VEC_SIZE; i++){
                size_t num_process = (i % (size - 1)) + 1;
                MPI_Send(&i, 1, MPI_INT, static_cast<int>(num_process), 0, MPI_COMM_WORLD);

    размер size_t i может отличаться от размера int, подразумеваемого типом MPI_INT. Этот код точно будет работать неправильно на big-endian архитектурах с LP64 ABI.
    Ответ написан
    Комментировать
  • Не работает деструктор. В чем ошибка?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Не работает деструктор

    "Не работает деструктор" -- это когда программу компилируешь, запускаешь, а деструктор не делает того, что в нём написано. У тебя же программа не компилируется. Из-за того, что в настройках компилятора стоит "трактовать предупреждения как ошибки" и "предупреждать о неэффективном коде С++". Ну и вдобавок, предупреждение тебе правильно говорит, что если ты не реализовал конструктор копирования и оператор присваивания для класса самостоятельно управляющего памятью, то жди проблем, когда начнёшь копировать объекты этого класса.
    Ответ написан
    6 комментариев
  • Как сделать сжатие и распаковку char* в C++ с помощью zlib?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    что я делаю не так?

    Несколько вещей. В compress_char ты сначала выделяешь strlen(source) байт для вывода, а потом находишь compSize, который может быть и больше чем strlen(source). Ты не возвращаешь compSize после вызова compress, а он может измениться. Ты возвращаешь только char*, как будто это строка, но это не строка, там могут быть нулевые байты.
    В uncompress_char почти те же самые ошибки. Плюс какая-то путаница между length, compSize и ucompSize. На вход uncompress в параметре compSize должен попасть настоящий размер сжатых данных, а не догадка о нём на основе вызова compressBound.
    Я бы поменял код как-то так:
    char* compress_char(const char* source, size_t *outCompSize, size_t *outUcompSize)
    {
        uLong ucompSize = strlen(source) + 1;
        uLong compSize = compressBound(ucompSize);
        char* bytes = malloc(compSize);
        compress((Bytef *)bytes, &compSize, (Bytef *)source, ucompSize);
        *outCompSize = compSize;
        *outUcompSize = ucompSize;
        return bytes;
    }
    
    char* uncompress_char(const unsigned char source[], size_t inCompSize, size_t inUcompSize)
    {
        uLong ucompSize = inUcompSize;
        uLong compSize = inCompSize;
        char* message = malloc(ucompSize);
        uncompress((Bytef *)message, &ucompSize, (Bytef *)source, compSize);
        return message;
    }
    
    void foo(void)
    {
        size_t a, b;
        char *p = compress_char("{\"a2\":\"b\"}", &a, &b);
        char *p1 = uncompress_char(p, a, b);
        printf("%s, %zd -> %zd\n", p1, b, a);
    }


    Плюс ты не обрабатываешь коды возврата из compress/uncompress.
    Ответ написан
    Комментировать
  • Статическое подключение zlib в .so?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    У меня есть код на C++, который использует zlib. Я хочу на линуксе его скомпилировать в .so библиотеку, но чтобы команда ldd не показывала libz.so или libz.so.1 (чтобы zlib был встроен в .so).

    Ключами линковщика -Bstatic и -Bdynamic можно переключать режим линковки библиотек:
    g++ -shared foo.cc -o libfoo.so -Wl,-Bstatic -lz -Wl,-Bdynamic

    Но если так сделать, то можно заметить, что не всякую статическую библиотеку можно прилинковать к .so, а только такую, объектники которой собраны с флагом -fpic.
    Ответ написан
  • Почему substr работает странно?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Последняя кавычка ни в какую не хочет убираться, не могу понять, в чем дело
    std::string variable_value = var.substr(variable_value_start + 1, variable_value_end - 1);

    Второй параметр std::string::substr -- это длина выделяемой подстроки, а не индекс конца.
    Должно быть
    std::string variable_value = var.substr(variable_value_start + 1,
                                            variable_value_end - variable_value_start - 1);
    Ответ написан
  • Что не так с кодом на C++?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    for (int i = 0; i<2; i++){
        group.PutCar(i,Car(marks[i],powers[i],costs[i],displaces[i]));
      }

    -- заполняет элементы группы 0 и 1
    group[2].Print();

    -- пытается напечатать элемент группы 2, не заполненный. Думаю, что дело в этом.

    Ещё пара глюков:
    void Group::Print(){
      for (int i = 0; i<sizeof(array); i++){
        array[i].Print();
      }
    };

    sizeof работает не так и тут делает не то, что вы ожидаете. Для количества элементов в группе используйте имеющийся в классе Group size.

    operator double (){
        int summ = 0;
        for(int i = 0; i < sizeof(array); i++){
        summ += array[i].getCost();
        }

    то же самое здесь.

    Традиционно добавлю, что правильным было бы использование std::string (в Car) и std::vector (в Group), но подозреваю, что это это было такое задание.
    Ответ написан
    Комментировать
  • Можно ли записать структуру в файл используя basic_ofstream?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Можно ли записать структуру в файл используя basic_ofstream?

    Можно, но для этого не нужно делать её тип аргументом шаблона basic_ofstream.
    Ответ написан
  • Почему переопределение метода без virtual -- это не переопределение?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Если в официальной доке с++ где-то написано "переопределение возможно тогда и только тогда, когда есть virtual у метода базового класса"

    Написано, ровно это: eelis.net/c++draft/class.virtual#2

    то почему именно так?

    Потому что это определение термина function override в C++. Для других вариантов взаимодействия имён функций есть другие термины.
    Ответ написан
    Комментировать
  • Почему возникает ошибка при компилации gtest?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    не могу понять в чем проблема
    ошибка: undefined reference to `CheckUP::checkUP(std::__cxx11::basic_string, std::allocator >&, float&)'

    Проблема в том, что ты не слинковал свой тест с кодом реализующим класс CheckUP.
    Ответ написан
  • Что не так с функцией на C++?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    error C2835: user-defined conversion 'operator`double'' takes no formal parameters

    Он говорит тебе о том, что такая перегрузка: operator double (int limit) невозможна. Вторая ошибка -- это следствие первой.

    Определим в объекте Group две функции с одинаковым именем, например double Age(); и double Age(int limit);.

    Так бы и сделал, зачем оператор для этого перегружать?
    Ответ написан
    Комментировать
  • Почему не возникает ошибка при вводе слова больше длины чем выделена память?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    должна возникнуть ошибка, но по факту её не возникает.

    А ты собери свою программу с санитайзером памяти (-fsanitize=address) и будет тебе ошибка.
    Обращение к памяти за пределами выделенных массивов -- это UB, отсутствие видимого эффекта -- допустимый вариант поведения при UB.
    Ответ написан
    Комментировать
  • Как исправить ошибку "invalid use of incomplete type"?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    как правильно выстроить определения данных классов, что бы программа компилировалась?

    Перенести определения классов A и S после определения класса M, добавить объявление класса S перед определением класса D.
    Ответ написан
  • Почему в С++ появляется Segmentation fault?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    конкретную причину ошибки в коде так и не уставновил

    Да вот же она: Tree* head{0}; -- в head у тебя NULL.
    А std::cin >> head->sizeOfChild пытается читать в поле объекта на который указывает head.
    Ответ написан
    Комментировать
  • Process finished with exit code 139 (interrupted by signal 11: SIGSEGV)?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    Никак не пойму что за ошибка, и что она хочет от меня.

    Ошибка -- обращение к памяти, к которой нельзя обращаться, первая ссылка по запросу sigsegv в гугле.
    Хочет она от тебя чтобы ты веделил память для colonies[i].aparts перед тем как к ним обращаться.
    Ответ написан
    Комментировать
  • Как исправить проблему с неоднозначностью?

    jcmvbkbc
    @jcmvbkbc
    "I'm here to consult you" © Dogbert
    void EcsSystems::Add(IEcsSystem* system) {
      _systems.push_back(system);
    
      if (IRunSystem* runSystem = static_cast<IRunSystem*>(system)) {
        _runSystems.push_back(runSystem);
      }
    }

    Это странное решение -- сначала потерять всю информацию о типе, а потом м помощью каста выяснять, а каким же он был. К тому же здесь очевидно имелся в виду dynamic_cast, потому что результат static_cast для не-NULL аргумента всегда будет отличным от NULL.
    Ответ написан
    Комментировать