Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles / Languages / C#

Understanding (ref)erences !!!

5.00/5 (1 vote)
30 Jul 2010CPOL1 min read 6.2K  
Understanding (ref)erences
Let us take a look at the following piece of code:
C#
public void Operate(IList iList2)
{
 iList2 = new List();
 iList2.Add(1);
 iList2.Add(2);
 iList2.Add(3);
}

public static void Main()
{
 IList iList= new List();
 iList.Add(10);
 Operate(iList);
 Console.WriteLine(iList[0].ToString());
}

Thinking about what the above program would print to the console? And that is what we are going to talk about in this post - simple but subtle.

I saw this code at CodeProject discussions. The author was confused with why the program was printing 10 instead of 1. I am writing about this since the 'gotcha' was not highlighted in the discussion.

So we passed the reference 'iList' to the function which is supposed to make it point to the 'List' that it creates and so must be printing 1. Well, a C++ programmer knowing how to program in C# would have said 'Gotcha' already. A reference (in C#), equivalent to a pointer in C++, is an entity that stores the address of an object in heap and accesses it using this address. So when we pass a reference (by value) to a function, then we are passing this address value. That is captured in another 4 byte variable local to that function; so creating assigning inside the function will make iList2 point to the newly created object - iList and iList are two different references pointing to the same object. So if you want to transmit the effect of the changes you make to the List inside the function, pass it by reference - use ref keyword.

Now the fun part!!! Let us try writing the same stuff in C++:

C++
// This function will not alter the source pointer
void Operator(IList* pList)
{
  pList = new List();
  pList->Add(1);
  pList->Add(2);
  pList->Add(3);
}

// This function will affect the source; similar to using ref in C#
// 1) const IList*& pList - Can make pList point elsewhere 
//    but cannot modify the existing object
// 2) IList* const &pList - pList cannot point to anywhere else 
//    but can modify the existing object
void Operator(IList*& pList)
{
  pList = new List();
  pList->Add(1);
  pList->Add(2);
  pList->Add(3);
}

Hope that was fun !!!

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)