А не задумывались ли вы, что узкое место при разработке чего-либо — это программист? Как бы вы быстро не печатали на клавиатуре, всё равно современные компиляторы компилируют намного быстрее. Мало кто знает, что оптимизация программ заключается не в том, чтобы программы быстро работали на процессоре, а в том, чтобы конечному пользователю было комфортно работать за компьютером. Предлагаю вам прототип параллельного компилятора, который компилирует исходный текст в тот самый момент, когда вы его пишите.
Изначально исходный файл пуст (см. рис. ниже), точно также объектная модель тоже пуста.
По мере того, как вы набираете текст программы, в параллельных потоках компилятор анализирует введенный текст, проверяет синтаксис, создает необходимые внутренние объекты (см. рис. ниже). Например, подключение заголовочных файлов — это вполне самостоятельный фрагмент программы, и компилятор сразу подключает необходимые пространства имен, идентификаторы и прочее. Это отражается в консоли сообщением «Библиотеки успешно включены», а соответствующие строки в редакторе имеют зелененький фон — «всё ок».
В процессе написания кода компилятор делает различные подсказки. В этом примере, в момент ввода последней строки компилятор встретил неизвестный идентификатор. Сразу же пишется сообщение в консоль с предложением задекларировать этот объект локально, глобально или как параметр. Также компилятор сообщает, что определение функции f1 ещё не завершено. Соответствующие строки кода подсвечиваются красненьким. На рисунке ниже показан результат автоматического определения объекта.
В момент написания последних строк программы (см. рис. ниже), компилятор определил, что текущий исходный файл не содержит ошибок и является завершенным. В консоли сообщается, что файл не содержит ошибок, а также информация о том, что функция f1 успешно определена.
Такой компилятор не нуждается в кнопке «Компилировать», потому что синтаксический анализ, построение внутренних объектов и генерация объектного кода происходит в фоновых параллельных потоках. Раз уж сейчас так популярны 2-, 4-ядерные процессоры, то почему бы не сделать такой подарок программистам.
Ответы на некоторые вопросы.
Вопрос: а не будет ли это тормозить, не будет ли трещать постоянно винчестер?
Ответ: если грамотно спроектировать и реализовать, то не будет. Речь идет не о полной перекомпиляции всего файла при вводе очередного символа, а об инкрементальной компиляции, когда перекомпилируется (или докомпилируется) блок, строчка или даже выражение исходного файла. Эти фрагменты достаточно малы, чтобы быть обработанными за считанные миллисекунды.
Вопрос: а как же быть, если мне попался уже готовый проект?
Ответ: при открытии проекта начать его компилировать в фоне, показывая прогресс в консоли.
Вопрос: где можно скачать такой компилятор или среду разработки?
Ответ: мне такие компиляторы не известны. Это только идея, как было бы удобно со стороны пользователя. Частично такая функциональность реализована в IntelliSence в Microsoft Visual Studio, когда на лету анализируются объекты, подчеркиваются неизвестные идентификаторы.
Вопрос: а если нет программы, то где ты делал скриншоты?
Ответ: скриншоты сделаны в Визио 2007.
Если ещё будут вопросы в комментариях, то обновлю пост и допишу ответы.
еще лучше чувствуешь себя с Re-Sharper'ом от JetBrains - он очень много полезных штук подсказывает и кучу кода сам генерит (в хорошем смысле этого слова), поэтому меньше отвлекаешься на написание и синтаксические ошибки и больше фокусируешься на смысловой нагрузке. А по поводу скорости набора, имхо, она не так критична - во-первых, все равно больше времени думаешь, чем пишешь, а во-вторых, скорость приходит с опытом :)
ОтветитьУдалитьДа, но когда у тебя цикл написание—компиляция—запуск—исправление—компиляция—запуск, то тут уже не очень уютно себя чувствуешь. :)
ОтветитьУдалитьРешарпер значительно расширяет возможности IntelliSence как раз в сторону основной идеи поста.
ОтветитьУдалитьНаверное, я плохо указал основную идею поста. Она заключается в инкрементальной компиляции на очень низком уровне (операторы, выражения) и полном отсутствии кнопки "Compile".
ОтветитьУдалитьЛинтинг тоже важен, но это ни для кого уже не новость.
Причем тут компиляция? Это обычная проверка синтаксиса, выполняющаяся постоянно. Давно уже есть такое. Например, в QDevelop.
ОтветитьУдалитьИзобретатель велосипедов, блин.
А первый абзац вообще набор бреда. Причем тут клавиатура и компилятор? Оптимизация и работа за компьютером?
Для тупого анона повторяю: компиляция проекта должна идти параллельно с набором исходного кода. Проверка синтаксиса здесь — это побочная фича. Ну что тебе ещё не понятно?
ОтветитьУдалитьНу-ну. Как понять "компиляцию на очень низком уровне"(операторы e.t.c.)? Разве современные компиляторы не компилируют на этом уровне? Или имеется ввиду - ввел оператор и сразу же скомпилировалось? Интересно. Интересно, как Вы собираетесь компилировать незаконченное выражение?
ОтветитьУдалитьАнониму, о вижу, начинаешь просвящаться. :) Сейчас проясню.
ОтветитьУдалитьТы правильно пишешь, что компилятор переводит операторы С++ в язык ассемблера. Т. е. что-то вроде таблицы соответствия. Когда ты набрал несколько операторов, например:
int main()
{
int a = 0, b = 0;
cin >> a >> b;
То компилятор уже может откомпилировать фрагмент программы, потому что эти строки самодостаточны. Ты продолжаешь:
int c = a + b;
cout << c;
return 0;
}
Эти строки тоже сами по себе самодостаточны и могут быть по отдельности откомпилированы. А потом тебе захотело поменять cin на scanf. Ты полез, поменял ту строку. А этот умный компилятор не всё перекомпилировал, а нашел входждение старого оператора и заменил его на новый. Т. е. транслировал в ассемблер одну единственную строчку.
Как-то так. Если всё равно мутно, то могу попробовать нарисовать диаграмки, прояснить, что имеется в виду выше.
>> Ты правильно пишешь, что компилятор переводит операторы С++ в язык ассемблера.
ОтветитьУдалитьЯ такого не писал. Вы наврали. Но хоть покраснели? Сознайтесь.
------
Не переводит компилятор "по таблице" ничего! Почитайте книги по устройству компиляторов, изучите, например, как работает GCC. Сначала происходит синтаксический разбор кода. Без завершенности кода невозможно провести корректный разбор. Потом код разбирается на токены. Строится абстрактное дерево программы. Тут выполняются различные оптимизации - оптимизируются циклы(например, если можно, переносится условие завершения в конец), оптимизируется хвостовая рекурсия и т.д. А потом уже перевод в процессорно-независимый ассемблерно-подобный код, оптимизация его, и только после генерация асм-кода под конкрентую платформу. Все выше было в основном о GCC, но общие принципы у всех компиляторов.
Кстати компилятору нужна инфа о всех переменных, используемых в программе. Строится граф зависимости этих переменных друг от друга. Это надо для оптимального распределения регистров под переменные. И тут не прокатит добавление одной переменной без перекомпиляции всего кода, связанного с этой переменной. Регистры будут уже распределены между имеющимися переменными. Как Вы изволите новую впихнуть? Компилируя только "одну самодостаточную строчку"? Без перекомпиляции и перераспределения регистров?
>> Как-то так. Если всё равно мутно, то могу попробовать нарисовать диаграмки, прояснить, что имеется в виду выше.
Судя по Вашему "как-то так" мутно у Вас в голове. Ненадо мне ничего рисовать и прояснять. Проясните себе. Учитесь, учитесь.
---
Итак, у Вас 2 выхода: либо Вы читаете книги по устройству компиляторов, после чего приходите сюда и прилюдно каетесь в том, что писали бред. Либо, если не осилите их, Вы честно подаете заявление об увольнении из университета по профнепригодности и идете заниматься соответствующей Вашему интеллекту деятельности - рыть каналы например. Это будет поведением порядочного человека. И студентов не испортите, и обществу пользу принесете. Но я надеюсь, Вы-таки прочтете и поймете.
Кстати, что за надпись "Современный компилятор" в заголовке окна? Вы отличаете редактор/IDE от компилятора?
товарищ анонимный комментатор, а вам не кажется, что вы немного палочку то перегибаете? я раньше считал, что порядочные люди представляются, а не дули в карманах на парах крутят ;)Есть мнение - выскажи, аргументируй, но зачем на ровном месте оскорблять? блог - это как раз то место, где можно разные идеи обсудить - brain storming могучая штука. Вобщем, искренне желаю Вам, побыстрее повзрослеть, вывести прыщики с лица, избавиться от комплексов и выработать уважение к себе и коллегам.
ОтветитьУдалитьА по теме поста - я себе тоже слабо представляю как оно эффективно работать должно, ведь ты когда код пишешь далеко не всегда меняешь одну строчку или оператор. Очень часто используется рефакторинг а ля Rename, Extract Мethod и т.п. С другой стороны билд проекта далеко не всегда из IDE вызывается, а иметь 2 разных компилятора для разных сценариев немного абсурдно. Вобщем я пока остаюсь при своем мнении - могучая IDE отдельно, и быстрый компилятор тоже отдельно :)
Тупому глупому анону: ты всё-таки туп и глуп. Ты пойми, что просто кинуть запрос в Гугл — это не достаточно, чтобы поддерживать здесь диалог.
ОтветитьУдалитьА теперь про устройство компиляторов. Во-первых, я очень близко знаком с устройством компилятора VHDL. А во-вторых, я лично написал парсер С++ (стандарт 2003 года) используя FLEX/BISON. А теперь подумай: нужно ли тебе мне объяснять, что происходит с исходником, и как он преобразовывается в токены, как работает BNF, как срабатывают правила в BNF? Специльно не пишу, что такое BNF, чтобы ты дал себе труд прочесть в Википедии. :)
> Тут выполняются различные оптимизации
Ты хоть знаешь, что это такое? А в Debug'е они тоже делаются? Если хочешь, я специально для интересующихся предметом написал методичку по оптимизации, можешь скачать и почитать. Ты можешь очень легко это проверить, если будешь писать программу и смотреть листинг ассемблера, как меняется код, когда ты меняешь что-либо внутри. Подчеркиваю, что код ассемблера в дебаге и в релизе будет разный. Компилятор есть? Возьми, попробуй.
Виталя, as-you-type компиляция — это всего-лишь идея, как можно было бы эксплуатировать многоядерные процессоры. Поэтому, каждый отдельный вопрос, типа а "что если" нужно рассматривать, составлять из них требования к такому компилятору и понять, сможет ли он за комфортное для программиста время сделать эти преобразования.
ОтветитьУдалитьВчера ещё придумали компромис: компилятор начинает работать не одновременно с набором текста, а через 2-3 секунды неактивности со стороны программиста. Как только программист вернулся к набору текста — отменять все задачи по компиляции.
Всё становится сложнее в режиме релиз. Ведь там есть межпроцедурные оптимизации, поэтому изменение одного оператора может повлечь изменения в других функциях или даже файлах.
ну может. Я так понимаю многое зависит от специфики разрабатываемого приложения. лично для меня пока такой компилятор неактуален, хотя, возможно, это дело привычки :) у меня при отладке больше времени занимает "запуск", но его иногда получается минимизировать за счет запуска конкретного юнит теста, а не всего приложения.
ОтветитьУдалитьЗЫ: формочка для текста комментария - отстой :( в ней не работает ни буфер обмена, ни влево-вправо :(
vitalya, оно не работает из-за того, что у тебя фаерфокс. У всех других оно работает. :) У Блоггера есть ряд недостатков, ты не видел, как я мучаюсь, когда постинги делаю. Ещё немного помучаюсь и перееду на Вордпресс на своем домене.
ОтветитьУдалитьА в той среде, где ты работаешь, нет edit and continue?
vitalya комментирует...
ОтветитьУдалить> ну может. Я так понимаю многое зависит от специфики разрабатываемого приложения
Вообще изначально было так: ты пишешь на С++, а в соседнем окне оно тебе рисует результат RTL-синтеза. И ты можешь визуально контролировать, какая логика у тебя получается, что-то менять, что-то оптимизировать.
2vitalya: я помоему нигде не перегибал. Веду спор нормально, не цепляюсь к недосказанностям и не приписываю оппоненту своих слов, короче: демагогию не развожу. Так что все порядочно. И мнение я неплохо аргументировал. Скажите, что неясно.
ОтветитьУдалитьА насчет анонимности: ну нету у меня аккаунта в ЖЖ, а если я напишу имя, оно скажет не больше, чем "Анонимный". Ну таки ладно, если Вам так нужна личность, а не аргументы, с этого поста начинаю именоваться RL-именем. Да, прыщи мне абсолютно не мешают, их немного.
2Volodymyr Obrizan: Вы ясновидец, наверное? Уже приписываете мне гугль. А про BNF, спасибо, знаком хорошо и без педевикий.
И что такое оптимизации, знаю.
Я перечислил основные оптимизации, используемые GCC на всех платформах. Ой-вей, не написал про перетасовку команд, так Вы сразу придрались. Демагог? Интересно, она где-нибудь нужна, кроме как на x86(-64)?
И спасибо за отсылку к листингам - моим развлечением компиляция с разными ключами и просмотр листингов было года 2-3 назад.
Андрей, если не ссышь быть ассоциирован с тем навозом, который ты уже написал, допиши ещё свою фамилию, город и номер школы. :)
ОтветитьУдалитьУ самоуверенного ламера закончились аргументы, а гугл новых не подкидывает?
ОтветитьУдалитьАндрей, вот смотри. У меня появилась идея, которая заключается в том, что не нужно ждать пока проект будет написан полностью, чтобы начать его компилировать. Это интересно не только для компиляции С++ в машинный код, но и для синтеза С++ в RTL (может ты ещё и знаешь, что такое "синтез" и RTL?). И вот я набравшись смелости, подписавшись реальным именем, указав где я работаю, выношу эту идею на суд своих друзей.
ОтветитьУдалитьТы пришел ко мне на сайт, не разобрался в этой идее, назвал меня "изобретателем велосипедов", сказал, что я пишу "бред". Дальше, не зная меня лично, не побывав на моих лекциях по оптимизации и параллельному программированию, ты предлагаешь мне написать заявление на увольнение, и называешь "ламером".
Вот как к тебе после этого относиться и как с тобой разговаривать?
я в студии работаю, но Edit and Continue не использую
ОтветитьУдалитьАндрей, лично мне, ваше высказывание по поводу проф пригодности\непригодности автора поста показалось абсолютно неадекватным, о чем я, собственно, и написал.
ОтветитьУдалить>> У меня появилась идея, которая заключается в том, что не нужно ждать пока проект будет написан полностью, чтобы начать его компилировать.
ОтветитьУдалитьТ.е. до Вас никто не компилировал незавершенные проекты? Я, например, иногда отдельно взятые функции тестирую, без всякого намека на завершенность всего проекта, и ничего.
>> назвал меня "изобретателем велосипедов"
То, что изображено на скринах никак не относится к компиляции, а вполне сходит за проверку синтаксиса. Ну чуть более интеллектуальную, чем УЖЕ использующиеся. Потому и назвал.
>> сказал, что я пишу "бред"
Бред-таки пишете, особенно это "таблицы соответсвия операторов асм. кодам". Или Вы кагбэ изложили попроще, чтоб мне, на Ваш взгяляд, тупому, было понятно? Это кстати, и есть ламеризм: считать, что все вокруг тупые и изъяснятся с ними надо на пальцах, как с дикарями. Пишите нормально, не стесняйтесь.
Таки да, теперь про идею. Сказать и описать суперидею, как все это будет фифигительно, это одно. Вы объясните теперь, как это будет реализованно. Например, у меня написана функция. И Ваш компилятор уже всю ее скомпилировал. Тут я добавляю объявление переменной, и где-то в середину действия с новой переменной и с уже там имевшимися. Как Ваш компилятор скомпилирует только добавленный фрагмент, разведет грамотно переменные по регистрам БЕЗ перекомпиляции как минимум всей функции? "БЕЗ перекомпиляции" -- это не мое условие, это Вы написали что один оператор вполне себе сферический в вакууме самодостаточный объект д;ля вашего "параллельного компилятора"(и вообще, параллельные есть, GCC давно умеет распараллеливаться на несколько потоков. Название забито другими).
А если у меня инлайн-функция и я ее изменяю. Тут и компиляции на уровне одной этой функции не хватит. Что, Ваш компилятор будет бегать по всем объектным файлам, раздвигать код и впихивать новые команды туда, куда надо? А как же перетасовка команд при оптимизации? Не помешает?
Таки заново перекомпилировать прощее. И оптимальнее выйдет. Если это вообще не единственный возможный вариант.
> Вы объясните теперь, как это будет реализованно.
ОтветитьУдалитьМожет мне тебе ещё саму реализацию с исходниками на почту скинуть?
На все твои вопросы ответ один: этот компилятор должен делать всё точно так же, как и обычные компиляторы, только параллельно с набором текста.
Нашел ссылку: один дядька описывает эту идею. Также есть обсуждение: http://cboard.cprogramming.com/general-discussions/85755-compile-you-type.html.
ОтветитьУдалить>> этот компилятор должен делать всё точно так же, как и обычные компиляторы, только параллельно с набором текста.
ОтветитьУдалитьВот запустил я компиляцию, открыл Emacs и начал текст набирать. И обычный компилятор(GCC) компилирует параллельно с набором текста. Вот чудеса-то! Вот где Ваша идея-то реализована! Не Вы случайно автор GCC? Ато прям по вашей идее работает!
>> Может мне тебе ещё саму реализацию с исходниками на почту скинуть?
Спасибо, ненадо. Сырцы GCC я найду.
Таки теперь серьезно. Я заметил, что наиболее адекватно ты реагируешь на грубое обращение. Вроде обзывания ламером(и не обхывание это, просто назвал все своим именем). И теперь я буду на "ты". Так лучше.
Так вот, объясни, майкрософтский ламер, какое отношение имеет компилятор к набору текста? Если всю жизнь сидел в таком дерьме, как MSVS и не отличаешь компилятора от, так сказать, "IDE", не значит, что это одно и то же. Это ты в силу своей ограниченности не отличаешь, нормальные люди отличают. И то, что ты выдаешь за свою суперидею, реализуется, как два пальца об асфальт. Просто запускаем GCC на компиляцию изменившегося файла. Если компиляция прошла успешно - то все, файл скомпилен. Если есть ошибки - то показываем их. Навесить просто продвинутую обработку ошибок, и все.
Касательно такой себе «реал-тайм компиляции»… Я себе с трудом представляю, как такую идею можно эффективно реализовать. С++ — это не BASIC и не Python, и компиляция проходит посложнее. IMHO есть много нюансов, когда такому компилятору придется проделывать не меньше работы, чем «обычному», а иногда и больше (нужно ведь выявить все зависимости от вновь добавленной/удаленной конструкции). А вообще, может я ошибаюсь, но Eclipse автоматически компилирует код, когда нажимаешь Save.
ОтветитьУдалитьКасательно оптимизаций в Debug-режиме: некоторые компиляторы делают-таки какие-никакие, но оптимизации. К примеру, тот же MSVC по умолчанию встраивает *не только* inline-функции (даже в дебаге), если посчитает нужным.
По поводу фичи с автоматической проверкой синтаксиса: http://blogs.msdn.com/vcblog/archive/2009/06/01/c-gets-squiggles.aspx
Касательно распараллеливания компиляции: и GCC, и MSVC умеют раскидывать таски на несколько ядер. А если нужно распределенно компилить на нескольких машинах, то есть IncrediBuild и distcc
Андрей, у тебя дурное воспитание. Передай это своему папе, если, конечно, он у тебя есть. Разговаривать здесь с тобой больше никто не будет.
ОтветитьУдалитьGooRoo, понятно, что идея здесь лежит на поверхности и нет труда её понять. Главная задача здесь — эффективная реализация. Но сейчас уже никого не удивишь четырьмя ядрами и 8 мегабайтами кеш-памяти. Так пусть не простаивают, компилируют. :)
ОтветитьУдалитьДа, гипотетически эффективная реализация такого компилятора может и возможна… но я не вижу ни малейшего смысла. Существуют такие огромные проекты (в принципе, это даже не редкость), где настолько много зависимостей, что перекомпиляция после каждого изменения будет занимать слишком много времени.
ОтветитьУдалить