I am working with Java 19. I have tried to use newly introduced virtual threads as follows:

public static void main(String[] args)  {

    System.out.println("Started with virutal threads");
    try (ExecutorService virtualService = Executors.newVirtualThreadPerTaskExecutor()) {
        virtualService.submit(() -> System.out.println("[" + Thread.currentThread().getName() + "] virtual task 1"));
        virtualService.submit(() -> System.out.println("[" + Thread.currentThread().getName() + "] virtual task 2"));
    }
    System.out.println("Finished");
}

The output of this program is:

Started with virutal threads
[] virtual task 2
[] virtual task 1
Finished

Why Thread.currentThread().getName() does not have any name?

Followup question: How to identify virtual threads between eachother? (how to recognize them) So the output would look like

[thread-1] virtual task 2
[thread-0] virtual task 1
2

There are 2 best solutions below

2
On BEST ANSWER

tl;dr

long threadId = Thread.currentThread().threadId() ;  // The thread ID is unique and remains unchanged during its lifetime.

Thread name optional

The name of any thread is entirely optional.

You may name a thread if you so desire. This can be handy while debugging. But you should not expect any preset value.

Virtual threads in particular are specified as having no set name. To quote the Javadoc (my emphasis):

Virtual threads do not have a thread name by default. The getName method returns the empty string if a thread name is not set.

Some frameworks, such as JavaFX/OpenJFX, give their threads an informative name.

Thread ID

If you want to identify a Thread, use its ID, a long.

To quote the Javadoc:

Returns the identifier of this Thread. The thread ID is a positive long number generated when this thread was created. The thread ID is unique and remains unchanged during its lifetime.

4
On

I am still not sure why the thread name is empty but the solution which I found for the moment might be to use ThreadFactory as follows:

    final ThreadFactory factory = Thread.ofVirtual().name("named-thread-", 0).factory();
    try (var virtualService = Executors.newThreadPerTaskExecutor(factory)) {
        virtualService.submit(() -> System.out.println("[" + Thread.currentThread().getName() + "] virtual task 1"));
        virtualService.submit(() -> System.out.println("[" + Thread.currentThread().getName() + "] virtual task 2"));
    } 

In this case the output is:

[named-thread-0] virtual task 1
[named-thread-1] virtual task 2