1. Overview

With a cron scheduler, we can automate repetitive tasks we’d otherwise need to handle manually. Additionally, the cron expression allows us to schedule jobs executing at the desired date and time.

For scheduling jobs in Java, we usually use the Quartz library. It’s an open-source solution for job scheduling written entirely in Java. Furthermore, if we’re working with the Spring framework, we can use the @Scheduled annotation to easily schedule tasks.

Although cron expression represents a powerful way to schedule tasks, its syntax can sometimes be confusing and overwhelming.

In this tutorial, we’ll examine the differences between the ? and the * symbols in cron expressions.

2. Fields in Cron Expression

Before we dive in, let’s explore the fields that can appear in the cron expressions.

In Quartz, a cron expression represents a string that involves up to seven fields separated by whitespace, each representing a specific unit of date and time:

Field

Required

Allowed Values

Allowed Special Characters

Seconds

Yes

0-59

, – * /

Minutes

Yes

0-59

, – * /

Hours

Yes

0-23

, – * /

Day of Month

Yes

1-31

, – * / ? L W

Month

Yes

0-11 (or JAN-DEC)

, – * /

Day of Week

Yes

1-7 (or SUN-SAT)

, – * / ? L C #

Year

No

1970-2099 (or empty)

, – * /

As we can see in the table above, all fields are mandatory except the field that specifies a year. If we don’t provide a value, the job will be executed every year.

Additionally, the syntax for the Unix cron expressions is a bit different:

Field

Required

Allowed Values

Allowed Special Characters

Minutes

Yes

0-59

, – * /

Hours

Yes

0-23

, – * /

Day of Month

Yes

1-31

, – * /

Month

Yes

1-12 (or JAN-DEC)

, – * /

Day of Week

Yes

0-6 (or SUN-SAT)

, – * /

The Unix cron expression consists of five fields followed by the command we’d like to execute. Unlike Quartz, there aren’t specific fields where we’d specify seconds and years. It focuses on scheduling tasks for the current year.

It’s worth noting that the cron expression in Unix doesn’t allow the ? symbol to appear in the expression.

In the next sections, we’ll primarily focus on the cron expressions with the Quartz library.

3. The ? in Cron Expression

Next, let’s examine the question mark symbol (?) in the cron expression. Simply put, it represents no specific value.

We can use it only within the fields that specify the day of the month and the day of the week.

However, it’s important to note the day of the month and the day of the week fields are mutually exclusive. In other words, we can’t specify values for both fields in the same expression.

For instance, the following expression results in an error:

0 30 10 1 OCT 2 2023

Additionally, to easily understand the expression, let’s see it in the table:

Seconds

Minutes

Hours

Day of Month

Month

Day of Week

Year

0

30

10

1

OCT

2

2023

We set values for both the day of the month and the day of the week parameters, which isn’t supported with Quartz.

The cron expression would be invalid even if we use the day of the month that falls on the correct weekday:

0 30 10 30 OCT 2 2023

Here, the 30th of October in 2023 falls on a Monday, but the expression is still not valid.

Furthermore, since we’re required to set values for both fields, we need to put the ? symbol on one of them to indicate the value is unset. The field where we set ? will be ignored:

0 0 0 30 OCT ?

From the example, the job runs at midnight on the 30th of October, every year.

Additionally, the ? can appear only once in a cron expression. Setting both values with ? would result in an error as well:

0 30 * ? OCT ?

4. The * in Cron Expression

On the other hand, the asterisk (*) in the cron expression means all the values. To put it differently, we’d use it to set all the values defined for a specific field.

*Furthermore, unlike the ?, we can use * within any field in the cron expression.*

As an example, let’s create a cron expression where we’ll set all the values from the field representing the hours:

0 30 * 1 OCT ?

Next, let’s see in the tabular format:

Seconds

Minutes

Hours

Day of Month

Month

Day of Week

Year

0

30

*

1

OCT

?

empty

The job executes on the first of October, at every hour, 30 minutes and 0 seconds.

Additionally, we can use the * for multiple fields as well:

* * * * OCT ?

This job runs every second, every day in October.

4.1. Day of Month and Day of Week in Linux Cron

When it comes to the day of the month and the weekday fields in Linux cron, they behave differently than the ones from Quartz.

Firstly, they’re not mutually exclusive. We can set both values in the same cron expression.

Secondly, if both fields contain values other than the asterisks, they form a union:

30 10 1 10 5

The job from the example above executes at 10:30 on the first of October and every Friday.

Lastly, if one of the values starts with the asterisk, they form an intersection:

30 10 */1 * 1

Here, the job runs at 10:30 on every day of the month only if it falls on Monday.

5. Comparison Between * and ?

To conclude, let’s list the main differences between the * and the ? special characters in the cron expression:

The * Symbol

The ? Symbol

Stands for all allowed values of a specific field

Means no specific value

Can be used in any field

Can be used only in fields representing the day of the month and the day of the week

Used to specify all the values from the field

Used to set empty value

Can appear multiple times in the same expression

Only one can exist per expression

6. Conclusion

In this article, we learned the differences between the asterisk and the question mark special characters in cron expressions.

To sum up, we’d use the * in the field of a cron expression to include all allowed values for that specific field. On the contrary, the ? represents no specific value and can be used only within the day of month and day of week fields.

Since Quartz doesn’t support implementation for both of those fields, we need to use ? in one of them to leave the field empty.