8

Possible Duplicates:
Nullable types and the ternary operator. Why won’t this work?
Conditional operator assignment with nullable<value> types?

This will not compile, stating "Type of conditional expression cannot be determined because there is no implicit conversion between 'System.DateTime' and ''"

task.ActualEndDate = TextBoxActualEndDate.Text != "" ? DateTime.Parse(TextBoxActualEndDate.Text) : null;

This works just fine

 if (TextBoxActualEndDate.Text != "")
    task.ActualEndDate = DateTime.Parse(TextBoxActualEndDate.Text);
else
    task.ActualEndDate = null;
Community
  • 1
  • 1
Daniel Coffman
  • 1,997
  • 3
  • 26
  • 34
  • 2
    Can you get rid of the dependencies on your code, so we can reproduce this? Replace `task.ActualEndDate` with a local variable, for example. – Jay Bazuzi Mar 15 '10 at 22:22

6 Answers6

8

This doesn't work because the compiler will not insert an implicit conversion on both sides at once, and null requires an implicit conversion to become a nullable type.

Instead, you can write

task.ActualEndDate = TextBoxActualEndDate.Text != "" ? 
    DateTime.Parse(TextBoxActualEndDate.Text) : new DateTime?();

This only requires one implicit conversion (DateTime to DateTime?).

Alternatively, you can cast either left side:

task.ActualEndDate = TextBoxActualEndDate.Text != "" ? 
    (DateTime?)DateTime.Parse(TextBoxActualEndDate.Text) : null;

This also requires only one implicit conversion.

SLaks
  • 868,454
  • 176
  • 1,908
  • 1,964
  • You really should use a DateTime.TryParse(TextBoxActualEndDate.Text, out someDateVar) there. Never trust input to give you a parseable string. – Tomas Mar 15 '10 at 22:26
  • 2
    Validation occurs in a couple places prior to this parse, and I would prefer an exception now rather than one when I attempt to insert DateTime.Min into the database. – Daniel Coffman Mar 15 '10 at 22:37
4

The conditional operator doesn't look at what the value is being returned into. It only looks at the values it's being asked to choose between: a DateTime and null. It can't identify these as instances of the same type (because null isn't a valid DateTime), hence the error. You and I know that Nullable<DateTime> could do the job, but the conditional operator isn't allowed to introduce "larger" types: it's only allowed to look at the types of the two expressions it's choosing between. (Thanks to Aaronaught in comments for clarification of this point and a nice clarifying example.)

To work around this, give the operator a hint by casting the DateTime:

TextBoxActualEndDate.Text != "" ? (DateTime?)(DateTime.Parse(TextBoxActualEndDate.Text)) : null;
                                  ^^^^^^^^^^^
Community
  • 1
  • 1
itowlson
  • 73,686
  • 17
  • 161
  • 157
  • 2
    Mostly correct (+1): `DateTime.Parse` returns a `DateTime` (not a `Nullable`), which is a value type and has no conversion to or from `null`. The compiler doesn't have the ability to introduce "larger" types into the equation when it tries to resolve the expression, it can only work with the types that are actually there. It's the same reason you can't write `Stream s = expr ? new MemoryStream() : new FileStream(...)`. – Aaronaught Mar 15 '10 at 22:29
  • Aaronaught: great explanation - I will fold that in. – itowlson Mar 15 '10 at 23:35
1

This is a duplicate of

Nullable types and the ternary operator: why is `? 10 : null` forbidden?

My answer to

Conditional operator cannot cast implicitly?

gives an analysis that is germane to this question.

I'll also be blogging about a similar issue with the conditional operator in April; watch the blog for details.

Community
  • 1
  • 1
Eric Lippert
  • 647,829
  • 179
  • 1,238
  • 2,067
0

The reason is that null is of type object so you have to cast it to the correct type, like this:

task.ActualEndDate = TextBoxActualEndDate.Text != "" ? 
    DateTime.Parse(TextBoxActualEndDate.Text) : ((DateTime?) null);
Keltex
  • 26,220
  • 11
  • 79
  • 111
0

The most correct way (IMO) is to do this

task.ActualEndDate = TextBoxActualEndDate.Text != "" ? 
    (DateTime?)(DateTime.Parse(TextBoxActualEndDate.Text) : null);

I use the null collaescing operator frequently in this manner.

Chris Marisic
  • 32,487
  • 24
  • 164
  • 258
0

This is the error probably which you get in this situation:

error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between '' and 'int')

The compiler is is explaining that it does not know how convert null into a DateTime.


Fix:

you need to cast explicitly the expression which may return null to the nullable type. This will work

((DateTime?) null);
Asad
  • 21,468
  • 17
  • 69
  • 94