В продолжение
темыАвтор: Shmj
Дата: 05.07.23
, дабы довести до логического завершения. Всем спасибо за старания.
В .Net я о стадиях сборки даже не думал. Знал что есть сборка и JIT-компиляция (вот там важно — иногда эта JIT-компиляция занимала 10 секунд, что удавалось решить Ngen-ом, причем важно было не перепутать битность платформы). А есть ли там какие стадии сборки — особо не думал об этом.
C C++, как оказалось, это критически важно и без этого даже не поймешь почему не работает прога. Думал хватит понимания что есть 3 стадии:
1. Препроцессор — раскрывает #include, #define, #if и пр.
2. Сам компилятор для каждого файла создает o-файл.
3. Линкер собирает все в единую библиотеку или exe-файл.
Оказалось что этого не достаточно, что видно из примера. Сейчас привожу максимально упрощенный пример, он из 3 файлов:
Файл
demo.hpp
#pragma once
template <typename T>
class Checker
{
public:
Checker()
{
static_assert(sizeof(T), "Incomplete type");
}
};
class DerivedClass;
class BaseClass
{
public:
//BaseClass() {}
static void fun1();
private:
Checker<DerivedClass> m;
};
Файл
demo.cpp
#include "demo.hpp"
class DerivedClass
{
};
void BaseClass::fun1()
{
}
Файл с функцией входа:
#include "demo.hpp"
int main()
{
BaseClass::fun1();
}
Вопрос: почему если раскомментировать контруктор по умолчанию //BaseClass() {} — оно перестает компилироваться? Оказывается без понимания стадий сборки не поймешь этого. Тут чел.
приводил стадии трансляции, возможно с этим тоже связано.
Честно сказать, даже доведя до мин. примера (общими силами), я не до конца понимаю почему возникает ошибка компиляции при явном консрукторе. Примерно так — если сборщик видит явный конструктор, то он уже не ищет реализацию класса DerivedClass а просто сразу отрабатывает assert. А вот если конструктор не указать явно, то почему assert не отрабатывает?
Вот положа руку на сердце — вы понимаете логику сбоки в этом случае?
Здравствуйте, Shmj, Вы писали:
S>C C++, как оказалось, это критически важно и без этого даже не поймешь почему не работает прога. Думал хватит понимания что есть 3 стадии:
S>1. Препроцессор — раскрывает #include, #define, #if и пр.
S>2. Сам компилятор для каждого файла создает o-файл.
S>3. Линкер собирает все в единую библиотеку или exe-файл.
S>Оказалось что этого не достаточно, что видно из примера. Сейчас привожу максимально упрощенный пример, он из 3 файлов:
S>Вопрос: почему если раскомментировать контруктор по умолчанию //BaseClass() {} — оно перестает компилироваться? Оказывается без понимания стадий сборки не поймешь этого. Тут чел. приводил стадии трансляции, возможно с этим тоже связано.
S>Честно сказать, даже доведя до мин. примера (общими силами), я не до конца понимаю почему возникает ошибка компиляции при явном консрукторе. Примерно так — если сборщик видит явный конструктор, то он уже не ищет реализацию класса DerivedClass а просто сразу отрабатывает assert. А вот если конструктор не указать явно, то почему assert не отрабатывает?
S>Вот положа руку на сердце — вы понимаете логику сбоки в этом случае?
А что тут понимать, тут просто все как угол двери. Но сначала мне бы хотелось увидеть твои комментарии по поводу "Оказалось что этого не достаточно, что видно из примера". Почему знаний о трех фазах трансляции не достаточно и каким образом пример это демонстрирует?
Здравствуйте, Shmj, Вы писали:
S>Честно сказать, даже доведя до мин. примера (общими силами), я не до конца понимаю почему возникает ошибка компиляции при явном консрукторе.
Вот тебе подсказка: добавь непосредственно перед main определение еще одной функции:
void foo()
{
BaseClass b;
}
Эту функцию не обязательно даже где-то вызывать. Просто добавь определение, посмотри, что получилось и подумай еще. Я в тебя верю
| | Ответ |
| | Компилятор генерирует дефолтные конструкторы по мере необходимости. Да, вероятно, он видит неполноту типа, но в то же время он видит, что экземпляры класса здесь не создаются, а значит, конструктор не нужен и поводов для ошибки нет. Возможно, экземпляры этого класса создаются где-то в другой единице трансляции и возможно, в том месте определение класса уже видно. И совсем другое дело, когда дефолный конструктор определен пользователем явно. В этом случае компилятор обязан его откомпилировать и во время компиляции конструктора и вылазит неполнота типа и ошибка компиляции. |
| | |