The purpose of Abstract Virtual Actors is to provide a very slim pure java implementation of actor based concurrency.
- No dependencies: Only use java util imports, no funny stuff.
- Ultra Slim: Keep all required source to one file, with minimal length for easy analysis. This is so the code is easily auditable and can be included by copying one file into a project
- Typed: Strong typing by design, to allow call heirarchy analysis, no suprise messages,and pure java actors
- Wireable: Usable with standard runtime and pre compiled application contexts. In short spring support out of the box.
- Lightweight use: No requirement to implement complex code to create or use actors.
- Anonymous & FunctionalMessage Actors can be created anonymously from a functionl interface, and typed actors are asked or told from clear functional blocks.
- Native Friendlly: Actors ahould work within a native image compiled from graalvm
- Java Conventional: No need for actor systems, registries or props, actors are instance objects, they can be called and the execution takes place within the actors context.
- Generic: Just as the java collections library demonstrates the ability to sperate the behaviour of collections from their contents, message actors and typed actors are purely generic.
- Performant: Java21 by default (native threads 8 compatible available too) with virtual threads consciously for the simplicity of memorybeing the limit on thread state, almost zero cost concurrency. No need for choices around actor execution environment.
Anything can be made a strongly typed actor:
//wrap a hashSet in an actor
TypedActor<Set<Integer>> intSet = new TypedActor<>(new HashSet<Integer>());
//add some stuff in parallel
IntStream.range(0, 10000)
.parallel()//safe to do to a hash set in an actor
.forEach(i -> intSet.apply(s -> s.add(i)));
//show that we have all our stuff
System.out.println(intSet.ask(s -> s.size()).recieve().get());
Message based actors don't have to be verbose either:
//Create a Message Actor with a lambda
MessageActor<Double> squareRouter = new MessageActor<>(
(msg, sender, self) -> sender.tell(Math.sqrt(msg), self));
//send the message and get the reply
System.out.println(squareRouter.ask(25.0).recieve().get());
This library takes inspriation from previous actor systems, and leans heavily on work already done especially from many years using akka in a commercial context and being inspired by Actr as to what could be done differently. I have borrowed much of the test cases from actr.
There are examples for a few major frameworks that lack any kind of actor like abstraction. Inside the test project are some examples of how it can be used.
It is a one file library, feel free to extend, by design it is intended to be extended and is designed to lend itself to abstraction, the name is a portmanteau of Abstractor and Actor.
There are framework simple examples (micronaut and spring) that can be run in the jvm or compiled to a native executable.