Haxe это высокоуровневый мультиплатформенный язык программирования с открытым исходным кодом, а также компилятор, с помощью которого можно создавать приложения и генерировать исходный код для разных платформ, сохраняя единую кодовую базу.[1][2][3][4]
Haxe | |
---|---|
Класс языка | Мультипарадигмальный |
Появился в | 2005 |
Разработчик | Haxe Foundation |
Выпуск | 3.2.1 (11 октября 2015 ) |
Система типов | статическая |
Испытал влияние | ActionScript, OCaml, Java |
Лицензия | GPL v2, library: MIT |
Сайт | haxe.org |
Платформа | IA-32, x64 |
ОС | Android, Unix-подобная операционная система и Windows |
Haxe включает в себя функциональность, поддерживаемую на всех платформах, например: числовые типы данных, строки, массивы, а также поддержку некоторых файловых форматов (xml, zip).[2][5] Haxe также включает в себя поддержку платформно-специфических API для Adobe Flash, C++, PHP и других языков.[2][6]
Код, написанный на языке Haxe, может быть транслирован в код ActionScript 3, JavaScript, Java, C#, C++, Python, PHP, Apache CGI, а также в приложение Node.js.[2][5][7]
Haxe — это также ActionScript 3-совместимый Adobe Flash компилятор, который может генерировать SWF приложения из Haxe кода.[2][8] Haxe код также транслируется в код языка Neko — язык и виртуальная машина созданный тем же разработчиком.
Основные пользователи Haxe — это TiVo, Prezi, Nickelodeon, Disney, Mattel, Hasbro, Coca Cola, Toyota и BBC.[9][10] OpenFL и Flambe — популярные Haxe фреймворки для создания мультиплатформенного контента и программ из единой кодовой базы.[10] В связи с всё большим вытеснением технологии Adobe Flash в последние годы в пользу HTML5, Haxe, Unity и другие кроссплатформенные инструменты уделяют последнему всё больше времени, сохраняя обратную поддержку с Adobe Flash Player.[10][11]
Архитектура
Единый язык
Самым значимым аспектом разработки архитектуры Haxe было решение о поддержке Adobe Flash, JavaScript и серверных приложений единой кодовой базой.[12][13] В типичных веб-проектах разработчики должны использовать множество разных языков, чтобы построить полноценное веб-приложение.[12][13]
- PHP или другой серверный язык для генерации HTML
- JavaScript для манипуляций над HTML на стороне клиента
- ActionScript для графической части, при использовании Flash
- XML или похожий протокол для коммуникации между компонентами
Haxe был создан с идеей объединения всех этих компонентов единой кодовой базой, а также упрощения взаимодействия между компонентами приложения.[12][13][14]
В книге, за авторством Николаса Кеннеси (основателя проекта Haxe), указываются первоначальные цели создания Haxe:[12]
- Создать язык более мощный, чем ActionScript 2 или ActionScript 3
- Сделать так, чтобы было легко портировать приложения с ActionScript на Haxe
- Сделать язык, поддерживающий разработку для Flash 6, 7, 8 и 9
- Сделать язык, поддерживающий разработку для JavaScript/AJAX
- Сделать язык, поддерживающий разработку серверно-ориентированных программ, вместо PHP или Java
Компилятор
Компилятор Haxe разделен на один фронтенд и множество бэкэндов. Фронтенд отвечает за парсинг и проверку типов, применение макросов, общую оптимизацию, различные трансформации кода и создания промежуточного представления кода в виде абстрактного синтаксического дерева (АСД). Каждый из бэкендов отвечает за трансляцию этого АСД в исходный код или байткод целевой платформы.
Компилятор написан на OCaml. Он может быть запущен в режиме сервера для поддержки автодополнения кода в IDE, также в этом режиме поддерживается кэш для уменьшения времени компиляции.[15]
Производительность
Компилятор Haxe — оптимизирующий компилятор, также использующий подстановку функций, свёртку констант, удаление мёртвого кода (DCE) для оптимизации производительности скомпилированных программ.
Производительность программ, написанных на Haxe, зависит от целевой платформы:
- ActionScript 3: Программы, скомпилированные на Haxe, обычно быстрее, чем программы, скомпилированные с помощью Flex SDK ActionScript Compiler.[16] Однако, используя ActionScript Compiler 2 (ASC2) и грамотно составленный код[17], можно добиться сравнимой производительности.
- JavaScript: Программы, скомпилированные на Haxe, сравнимы по скорости с программами, написанными просто на JavaScript.[18] OpenFl — самый распространенный фреймворк, который можно запустить на HTML5/JavaScript, но приложения, построенные с помощью этого фреймворка, на текущий момент страдают от потерь производительности на мобильных устройствах.[18]
- C++: Программы, скомпилированные на Haxe, работают почти также быстро, как и написанные просто на C++, но приложения, построенные с помощью OpenFL, могут с��радать от потерь производительности.[19]
Разработка
Разработка Haxe началась в октябре 2005[20], а первая бета-версия была выпущена в феврале 2006. Haxe 1.0 был выпущен в апреле 2006 и поддерживал трансляцию в Adobe Flash, Javascript и Neko
Haxe был разработан Николасом Хеннесси (Nicolas Cannasse) и другими авторами, и первоначально был назван haXe, потому что это короткое, простое имя, а также «у него есть X в названии» — атрибут, необходимый для того, чтобы новая технология стала успешной, с юмором отмечал автор языка.[21]
Haxe — это преемник ActionScript 2 компилятора MTASC с открытым исходным кодом, также сделанный Николасом Хеннесси[12][22] и выпущенный под лицензией GNU General Public License версии 2 или выше.[23]
Haxe имеет много общего с ActionScript 3. Компилятор Haxe разрабатывается на языке OCaml, но для того, чтобы писать на Haxe, знаний OCaml не требуется.
Преимущества использования Haxe включают в себя:
- Платформо-независимая разработка
- Разработка на высокоуровневом языке программирования
- Возможность использовать, как целевую платформу, передовые C#.NET и Adobe AIR
- Возможность разработки для устройств, поддерживающих только C++[24]
Исходный код
Рекомендуемая IDE для разработки на Haxe — это FlashDevelop[12], которая поддерживает ActionScript 2, 3 и Haxe, как основные языки с подсветкой синтаксиса, автодополнением кода и другими возможностями.[12][25] Эта IDE также поддерживает сворачивание кода, рефакторинг и интерактивную отладку.[26]
Для того, чтобы использовать уже существующий код, сообщество открытого ПО создало конверторы исходного кода для:
Поддержка платформ
Язык Haxe можно транслировать в байткод различных виртуальных машин, таких как Adobe Flash Player и Neko, а также в исходные коды ActionScript 3, JavaScript, включая экспериментально поддерживаемые C++ и C#. Эта стратегия «компиляции» в различные исходные коды была разработана под вдохновение парадигмы «один раз написать, запускать где угодно». Данная стратегия также позволяет выбирать программисту наилучшую платформу для работы программ.
Генератор кода | Результат | Платформа | Использование | Начиная с какой версии Haxe |
---|---|---|---|---|
AVM1[5] | Байткод | Adobe Flash Player 6+ | Desktop, Browser | 2005 (alpha) |
AVM2[5] | Байткод | Adobe Flash Player 9+, Adobe AIR, Tamarin VM | Desktop, Browser, Server | 2005 (alpha) |
ActionScript 3[5] | Исходный код | Adobe Flash Player 9+ | Server, Desktop | 2007 (1.12) |
C++ (hxcpp)[5] | Исходный код | Windows, Linux, Mac OS X | Server, Desktop, CLI | 2009 (2.04) |
C++ | Исходный код | Android,[29] Apple iOS,[7] Palm webOS[30] | Mobile | 2009 (2.04) |
C#[5] | Исходный код | .NET Framework | Server, Desktop, Mobile | 2012 (2.10) |
Java[5] | Исходный код | Java | Server, Desktop | 2012 (2.10) |
JavaScript[5] | Исходный код | HTML 5, Node.js, PhoneGap | Server, Desktop, Browser, Mobile | 2006 (beta) |
Neko[5] | Байткод | NekoVM | Server, Desktop, CLI | 2005 (alpha) |
PHP[5] | Исходный код | PHP | Server | 2008 (2.0) |
python[5] | Исходный код | python | CLI, Web, Desktop | 2014 (3.2) |
Язык
Haxe — это объектно-ориентированный язык общего назначения, с поддержкой механизма обработки исключений и вывода типов для параметров классов. Также языком и библиотеками поддерживаются обобщённое программирование, рефлексия, итераторы и функциональное программирование.[31] Haxe также, в отличии от многих других языков, одновременно поддерживает и статическую и динамическую типизацию. Компилятор может проверять вывод типов и выдавать ошибки времени компиляции, но также разработчики могут выключить эту проверку, и положиться на динамическую проверку типов целевой платформы.
Язык Haxe похож на ECMAScript, хотя практически любой код на ECMAScript не сможет быть скомпилирован на Haxe без модификации. В отличии от ECMAScript, Haxe компилируемый язык. Haxe был создан под влиянием ActionScript, Java, и OCaml.[13]
Так как Haxe был основан на ActionScript 3, он поддерживает все функции Flash API, хотя и требует лучше оформленный код и более высокие стандарты разработки, нежели компиляторы Adobe.
Hello world
Эта программа напишет «Hello World» после компиляции и запуска:
class Main {
static public function main():Void {
trace("Hello World");
}
}
Проверить этот код можно, сохранив его в файл с именем Main.hx
и запустив компилятор Haxe со следующими параметрами: haxe -main Main --interp
. Эта команда запустит Haxe Compiler в режиме интерпретации кода и выведет на экран терминала Main.hx:3: Hello world
.
Система типов
Haxe — статически типизированный язык. Он имеет богатую систему типов, включая классы, интерфейсы, функциональные типы, анонимные типы, алгебраические типы данных (ADT, называемые «перечислениями» в Haxe), а также абстрактные типы данных. Классы, алгебраические типы данных и функциональные типы поддерживают параметрический полиморфизм, основанный на стирании типов, в других объективно-ориентированных языках часто называемый «Дженериками».
Haxe включает с себя поддержку ограниченного полиморфизма и полиморфизм подтипов.
К тому же, Haxe поддерживает структурную типизацию и номинальную типизацию. Для облегчения работы программистов и без ущерба безопасности типов, Haxe поддерживает вывод типов, который во многих случаях позволяет не писать типы вручную.
Классы
Классы (ключевое слово «class») в Haxe похожи на таковые в Java или AS3. Их полями могут быть методы, статические переменные класса или свойства экземпляра класса. Haxe поддерживает атрибуты доступа «public» и «private», а также более продвинутые методы контроля доступа (ACL, ссылки), которые описываются аннотациями. Методы и статические переменные с постоянным значением могут быть встроены с помощью ключевого слова «inline».
Интерфейсы в Haxe похожи на интерфейсы Java.
interface ICreature {
public var birth:Date;
public var name:String;
public function age():Int;
}
class Fly implements ICreature {
public var birth:Date;
public var name:String;
public function age():Int return Date.now().getFullYear() - birth.getFullYear();
}
Перечисления
Перечисляемые типы — это ключевая особенность языка. Перечисления могут иметь собственные параметры, а также быть рекурсивными.[32] Они похожи на алгебраические типы данных, как таковые в языках вроде ML или Haskell. Строго говоря, это правильные типы-суммы, при условии, что типы-произведения, включенные в них, обязаны быть определены внутри этих типов-сумм. Это значит, что перечисления не просто именованные «магические числа», как в большинстве языков, ими можно элегантно решать сложные архитектурные проблемы:
enum Color {
red;
green;
blue;
rgb( r : Int, g : Int, b : Int );
}
class Colors {
static function toInt ( c : Color ) : Int {
return switch ( c ) {
case red: 0xFF0000;
case green: 0x00FF00;
case blue: 0x0000FF;
case rgb(r, g, b): (r << 16) | (g << 8) | b;
}
}
static function validCalls() {
var redint = toInt(Color.red);
var rgbint = toInt(Color.rgb(100, 100, 100));
}
}
Haxe также поддерживает параметрические перечисляемые типы. Примером могут послужить реализация типов Option, Either и ConsList, причем ConsList ещё и рекурсивный:
enum Option<T> {
Some(v:T);
None;
}
enum Either<T,U> {
Left(v:T);
Right(v:U);
}
enum ConsList<T> {
Nil;
Cons(head:T,tail:ConsList<T>);
}
Документация на сайте указывает[33], что Haxe также поддерживает обобщённые алгебраические типы (GADT), но не приводит пример создания такового.
Анонимные типы
Анонимные типы определяются с помощью явного описания их структуры, им также можно назначить псевдоним, используя определение типа (ключевое слово «typedef»):
typedef Anon = { a:Int, b:String, c:Float->Void };
Функциональные типы
Функциональные типы являются объектами первого класса в Haxe. Они описываются, используя стрелки между типами аргументов, и между типами и возвращаемым значением, как и в многих других функциональных языках. Однако, в отличие от Haskell или семейства ML, не все функции в Haxe унарные (функции с одним аргументом), по умолчанию они не могут быть частично применены. Таким образом сигнатуры типов в следующих примерах имеют отличное от вышеупомянутых языков значение.
Тип F — это функция, которая принимает Int и String как аргументы и возвращает Float как результат.
В языках, где существуют только унарные функции, этот тип означал бы функцию, которая принимает Int как аргумент и возвращает функцию типа String->Float.
Типы F2 и F3 описывают один и тот же тип. Они оба описывают бинарные функции, которые возвращают бинарную функцию типа F. Для F3 описывается случай использования функционального типа внутри другого определения.
typedef F = Int->String->Float;
typedef F2 = Int->String->F;
typedef F3 = Int->String->(Int->String->Float);
Абстрактные типы
Концепция, названная абстрактные типы, — это последнее дополнение в систему типов Haxe. Они позволяют повторно использовать существующие типы для специфических целей, например, реализации типов для единиц измерения, при этом сильно уменьшая возможность смешивания разных систем (например мили и километры). Термин «абстрактный тип» в контексте языка Haxe имеет другое значение, в отличие от обычных абстрактных типов.
Следующий пример предполагает, что метрическая система используется по умолчанию, а конвертация в мили необходима, чтобы поддерживать устаревшие данные. Haxe способен автоматически преобразовывать мили в километры, но не в обратную сторону.
abstract Kilometer(Float) {
public function new(v:Float) this = v;
}
abstract Mile(Float) {
public function new(v:Float) this = v;
@:to public inline function toKilometer():Kilometer return (new Kilometer(this / 0.62137));
}
class Test {
static var km:Kilometer;
static function main(){
var one100Miles = new Mile(100);
km = one100Miles;
trace(km); // 160.935
}
}
Пример показывает, что не обязательно делать явное преобразование km = one100Miles;
для использования корректных единиц.
Структурная типизация
Структурная типизация играет важную роль во многих функциональных языках программирования и, в то же время, довольно малую в распространенных ООП языках. В отличие от номинальной системы типов, равенство двух типов определяется не равенством каких-либо имён типа, а скорее устройством типа. Структурные типы можно рассматривать как неявные интерфейсы:
class FooBar {
public var foo:Int;
public var bar:String;
public function new(){ foo=1; bar="2";}
function anyFooBar(v:{foo:Int,bar:String}) trace(v.foo);
static function test(){
var fb = new FooBar();
fb.anyFooBar(fb);
fb.anyFooBar({foo:123,bar:"456"});
}
}
См. также
Также, на платформе Haxe:
Другие языки, компилирующиеся в JavaScript:
Другие мультиплатформенные языки:
Примечания
- ↑ Nicolas' announcement of spelling change on Haxe official mail list .
- ↑ 1 2 3 4 5 Ponticelli, Franco. Professional haXe and Neko. — Wiley, 2008-02-11. — ISBN 0470122137.
- ↑ Ivanov, Michael. Away3D 3.6 Cookbook. — Packt Publishing Ltd, 2011-05-24. — ISBN 1849512817.
- ↑ Doucet, Lars Haxe/OpenFL for home game consoles . Gamasutra (3 июня 2015).
- ↑ 1 2 3 4 5 6 7 8 9 10 11 12 Introduction to the Haxe Standard Library, Haxe Docs
- ↑ Target Specific APIs, Introduction to the Haxe Standard Library, Haxe Docs
- ↑ 1 2 Haxe, iPhone & C++ At Last, GameHaxe website
- ↑ Dasnois, Benjamin. HaXe 2 Beginner's Guide. — Packt Publishing Ltd, 2011-07-26. — ISBN 1849512574.
- ↑ Companies using Haxe, Haxe Docs
- ↑ 1 2 3 Doucet, Lars Dear Adobe: Support Haxe, save your Tools . Gamasutra (24 июня 2014).
- ↑ Doucet, Lars Flash is dead, long live OpenFL! Gamasutra (18 марта 2014).
- ↑ 1 2 3 4 5 6 7 Grden, John. The Essential Guide to Open Source Flash Development / John Grden, Patrick Mineault, Aral Balkan … [и др.]. — Apress, 2008-07-16. — P. Chapter 9 (Using Haxe). — ISBN 1430209941.
- ↑ 1 2 3 4 "Haxe Interview". Io Programmo. 2009-04-01: 1—6.
{{cite journal}}
: Cite journal требует|journal=
(справка) - ↑ Fisher, Matt. HTML5 for Flash Developers. — Packt Publishing Ltd, 2013-01-01. — ISBN 1849693331.
- ↑ Server mode command-line :
haxe --wait [host:]port
- ↑ AS3 vs haXe performance, SplashDust website
- ↑ AS3 Performance Optimization, Starling Wiki
- ↑ 1 2 Vadim Dyachenko. On “You can’t make good HTML5 games in Haxe” . Yellow After Life (5 декабря 2013).
- ↑ Kaya, Talha OpenFL & Haxe, A Bumpy Start . Gamasutra (4 июля 2014).
- ↑ Haxe 3.2 Release . Github.
- ↑ Haxe mailing list post on naming .
- ↑ MTASC Compiler, MTASC website
- ↑ Haxe license page .
- ↑ Haxe introduction page .
- ↑ Main Page, FlashDevelop Wiki, «first class support for Flash ActionScript (AS2 and AS3) and Haxe development»
- ↑ Haxe Support, FlashDevelop Wiki
- ↑ as3hx, AS3 to Haxe converter, Haxe source code repository
- ↑ CS2HX — C# to haXe converter, CodePlex project hosting
- ↑ Blog post mentioning Android port progress .
- ↑ How to get started with Haxe 2.06 and the webOS PDK [archived on WayBackMachine] .
- ↑ Haxe language reference .
- ↑ Haxe reference detailing the use of enum .
- ↑ Language Features . Haxe - The Cross-platform Toolkit. Haxe Foundation. Дата обращения: 30 апреля 2015.
- ↑ A Scala to JavaScript compiler .