Язык программирования D
Доброго всем времени суток!
Возникло желание (и даже возможность!) рассказать сообществу о замечательном языке программирования D. Как ни печально, в сети по нему все еще минимум информации, а на хабре и того меньше.
Все скорее всего о нем даже знают, но мало кто пытался писать на нем проекты, а все из за того, что он, видите ли, «не выстрелил», что у него нет будущего. Основная задумка, с которой я пишу эту статью — развеять некоторые заблуждения по поводу этого, несомненно, замечательного языка программирования, ну и заодно, конечно, просто отдать в широкие массы еще немного информации о нем.
О чем же пойдет речь?
Да, первым делом я не хочу писать «почтовый сервер в 200 строк кода», как делают любители хаскелла. Это, несомненно, демонстрирует мощь языка, но не это цель данной статьи.
А буду я следовать задумке авторов языка, и косвенно пропагандировать его, как замену (или аналог) C++ — как язык, в котором если возможности для всего, и как достойный конкурент других современных языков.
Все мы знаем основные стили программирования: процедурный, обьектно ориентированный, функциональный. При чем обьектно ориентированный может быть как естественный для статических языков классовый, так и принятый в динамически типизированных языках прототипный.
Для всезнаек: не ругайтесь, я, конечно, перечислил не все, а основные направления современного прикладного программирования.
Первым делом я хотел бы показать, что язык D поддерживает все эти стили, при чем совершенно естественным образом (рискуя нарваться на неодобрение: гораздо более естественно, чем C++), что этот язык буквально — для всех.*
Для начала немного общих сведений:
D — компилируемый, статически и строго типизированный язык с си-подобным синтаксисом. Позиционируется, как современный язык для системного программирования. В языке присутствует сборка мусора, что, в принципе, не мешает спокойно заниматься ей в ручную, работая напрямую с кучей си — через всем известные malloc и free, хотя это и не рекомендуется.
Ничего кардинально нового (ну почти, но не будем вдаваться) в языке нет, главное его преимущество — удобство для всех.** Что это значит? Это значит, что язык комбинирует мощь и выразительность (а соответственно и так же уменьшает время разработки) языков типа Java и C#, но не теряет прямой связи с системой на уровне C ( и скорость скомпилированного кода!).
Это значит, что почти каждый программист, привыкший писать приложения на языке %langname% сможет легко и непринужденно приспособиться к D, даже не сменив свой стиль программирования!***
Ну, это все реклама, перейдем к конкретике.
Итак, я обещал показать, что в языке можно писать (в разумных пределах) используя любой стиль.
Что ж, начнем с простого: процедурный.
Ну, тут вряд ли можно придумать что-то оригинальное, так что вот не очень осмысленная, но все же функция:
Приятным добавлением по сравнению с C++ являются вложенные функции (в этом месте программисты на паскале торжествуют):
int incTwice(int a) < int inc(int a) < return a+1; >return inc(inc(a)); >
Очень удобно для инкапсуляции, не так ли?
Это, наверное, все, что можно сказать о процедурном стиле, перейдем дальше — к более интересному.
Функциональный стиль: во-первых характеризуется тем, что функции — обьекты первого рода:
Для любопытных, вот обьявления без автоматического вывода типа:
int function(int) inc = function (int a) ;
Более того, в D можно создавать делегаты — функции, поддерживающие замыкания.
int c = 1; int delegate(int) inc = delegate (int a) ;
Функциями и делегатами легко манипулировать, они очень прозрачны и код не зашумлен из за вывода типов.
В принципе, по моим ощущениям D не менее функциональный, чем Python, хоть и не дотянет до haskell-а.
В добавок упомяну еще одну функциональную «фишку»: pure functions. Посмотрим:
Это значит, что функция чистая, то есть не содержит и не изменяет состояния, как своего, так и системы в целом и для одинаковых аргументов даст одинаковые значения.
Такие функции можно выполнять, например, параллельно, их результат можно кэшировать, то есть компилятор получает огромное поле для оптимизации.
Отмечу, что он так же следит за корректностью. Не получится сделать вот так:
int c = 1; pure int inc(int a) < return a+c; // ошибка, программист не прочитал определение чистой функции. >
Теперь перейдем к обьектно ориентированному стилю.
Во-первых, ни для кого не будет сюрпризом возможность создавать классы и инстанцировать обьекты этих классов:
Даже возможность их наследовать никого не удивит!
А раз не удивит — значит, изначальный посыл можно считать подтвержденным — программистам на ООП-языках будет легко приспособиться.
Добавлю, что в языке нет множественного наследования классов, только множественное наследование интерфейсов.
Так же в языке есть хитрость: множественное наследование классов можно симулировать, не нарушая древовидную структуру графа родственных связей классов. Для тех, кто заинтересовался, не буду раскрывать интригу — читайте книгу Александреску «The D programming language».****
Это очень забавная и, в то же время, удобная особенность.
Что я еще обещал? Ах да, прототипное наследование.
Все, кто пишет на языках типа C++, не привыкли видеть такие вещи и писать в своем коде. Казалось бы, это противоестественно для компилируемых языков из за статической типизации.
Однако, тут такое тоже возможно (простите уж за много строк, но сами знаете, что ООП-код получается немного жирным):
class A < string s; this() < s = ""; >// да, так называются конструкторы, по-моему удобней, чем по названию класса. this(string s) < this.s = s; >override string toString() // перегрузим метод корневого класса Object < return s; >> A createA(string s) < class B : A // точно, вложенные в функцию классы! < this() < s = ""; >this(string s) // конструкторы не наследуются. < this.s = s; >override string toString() < return "trollface"; >> return new B(s); > void main() < A a = new A("hi!"); A b = createA("hi!"); writeln(a, " : ", b); // ну да, выведет "hi! : trollface" readln(); >
Более того, для ленивых и любителей прекрасного есть анонимные классы, перепишем кусок кода:
A createA(string s) < return new class A // это значит, что наследуется от A < this() < s = ""; >this(string s) override string toString() < return "trollface"; >>; >
Итак, все анонсированные концепции программирования поддерживаются языком D, и при том вполне лаконично.
Конечно, этого не достаточно, чтобы язык стал хорошим. Чтобы он стал по-настоящему мощным и удобным, необходимо красивое и элегантное взаимодействие всех этих концепций.
Было бы неплохо, например, чтобы программист на javascript и программист на C++ могли писать один проект на D вдвоем, писать так, как привыкли в своей области, или, хотя бы, приближаться к этому, и при этом так, чтобы не терялась ни читабельность ни эффективность, и что еще важнее, чтобы оба стиля могли органично и без костылей взаимодействовать друг с другом.
Но это уже тема для другой статьи.
Я сам только изучаю этот замечательный язык, так что:
Во-первых, если тут найдутся гуру, которые найдут ошибки — с радостью выслушаю указания.
Во-вторых, я не собираюсь останавливаться на достигнутом, и, если сообщество заинтересуется, с удовольствием буду продолжать писать статьи о D по мере его изучения.
В-третьих, если возникнут вопросы, то с радостью готов ответить, обсудить и подискутировать, только без фанатизма.
И в заключение надеюсь, что моя статья поможет вам осознать, что это не язык «не взлетел», это люди закостенели, или же боятся нового, и в любом случае, язык стоит того, чтобы на него перейти с многих других, или, хотя бы, рассмотреть его как достойную альтернативу.
* для многих.
** да-да, поймали — опять для многих.
*** И код при этом не станет откровенным говнокодом — каким он становится при переходе, например, от C++ к haskell (это я о себе :), а вполне вписывается в общий стиль языка.
**** А вы действительно думали, что название статьи я сам придумал?
P.S.
Как мне намекнули в комментариях, в Python все-таки есть замыкания. Видимо, давно я его пробовал, либо что-то изменилось, либо я просто забывчив. Приношу свои извинения.
Язык программирования D: история и компиляторы
Языки программирования традиционно разделяют на скриптовые, системные и «языки продуктивности». Конек скриптовых языков (Perl, PHP, Python и т. д.) — очень быстрая разработка и отладка без затрат времени на компиляцию, но ресурсоемкость и медлительность интерпретаторов четко очерчивают область их применения. Среди компилируемых системных языков доминируют С и С++. Они универсальны, но разработка на них идет медленно из-за сложного синтаксиса и долгих циклов компиляции и отладки. «Языками продуктивности» традиционно считают Java и C#. По производительности они занимают промежуточное положение и прекрасно подходят ,чтобы «склеить» различные готовые компоненты и библиотеки в единую систему, что и обеспечивает им ведущую роль в современном коммерческом программировании.
В последнее время технологии создания компиляторов достигли такого уровня, когда появилась реальная возможность объединить скорость и универсальность системных языков с удобством «языков продуктивности» и, в определенной мере, даже с гибкостью скриптовых языков. Это породило «новую волну» языков программирования, основными представителями которой являются D, Vala, Genie и Go. Все они — компилируемые строго типизированные языки, содержащие конструкции и парадигмы «сверхвысокого уровня», ранее доступные только в «продуктивных» и скриптовых языках. Все эти языки рассматриваются как инструменты высокопродуктивного кроссплатформенного программирования и все изначально свободны или имеют свободные реализации. D и Go претендуют также на то, чтобы сломать устоявшиеся (и уже порядком устаревшие) подходы к многопоточному программированию. Сложно сказать, потеснят ли представители этой «новой волны» сегодняшних лидеров, но познакомиться с ними однозначно стоит — возможно, за ними будущее.
Среди компилируемых языков высокого уровня альтернативы С++ долгое время просто не было. Однако любой программист на С++ скажет, что этот язык на самом деле ужасен. Его синтаксис очень сложен и нелогичен, макросы и директивы препроцессора устарели десятки лет назад, шаблоны используются как «тайная шаманская технология» и т.п. Это хорошо понимал и Уолтер Брайт (Walter Bright) из небольшой компании Digital Mars. В 1999 г. он решил создать «идеальный С++». Новый язык был назван D, как следующая логическая ступень развития С++. Очень скоро D перерос идею «улучшенного С++» и превратился в своего рода концепцию «идеального системного языка», вобравшего все лучшее из С++, С#, Java, Python и Eiffel. D вырос из прагматического осмысления обширного практического опыта, а не из теоретических предпосылок. Его базовый синтаксис очень похож на С++, но все неудачные элементы дизайна этого языка тщательно «вычищены», а из других языков привнесены многие современные идеи и конструкции. D предельно логичен, в нем нет никаких «мистических странностей», которыми так богат С++. Обратная совместимость с С сохранена на уровне типов данных, а не синтаксиса. Это позволяет не тянуть в новый язык весь устаревший багаж языка С (вроде препроцессора), но позволяет напрямую вызывать библиотеки, написанные на С.
Главная страница языка D не блещет дизайном, но содержит много полезной информации
В 2007 г. в развитии D случился перелом. Текущая версия языка была переведена в «режим поддержки» и стала называться D1, а весь накопленный опыт и новые идеи — перенесены в несовместимую версию D2. D2 стабилизировался только в 2010 г. и по праву может претендовать на роль самого молодого и самого совершенного системного языка. В D2 есть практически все современные наработки в области языков программирования:
— Модули и их встроенное тестирование.
— Ручное и автоматическое управление памятью.
— Динамические массивы со срезами и векторизацией.
— Ассоциативные массивы.
— Анонимные функции и замыкания.
— Шаблоны и «подмешивания».
— Выполнение кода на этапе компиляции.
— Встроенный ассемблер.
— Контрактное программирование.
— Конкурентное программирование и т. д.
Все это входит в базовый язык, поэтому D получился «большим» — в нем намного больше конструкций и ключевых слов, чем в С++, но многое их того, что в С++ делается только с помощью STL и boost, в D доступно сразу.
Простой и логичный синтаксис языка облегчает жизнь не только человеку, но и лексическому анализатору. Компиляторы D работают на порядок быстрее компиляторов С++, обеспечивая сопоставимую производительность. И D1, и D2 полностью совместимы с С на уровне типов данных и бинарного интерфейса (ABI), позволяют напрямую компоновать программы с библиотеками, написанными на С, без потерь производительности (нужен только модуль-обертка для заголовочных файлов С). D2 также «понимает» модель вызова extern в С++, т. е. может компоноваться с теми объектами С++, которые могут вызываться из С. В остальном (например, на уровне классов и шаблонов) D и С++ принципиально несовместимы.
Итак, с изучением языка программирования D мы на сегодня закончили и теперь можно приступить к делам более насущным! Для начала купить детские ветровки (http://www.kakosha.ru/verxnyaya-odejda/vetrovki/) для ваших малышей! Для этого Вам следует посетить сайт www.kakosha.ru, где продаются отличные и недорогие вещи для детей!