Null references (oh no, not again!)

Ary Borenszweig ary at esperanto.org.ar
Thu Mar 5 04:02:31 PST 2009


Ary Borenszweig wrote:
> Walter Bright escribió:
>> Ary Borenszweig wrote:
>>> Walter Bright escribió:
>>>> Ary Borenszweig wrote:
>>>>> Walter Bright escribió:
>>>>>> Ary Borenszweig wrote:
>>>>>>> It's not like that. They don't require you to initialize a 
>>>>>>> variable in it's initializer, but just before you read it for the 
>>>>>>> fist time. That's very different.
>>>>>>
>>>>>> The only way to do that 100% reliably is to instrument the running 
>>>>>> code.
>>>>>
>>>>> Java does it on compile time.
>>>>
>>>> Java is a severely constrained language. Even so, how does it do 
>>>> with this:
>>>>
>>>> Foo f;
>>>> if (x < 1) f = new Foo(1);
>>>> else if (x >= 1) f = new Foo(2);
>>>> f.member();
>>>
>>> Whenever there are branches in code and a variable still doesn't have 
>>> a value at that point:
>>>  - if all branches assign a value to that variable, from now on the 
>>> variable has a value
>>>  - if not, at then end of the branches the variable still doesn't 
>>> have a value
>>
>> That rule gets the wrong answer in the above case. Consider that in 
>> order to get where you want to go with this, the flow analysis has to 
>> always work, not most of the time work. Otherwise you get bug reports 
>> with phrases like "seems to", "sometimes", "somehow", "I can't figure 
>> it out", "I can't reproduce the problem", etc.
>>
>> Here's another lovely case:
>>
>> Foo f;
>> if (x < 1) f = new Foo();
>> ... lots of code ...
>> if (x < 1) f.member();
>>
>> The code is quite correct and bug-free, but flow analysis will tell 
>> you that f in f.member() is "possibly uninitialized".
> 
> That's exactly the example of a code with a possible bug that'll get an 
> null pointer access. Imagine in "... lots of code ..." you use f and the 
> compiler complains. But you are sure "x < 1" will be true! Then you need 
> to add an else and some assert(false) or throw an exception and that's it.
> 
> If the second condition is "if (x >= 1)" and in the next line you use f, 
> then yes, you are definitely sure that "f" is initialized if you didn't 
> touch "x" in "... lots of code ...". Well... not really, if some other 
> thread changed "x" in the middle of the code, then you are not really 
> sure about that. So again, it's an error.
> 
> If you are really, really sure that this won't happen, you do:
> 
> Foo f = null;
> 
> and that's it. You make the compiler help you only when you want it to, 
> and that'll be most of the time.
> 
> The same goes with:
> 
> Foo f;
> bar(&f);
> 
> If you are sure that bar will assign a value to f if it doesn't have 
> one, go ahead and initialize "f" with null.

By the way, I'm defending this functionality because I received the 
"variable might not be initialized" errors many times now, and I know my 
coworkers also received them many times. In those cases, we were always 
making a mistake. I can't remember even one time when I had to put a 
dummy initializer in a variable just to make the compiler happy.



More information about the Digitalmars-d mailing list