Dimension |
Java & JDK |
Rust *** |
1. Language Goals |
1. Memory Safe and Secure 2. Portability (compile once run anywhere via the JVM) 3. Simple, Object Oriented and Familiar (coming from C/C++) 4. Interpreted, Threaded, and Dynamic
|
1. Memory Safe and Secure 2. Portability (compile for each platform via LLVM) 3. Expressiveness, Efficiency and performance (replace C / C++).
|
2. Licensing |
OracleJDK (stay away from this) License OpenJDK (preferred)
|
The Rust Programming Language and all other official projects are dual-licensed:
|
3. License Cost for commercial projects |
Oracle announced that beginning from Java JDK 17 and onwards Java is free for commercial usage. “If you are running older versions of Java 1-16, you are not affected by the new licensing agreement.”
|
Free
Company donations are appreciated: https://www.rust-lang.org/sponsors
|
4. Security |
||
4.1 Known vulnerabilities |
|
|
4.2 Working groups |
||
5. Memory Management |
Garbage collector.
Possible to suffer from memory-related issues such as memory leaks, garbage collection pauses, or poor performance due to poor memory usage patterns.
May require careful tuning and optimization of the program and the garbage collector to achieve good performance. |
Mostly automatic, enforced by the borrow checker.
The borrow checker is a static analysis tool that ensures safe and correct use of memory by checking that references to data are valid and that data is not borrowed mutably while it is also borrowed immutably.
Developers still need to think carefully about the lifetime of values. |
5.1 Dangling pointers |
// Impossible, java will not garbage collect the underlying object until there are
no more references. |
// Impossible, the borrow checker will keep you in check: |
5.2 Null pointer exceptions |
Null pointers are a big problem in Java: Meta Java nullsafe
The language does allow for Null pointers which can result in Exceptions, use Checker framework to prevent them.
public class App {
error: [assignment] incompatible types in assignment. |
// Impossible, there's no concept of null pointer in
Rust. |
6. Compilation (look at tools for specific tools) |
Generates bytecode.
Bytecode is executed by the Java Virtual Machine (JVM), which is mostly a platform-independent execution environment. |
Generates native machine code that can be run on the target platform.
Uses LLVM as the compiler backend (just like Swift). |
7. Syntax |
||
7.1 Variables mutability |
// Mutable by default, unless final is used. |
// Immutable by default, use mut to make it mutable. |
7.2 Structure |
Code organized into packages.
Classes are templates for objects that define behavior (methods) and state (properties).
Composition is possible via interfaces.
Class inheritance is implemented using extends. |
Code organized into crates (similar to Java packages).
Structs are used mostly to store data (properties). Traits and stand alone functions are used to define behavior.
Composition is preferred and encouraged via traits on structs and enums.
Struct inheritance is not possible, instead, use trait inheritance. |
7.3 Generics |
public class List<T extends Comparable<T>> { |
struct Vec<T: Ord> { |
7.4 Metaprogramming |
Ability to write code that can manipulate other code at runtime:
|
Powerful macro system that allows you to define custom syntax and generate code at compile time.
No built-in reflection, but there are 3rd parties like: https://github.com/dtolnay/reflect
|
8. Concurrency |
||
8.1 Blocking |
||
8.1.1 Deadlock sample
In both languages, what causes the deadlock is that locks are acquired in a different order.
I recommend using non-blocking concurrency techniques to avoid this. |
class Deadlock { |
fn main() { |
8.2 Non blocking |
||
8.2.1 Parallel collections |
Streams API Streams API |
Rayon: Rayon
|
8.2.2 Atomic objects |
The specifications of these methods enable implementations to employ efficient machine-level atomic instructions that are available on contemporary processors. However on some platforms, support may entail some form of internal locking. Thus the methods are not strictly guaranteed to be non-blocking -- a thread may block transiently before performing the operation
|
All atomic types in this module are guaranteed to be lock-free if they’re available. This means they don’t internally acquire a global mutex. Atomic types and operations are not guaranteed to be wait-free.
|
8.2.3 MPSC: Multi-producer single consumer |
I did not find a specialized class to do this but you can use MPMC.
|
fn main() { |
8.2.4 MPMC: Multi-producer multi-consumer |
public static void main(String[] args)
throws InterruptedException { |
fn main() { } |
8.2.5 Actor model |
Akka https://akka.io/
why-we-are-changing-the-license
|
Actix https://github.com/actix/actix
MIT License: LICENSE-MIT
|
8.2.6 async/await |
NONE |
Tokio, async-std, actix |
9. Tooling * |
||
9.1 Compiler |
javac |
rustc (backend: llvm)
|
9.2 Toolchain manager |
sdkman https://sdkman.io/ |
cargo cargo installation
|
9.3 Dependency Manager |
Gradle, Maven (old) |
cargo |
9.4 Security and license audit. |
Dependency-check is a Software Composition Analysis (SCA) tool that attempts to detect publicly disclosed vulnerabilities contained within a project’s dependencies. It does this by determining if there is a Common Platform Enumeration (CPE) identifier for a given dependency. If found, it will generate a report linking to the associated CVE entries.
Maybe blackduck?
|
Cargo-audit: Audit your dependencies for crates with security vulnerabilities reported to the RustSec Advisory Database.
Cargo-deny: cargo plugin that lets you lint your project's dependency graph to ensure all your dependencies conform to your expectations and requirements. (license, vuls)
|
10. Resources needed to run a web server hello world (dev mode) **
This metric is useful to me to understand the resources needed to run an app with no load. |
Spring Boot 3.0.0
Process P1 | P2 |
Total |
Actix-web 4.0 |
All the code is available on Github
* These are my personal preferences.
** Tested with MacBook Pro (16-inch, 2021) M1
*** Assuming that no unsafe rust is used.
Disclosure: Oracle is infamous for breaking their licenses and suing people, so take this information with a grain of salt.