Haxe

Это старая версия этой страницы, сохранённая VasilievVV (обсуждение | вклад) в 06:07, 9 января 2016 (Ссылки: удаление {{JavaScript}} согласно итогу на ВП:КУ с помощью AWB). Она может серьёзно отличаться от текущей версии.

Haxe это высокоуровневый мультиплатформенный язык программирования с открытым исходным кодом, а также компилятор, с помощью которого можно создавать приложения и генерировать исходный код для разных платформ, сохраняя единую кодовую базу.[1][2][3][4]

Haxe
Класс языка Мультипарадигмальный
Появился в 2005
Разработчик Haxe Foundation
Выпуск 3.2.1 (11 октября 2015; 9 лет назад (2015-10-11))
Система типов статическая
Испытал влияние 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]

Для того, чтобы использовать уже существующий код, сообщество открытого ПО создало конверторы исходного кода для:

  • ActionScript 3 в Haxe[27]
  • C# в Haxe[28]

Поддержка платформ

Язык 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:

Другие мультиплатформенные языки:

Примечания

  1. Nicolas' announcement of spelling change on Haxe official mail list.
  2. 1 2 3 4 5 Ponticelli, Franco. Professional haXe and Neko. — Wiley, 2008-02-11. — ISBN 0470122137.
  3. Ivanov, Michael. Away3D 3.6 Cookbook. — Packt Publishing Ltd, 2011-05-24. — ISBN 1849512817.
  4. Doucet, Lars Haxe/OpenFL for home game consoles. Gamasutra (3 июня 2015).
  5. 1 2 3 4 5 6 7 8 9 10 11 12 Introduction to the Haxe Standard Library, Haxe Docs
  6. Target Specific APIs, Introduction to the Haxe Standard Library, Haxe Docs
  7. 1 2 Haxe, iPhone & C++ At Last, GameHaxe website
  8. Dasnois, Benjamin. HaXe 2 Beginner's Guide. — Packt Publishing Ltd, 2011-07-26. — ISBN 1849512574.
  9. Companies using Haxe, Haxe Docs
  10. 1 2 3 Doucet, Lars Dear Adobe: Support Haxe, save your Tools. Gamasutra (24 июня 2014).
  11. Doucet, Lars Flash is dead, long live OpenFL! Gamasutra (18 марта 2014).
  12. 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.
  13. 1 2 3 4 "Haxe Interview". Io Programmo. 2009-04-01: 1—6. {{cite journal}}: Cite journal требует |journal= (справка)
  14. Fisher, Matt. HTML5 for Flash Developers. — Packt Publishing Ltd, 2013-01-01. — ISBN 1849693331.
  15. Server mode command-line : haxe --wait [host:]port
  16. AS3 vs haXe performance, SplashDust website
  17. AS3 Performance Optimization, Starling Wiki
  18. 1 2 Vadim Dyachenko. On “You can’t make good HTML5 games in Haxe”. Yellow After Life (5 декабря 2013).
  19. Kaya, Talha OpenFL & Haxe, A Bumpy Start. Gamasutra (4 июля 2014).
  20. Haxe 3.2 Release. Github.
  21. Haxe mailing list post on naming.
  22. MTASC Compiler, MTASC website
  23. Haxe license page.
  24. Haxe introduction page.
  25. Main Page, FlashDevelop Wiki, «first class support for Flash ActionScript (AS2 and AS3) and Haxe development»
  26. Haxe Support, FlashDevelop Wiki
  27. as3hx, AS3 to Haxe converter, Haxe source code repository
  28. CS2HX — C# to haXe converter, CodePlex project hosting
  29. Blog post mentioning Android port progress.
  30. How to get started with Haxe 2.06 and the webOS PDK [archived on WayBackMachine].
  31. Haxe language reference.
  32. Haxe reference detailing the use of enum.
  33. Language Features. Haxe - The Cross-platform Toolkit. Haxe Foundation. Дата обращения: 30 апреля 2015.
  34. A Scala to JavaScript compiler.

Ссылки