JS++
JS++ is a programming language for web development that extends JavaScript with a sound type system. It includes imperative, object-oriented, functional, and generic programming features. It is free and open-source software released under a BSD license.
History
JS++ first appeared on October 8, 2011. The modern implementation was announced at DeveloperWeek 2016 and released on May 31, 2016. The language is designed by Roger Poon and Anton Rapetov.Features
Sound gradual type system
Since JS++ is a superset of JavaScript, declaring data types for variables is optional. However, when types are declared, the types are enforced at both compile time and runtime.Type annotations in JS++ use the traditional C/C++ syntax:
int x = 1;
var y = 2;
bool z = true;
Notably, this differs from TypeScript and ActionScript, which use a more verbose style:
var x: number = 1;
var y: any = 2;
var z: boolean = true;
The type system in JS++ is sound for ECMAScript and DOM API corner cases, including host objects, dynamic key-value pairs, Comet, JScript conditional compilation, dynamic return types, ActiveX, ECMAScript for XML, web browser garbage collector and cyclic reference counting bugs, conditional logic, and other edge and corner cases. This differs from other JavaScript supersets where types are optional and discarded at runtime via type erasure, such as in TypeScript.
Importing JavaScript libraries
JS++ can use JavaScript libraries using the one-lineexternal statement as in the following example from the homepage of JS++:import System;
// Import JavaScript libraries in one line of code
external jQuery, $;
class Example
Object-oriented programming
While classes in JavaScript are syntactic sugar for prototypes under the hood, JS++ classes resemble the classes found in classical programming languages such as C++, Java, and C# in terms of memory layout, performance, and semantics. "Classes" are a static concept, and they cannot be altered at runtime as is the case for JavaScript, Smalltalk, Lisp, and TypeScript, which rely on prototypes. For example, private methods are private at both compile time and runtime, and external JavaScript objects cannot access private JS++ fields or methods—even if a reference to a JS++ object is obtained from JavaScript.Example: object-oriented sorting
The following source code illustrates object-oriented sorting in JS++ using the and for type-safe and readable comparisons. The custom sorting logic is one line of code in the overriddencompare method below:import System;
class Employee : IComparable
Employee zig = new Employee;
Employee john = new Employee;
Employee abe = new Employee;
Employee employees = ;
employees.sort;
Console.log;
// Output:
// Abe Lincoln, John Smith, Zig Ziglar
Thus, in the code above, the custom sorting logic provided is:
return this.lastName.compare;
Likewise, to call the sort:
employees.sort;
For printing the sorted results:
Console.log;
Example: encapsulation by default
JS++ provides encapsulation by default. In the following example, the fieldsx and y are private by default, even if no access modifier is specified. The methods getX and getY are public by default. This enables a more concise class definition syntax, as illustrated in the Point class below:class Point
Out-of-bounds analysis
An out-of-bounds access usually occurs with arrays and other containers. For example, when we access the 100th element of a 3-element array, we have an out-of-bounds access:int arr = ;
Console.log; // out-of-bounds access, 'arr' does not have an element at index 100
In Java and C#, this can result in an exception and program termination. In C, this can result in buffer overflows or segmentation faults. C++ has varying semantics, such as default initialization, exceptions, segmentation faults, or buffer overflows.
JS++ can efficiently analyze and prevent out-of-bounds errors at compile time.
JavaScript has the notion of
null and undefined values, where null means a value is present but it's an empty value, and undefined means there isn't a value there at all. JS++ extends this intuition further to differentiate between empty values and out-of-bounds accesses.Consider the following code, with a nullable
int type represented with int?:int a = ;
int? value1 = a;
if
int? b = ;
int? value2 = b;
if
While nullable types can represent an out-of-bounds access, it falls apart when the array might contain nullable values as illustrated above. Instead, JS++ introduces an additional concept in addition to null values: undefined values. Recall that JS++ extends the JavaScript notion that null means a value is present but is an empty value, while an undefined value means a value does not exist at all. JS++ uses the concept of "a value does not exist at all" to mean an out-of-bounds access has occurred, and this concept is known in JS++ as "existent types."
Therefore, the previous example can be amended. The existent type
int+ means "int or out of bounds" and int?+ means "int, null, or out of bounds":int a = ;
int? value1 = a;
if
int? b = ;
int?+ value2 = b;
if
Intuitively, this means existent types cannot be used as the underlying type for array elements. JS++ enforces this at compile time:
int+ arr = ; // ERROR
JSPPE5204: Existent type `int+' cannot be used as the element type for arrays
Instead of following every conditional branch or virtual method call path, which would result in path explosion and exponential compile times, existent types have essentially the same compile-time analysis cost as
int, bool, and other primitive types. Consequently, compile times have been shown to be unaffected by the introduction of existent types. Since existent types are used for all array and container types in JS++, JS++ containers are thus guaranteed to not have out-of-bounds errors.In JS++,
undefined is a value that cannot be changed. In JavaScript, undefined is a mutable property of the global object, resulting in circumstances where "undefined" can be "defined." Thus, existent types would not have been possible in pure JavaScript, as arrays can contain elements with the undefined value, undefined can be defined, or other edge and corner cases that are prevented in JS++.Also, in comparison to Java and early object-oriented languages such as Eiffel, JS++ does not default initialize objects to
null. Instead, the compiler enforces initialization by the programmer:class Car
Car car;
JSPPE6000: Variable `car' is not initialized at line 3 char 4
Therefore, since existent types are deeply embedded into the language, JS++ can guarantee that out-of-bounds errors never occur.
Databases
The concept of existent types can be extended outside of containers. For example, in MySQL, columns can be nullable. If the row does not exist for a specified condition, theundefined value can be returned. However, if the row does exist but the value at the column is empty, a null value can be returned instead. This can simplify the code and interfaces to the data access layer.Integer types
JS++ provides 8-bit, 16-bit, 32-bit, and 64-bit integer types as well as floating point types:byte b1 = 0xFF; // unsigned 8-bit
signed byte b2 = -1; // signed 8-bit
short s1 = 1; // signed 16-bit
unsigned short s2 = 1; // unsigned 16-bit
int x1 = 0; // signed 32-bit
unsigned int x2 = 0; // unsigned 32-bit
long z1 = 1; // signed 64-bit
unsigned long z2 = 1; // unsigned 64-bit
float f = 0.5; // single-precision floating point
double d = 0.5; // double-precision floating point
From the project homepage:
// You're not required to declare the type for a variable and you can just
// keep writing regular JavaScript :
var offset = 0;
// But, if you do, this next variable will always remain an 'unsigned int' - even at runtime.
// You'll never see NaN runtime errors again.
// This variable is 'unsigned' so it's also guaranteed to never be negative.
unsigned int pageHeight = $.height;
Block scoping
All variables in JS++ are block-scoped, including the JavaScriptvar statement. Thus, there is no need for two different variable declaration keywords with different scoping rules, such as var and let co-existing simultaneously in JavaScript.Development tools
Compiler
The JS++ compiler is available for Windows, macOS, and Linux. It is a source-to-source compiler which emits JavaScript source code as an intermediate representation.The compiler is developed with C/C++, and the developers claim there are "fewer than 10 open bug reports in the core compiler" after 3.5 years of engineering and 400,000 lines of code.