894

What is the equivalent of this expression in Kotlin?

a ? b : c 

This is not valid code in Kotlin.

7
  • 10
    Official discussion: discuss.kotlinlang.org/t/ternary-operator/2116 Commented Jun 17, 2017 at 14:18
  • 2
    Maybe the showing the assignment would make this more clear. "d = (a) ? b : c" is now "d = if (a) b else c. Same works for when: "d = when { a -> b; c -> x; else ->d }". Personally I prefer the java/c/PERL syntax. Commented Jan 1, 2021 at 21:42
  • Yep, the good designers of kotlin threw away the most elegant construct in java, replacing it with something that's 1) less readable, 2) harder to type (ie more "boilerplate code"), and 3) harder to understand for beginners (conflating expressions with statements). I think they changed the ternary because...it was java, which by their definition has to be bad. Commented Jan 31, 2021 at 14:42
  • Using if-else expression in one single line statement is like ternary operator in Java. Kotlin does not support any ternary operator. Commented Feb 23, 2021 at 11:10
  • I am working with both C# and Kotlin already for years and I am still missing the ? : operator in Kotlin. It's just more elegant, shorter and better to read than if () else construct. That's why this question has so high rating. Commented May 13, 2021 at 9:04

33 Answers 33

1177

In Kotlin, if statements are expressions. So the following code is equivalent:

if (a) b else c 

The distinction between expression and statement is important here. In Java/C#/JavaScript, if forms a statement, meaning that it does not resolve to a value. More concretely, you can't assign it to a variable.

// Valid Kotlin, but invalid Java/C#/JavaScript var v = if (a) b else c 

If you're coming from a language where if is a statement, this might seem unnatural but that feeling should soon subside.

Sign up to request clarification or add additional context in comments.

11 Comments

Additionally you can use when.
just to add, if it's a boolean expression, you can even go with x = a==b
Special mention for var v = a ?: b. This is same as var v = if(a != null) a else b
@AdeelAnsari No, it is not rectifying. It is worse. Compare this. b + if (a) c else d vs. b + (c if (a) else d) The latter one requires additional parentheses. because c is not enclosed by the condition and else.
Here is a little discussion about this topic. discuss.kotlinlang.org/t/ternary-operator/2116/141
|
181

TL;DR

if (a) b else c 

^ is what you can use instead of the ternary operator expression a ? b : c which Kotlin syntax does not allow.


In Kotlin, many control statements, such as if, when, and even try, can be used as expressions. As a result, these statements can have a result which may be assigned to a variable, be returned from a function, etc.

Syntactically, there's no need for ternary operator

As a result of Kotlin's expressions, the language does not really need the ternary operator.

if (a) b else c 

is what you can use instead of the ternary operator expression a ? b : c.

I think the idea is that the former expression is more readable since everybody knows what ifelse does, whereas ? : is rather unclear if you're not familiar with the syntax already.

Nevertheless, I have to admit that I often miss the more convenient ternary operator.


Other Alternatives

when

You might also see when constructs used in Kotlin when conditions are checked. It's also a way to express if-else cascades in an alternative way. The following corresponds to the OTs example.

when(a) { true -> b false -> c } 

Extensions

As many good examples (Kotlin Ternary Conditional Operator) in the other answers show, extensions can also help with solving your use case.

Comments

107

You could define your own Boolean extension function that returns null when the Boolean is false to provide a structure similar to the ternary operator:

infix fun <T> Boolean.then(param: T): T? = if (this) param else null 

This would make an a ? b : c expression translate to a then b ?: c, like so:

println(condition then "yes" ?: "no") 

Update: But to do some more Java-like conditional switch you will need something like that

infix fun <T> Boolean.then(param: () -> T): T? = if (this) param() else null

println(condition then { "yes" } ?: "no") pay attention on the lambda. its content calculation should be postponed until we make sure condition is true

This one looks clumsy, that is why there is high demanded request exist to port Java ternary operator into Kotlin

3 Comments

infix inline fun<T> Boolean.then(param: ()->T):T? = if(this) param() else null
Use <T:Any>, otherwise that will work incorrectly: true then { null } ?: "not-null"
BTW, the ?: operator here is elvis-operator: kotlinlang.org/docs/reference/null-safety.html#elvis-operator
63

Java's equivalent of ternary operator

a ? b : c 

is a simple IF in Kotlin in one line

if(a) b else c 

there is no ternary operator (condition ? then : else), because ordinary if works fine in this role.

https://kotlinlang.org/docs/reference/control-flow.html#if-expression


Special case for Null comparison

you can use the Elvis operator

if ( a != null ) a else b // equivalent to a ?: b 

Comments

54

Some corner cases not mentioned in other answers.

Since appearance of takeIf in Kotlin 1.1 the ternary operator a ? b : c can also be expressed like this:

b.takeIf { a } ?: c 

This becomes even shorter in case c is null:

b.takeIf { a } 

A general workaround to make b lazily-evaluated:

true.takeIf { a }?.let { b } ?: c 

Also note that typical in Java world null checks like value != null ? value : defaultValue translate in ideomatic Kotlin to just value ?: defaultValue.

Similarly, a != null ? b : c can be translated to a?.let { b } ?: c.


See is also Boolean?.ifTrue proposal in KT-6938 which should allow to streamline a ? b : null expressions to a.ifTrue { b } without prematurely evaluating the b.


See also letIf / runIf / alsoIf / applyIf functions which are handy for streamlining complex conditional chaining.

6 Comments

How is b.takeIf { a } ?: c shorter and more readable than if (a) b else c? Terneray operator is certainly a missing feature in Kotlin since variable names and the condition can be long and make you split the line which is bad
It should also be noted that takeIf always evaluates the true-case (here a). Not only may that expression be computed uselessly if a happens to be false, but you can't benefit from smart casts à la if (a is Int) { a + 3 }.
@TheOperator, wrong. { a } is a lazily evaluated lambda.
I wrote it wrong, should be "always evaluates the true-case (here b)". But even { a }, while lazy, must be evaluated to determine the result of the expression.
b.takeIf { a } is technically equivalent to b = a
|
40

There is no ternary operator in kotlin, as the if else block returns the value.

so, you can do: val max = if (a > b) a else b instead of java's max = (a > b) ? b : c

We can also use when construction, it also returns value:

val max = when(a > b) { true -> a false -> b } 

Here is the link for kotlin documentation : Control Flow: if, when, for, while

Comments

38

For myself I use following extension functions:

fun T?.or<T>(default: T): T = if (this == null) default else this fun T?.or<T>(compute: () -> T): T = if (this == null) compute() else this 

First one will return provided default value in case object equals null. Second will evaluate expression provided in lambda in the same case.

Usage:

1) e?.getMessage().or("unknown") 2) obj?.lastMessage?.timestamp.or { Date() } 

Personally for me code above more readable than if construction inlining

4 Comments

It's not that relevant to the question, but why not use ?:, the elvis operator? The first function would be replaced with e.getMessage() ?: "unknown". The second can be expressed as obj?.lastMessage?.timestamp ?: { Date() }()
@hotkey there is no special purpose for that. From my point of view it's looks more consistent and visually less noisy in chain operations as you shouldn't wrap construction in the brackets
@ruX the elvis operator is specifically for this and your use is rather unusual.
While ?: is fine, let's not go too far down the road to Perl.
37

if condition is false then "wrong", otherwise it's some string()

Let's consider the following prototype before writing a ternary conditional operator:

if (!answer.isSuccessful()) { result = "wrong" } else { result = answer.body().string() } return result 

Solutions

You can use ! (logical not) operator in Kotlin's if-expression:

return if (!answer.isSuccessful()) "wrong" else answer.body().string() 

It can be much better if you flip if-expression (expression without ! operator):

return if (answer.isSuccessful()) answer.body().string() else "wrong" 

Kotlin’s Elvis operator ?: can do a job even better:

return answer.body()?.string() ?: "wrong" 

Also, use an Extension function for the corresponding Answer class:

fun Answer.bodyOrNull(): Body? = if (isSuccessful()) body() else null 

In the Extension function you can reduce a code thanks to Elvis operator:

return answer.bodyOrNull()?.string() ?: "wrong" 

Or just use when conditional expression:

when (!answer.isSuccessful()) { parseInt(str) -> result = "wrong" else -> result = answer.body().string() } 

1 Comment

I'm a few tens of hours into my Kotlin journey, and I am surprised how often an extension method is the right answer. Often, I find I can't use the ?. and ?: operators because I'm wanting to pass the Type? to a method elsewhere, rather than invoke a method on it. (eg. pass a String? to DateTimeFormatter. for parsing). In that case, we're back to an if (a != null) DateTimeFormatter.parse(s). But quick extension method 'turns it around' to s?/toSpecialDate()` and almost always helps shove details out of the flow of the logic! I am loving Kotlin :-)
31

In Kotlin, if is an expression, i.e. it returns a value. Therefore there is no ternary operator (condition ? then : else), because ordinary if works fine in this role. manual source from here

// Traditional usage var max = a if (a < b) max = b // With else var max: Int if (a > b) { max = a } else { max = b } // As expression val max = if (a > b) a else b 

Comments

27

Take a look at the docs:

In Kotlin, if is an expression, i.e. it returns a value. Therefore there is no ternary operator (condition ? then : else), because ordinary if works fine in this role.

Comments

24

Another interesting approach would be to use when:

when(a) { true -> b false -> c } 

Can be quite handy in some more complex scenarios. And honestly, it's more readable for me than if ... else ...

1 Comment

Much more expressive, as there are visual hints (->) and each option is on its own line. I usually move condition inside: when { a -> b; else -> c } so it effectively becomes standard if-else
18

Java

int temp = a ? b : c; 

Equivalent to Kotlin:

var temp = if (a) b else c 

Comments

16

Kotlin doesn't have a ternary operator. You can use the regular if expression as shown below:

if (condition) exp1 else exp2

Also in addition to the fact that if in Kotlin is not a statement but an expression (i.e. it evaluates to a value), in the case where you have multiple statements inside the body of an if branch (or the body of else or else if), the last line of the block is the value of that branch. For example:

if (a>b) { println("a is greater than b") a // value of this if } else { println("b is greater than a") b // value of else } 

Comments

15

There is no ternary operator in Kotlin. It seems problematic at the first glance. But think we can do it with inline if else statement because this is expression here. Simply we have to do -

var number = if(n>0) "Positive" else "Negetive" 

Here we can else if block too as many as we need. Like-

var number = if(n>0) "Positive" else if(n<0) "Negative" else "Zero" 

So this line is so simple and much readable than ternary operator. when we use more than one ternary operator in java it seems horrible. But here we have a clear syntax. even we can write it in multiple line too.

Comments

15

You can use var a= if (a) b else c in place of the ternary operator.

Another good concept of kotlin is the Elvis operator. You don't need to check null every time.

val l = b?.length ?: -1 

This will return length if b is not null otherwise it executes the right side statement.

Comments

13

when replaces the switch operator of C-like languages. In the simplest form it looks like this

when (x) { 1 -> print("x == 1") 2 -> print("x == 2") else -> { print("x is neither 1 nor 2") } } 

1 Comment

True, but the example you show has when as a statement, not an expression. A more relevant comparison with ternary conditional expressions would be to have each branch return a value, such that the entire when expression evaluates to a value (as happens with ternary conditionals).
13

as Drew Noakes quoted, kotlin use if statement as expression, so Ternary Conditional Operator is not necessary anymore,

but with the extension function and infix overloading, you could implement that yourself, here is an example

infix fun <T> Boolean.then(value: T?) = TernaryExpression(this, value) class TernaryExpression<out T>(val flag: Boolean, val truly: T?) { infix fun <T> or(falsy: T?) = if (flag) truly else falsy } 

then use it like this

val grade = 90 val clazz = (grade > 80) then "A" or "B" 

4 Comments

Maybe remove <T> better?infix fun or(falsy: T?) = if (flag) truly else falsy
But add <T> can make it work: (grade > 80) then null or "B"
This is really cool, I'm going to use it :P But do note that, unless I'm mistaken, it'll cause an object allocation every single time it's called. Not a huge deal, but worth knowing it's not a zero cost abstraction.
There could be unwanted side-effect because both truly and falsy must be evaluated. Writing (grade > 80) then a() or b() will invoke both a() and b(), but if (grade > 80) a() else b() will invoke either a() or b(), but not both.
11

You can do something like this:

val ans = (exp1 == exp2) then "yes" ?: "no" 

by using this extension:

infix fun<T> Boolean.then(first: T): T? = if (this) first else null 

P.S: Don't modify the above infix function to accept first: T?, the expression will become logically incorrect. Eg: If you modify it to accept nullable first: T?, then val ans = (true == true) then null ?: "abcd", ans will be "abcd", which would not be correct.

1 Comment

Inspired by your answer, I came up with this: pl.kotl.in/tB7xXs6lR. Of course, the performance would be bad as it creates an intermediate object.
9

You can do it many way in Kotlin

  1. Using if

    if(a) b else c 
  2. Using when

    when (a) { true -> print("value b") false -> print("value c") else -> { print("default return in any other case") } } 
  3. Null Safety

    val a = b ?: c 

Comments

8

There is no ternary operation in Kotlin, but there are some fun ways to work around that. As others have pointed out, a direct translation into Kotlin would look like this:

val x = if (condition) result1 else result2 

But, personally, I think that can get a bit cluttered and hard to read. There are some other options built into the library. You can use takeIf {} with an elvis operator:

val x = result1.takeIf { condition } ?: result2 

What is happening there is that the takeIf { } command returns either your result1 or null, and the elvis operator handles the null option. There are some additional options, takeUnless { }, for example:

val x = result1.takeUnless { condition } ?: result2 

The language is clear, you know what that's doing.

If it's a commonly used condition, you could also do something fun like use an inline extension method. Let's assume we want to track a game score as an Int, for example, and we want to always return 0 if a given condition is not met:

inline fun Int.zeroIfFalse(func: () -> Boolean) : Int = if (!func.invoke()) 0 else this 

Ok, that seems ugly. But consider how it looks when it is used:

var score = 0 val twoPointer = 2 val threePointer = 3 score += twoPointer.zeroIfFalse { scoreCondition } score += threePointer.zeroIfFalse { scoreCondition } 

As you can see, Kotlin offers a lot of flexibility in how you choose to express your code. There are countless variations of my examples and probably ways I haven't even discovered yet. I hope this helps!

1 Comment

takeIf is my favorite option indeed, very elegant.
8

Remember Ternary operator and Elvis operator hold separate meanings in Kotlin unlike in many popular languages. Doing expression? value1: value2 would give you bad words by the Kotlin compiler, unlike any other language as there is no ternary operator in Kotlin as mentioned in the official docs. The reason is that the if, when and try-catch statements themselves return values.

So, doing expression? value1: value2 can be replaced by

val max = if (a > b) print("Choose a") else print("Choose b")

The Elvis operator that Kotlin has, works only in the case of nullable variables ex.:

If I do something like value3 = value1 ?: value2 then if value1 is null then value2 would be returned otherwise value1 would be returned.

A more clear understanding can be achieved from these answers.

Comments

6

If you do not what to use the standard notation you can also create/simulate it using infix with something like this:

create a class to hold your target and result:

data class Ternary<T>(val target: T, val result: Boolean) 

create some infix functions to simulate a ternary operation

infix fun <T> Boolean.then(target: T): Ternary<T> { return Ternary(target, this) } infix fun <T> Ternary<T>.or(target: T): T { return if (this.result) this.target else target } 

Then you will be able to use it like this:

val collection: List<Int> = mutableListOf(1, 2, 3, 4) var exampleOne = collection.isEmpty() then "yes" or "no" var exampleTwo = (collection.isNotEmpty() && collection.contains(2)) then "yes" or "no" var exampleThree = collection.contains(1) then "yes" or "no" 

1 Comment

To have it completely equivalent to an actual ternary operator, the target values can also be lambda supplying T
4

You can use the if expression for this in Kotlin. In Kotlin if is an expression with a result value. So in Kotlin, we can write

fun max(a: Int, b: Int) = if (a > b) a else b 

and in Java, we can achieve the same but with larger code

int max(int a, int b) { return a > b ? a : b } 

Comments

2

Why would one use something like this:

when(a) { true -> b false -> b } 

when you can actually use something like this (a is boolean in this case):

when { a -> b else -> b } 

3 Comments

Because the first one is semantically clear & easily understandable to someone else reading it even if they're not familiar w/Kotlin, while the 2nd one is not.
Well, you've got the point, however I can't understand why Kotlin developers didn't introduce ternary expression
I think ? and : contradicts with the nullable/ type declaration rather than a type check. Apart from it I don't see any reason. I think someone would have definitely put some thought, if there is inline if-else condition check. Let's wait and see in future versions.
2

In Kotlin you can use ternary operation like this: val x = if(a) "add b" else "add c"

1 Comment

This question has been answered already enough, and has not been updated recently. There's no need to post now another answer that doesn't differ from earlier ones.
2

Other answers here with e.g. takeIf assume something like x = (a != null) ? a : c or x = (a != null) ? a.method() : c but this is not in fact equivalent to x = a ? b : c

To achieve a true equivalent of the ternary operator with null safety-check (i.e. x = (a != null) ? b : c) you can use let instead of takeIf, i.e.

val x = a?.let {b} ?: c

where b may or may not be a method that invokes something on a.

Comments

1

Another short approach to use

val value : String = "Kotlin" value ?: "" 

Here kotlin itself checks null value and if it is null then it passes empty string value.

Comments

1

There is no ternary operator in Kotlin, the most closed are the below two cases,

  • If else as expression statement

val a = true if(a) print("A is true") else print("A is false")

  • Elvis operator

If the expression to the left of ?: is not null, the elvis operator returns it, otherwise it returns the expression to the right. Note that the right-hand side expression is evaluated only if the left-hand side is null.

 val name = node.getName() ?: throw IllegalArgumentException("name expected") 

Reference docs

Comments

1

When working with apply(), let seems very handy when dealing with ternary operations, as it is more elegant and give you room

val columns: List<String> = ... val band = Band().apply { name = columns[0] album = columns[1] year = columns[2].takeIf { it.isNotEmpty() }?.let { it.toInt() } ?: 0 } 

Comments

0

With the following infix functions I can cover many common use cases pretty much the same way it can be done in Python :

class TestKotlinTernaryConditionalOperator { @Test fun testAndOrInfixFunctions() { Assertions.assertThat(true and "yes" or "no").isEqualTo("yes") Assertions.assertThat(false and "yes" or "no").isEqualTo("no") Assertions.assertThat("A" and "yes" or "no").isEqualTo("yes") Assertions.assertThat("" and "yes" or "no").isEqualTo("no") Assertions.assertThat(1 and "yes" or "no").isEqualTo("yes") Assertions.assertThat(0 and "yes" or "no").isEqualTo("no") Assertions.assertThat(Date() and "yes" or "no").isEqualTo("yes") @Suppress("CAST_NEVER_SUCCEEDS") Assertions.assertThat(null as Date? and "yes" or "no").isEqualTo("no") } } infix fun <E> Boolean?.and(other: E?): E? = if (this == true) other else null infix fun <E> CharSequence?.and(other: E?): E? = if (!(this ?: "").isEmpty()) other else null infix fun <E> Number?.and(other: E?): E? = if (this?.toInt() ?: 0 != 0) other else null infix fun <E> Any?.and(other: E?): E? = if (this != null) other else null infix fun <E> E?.or(other: E?): E? = this ?: other 

Comments

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.