Is there a common way to express the usage of arguments in C++? I want to implicitly tell the consumers of my class how the arguments they pass will be used by the class.
Examples:
- I own your argument (will clean it up)
- I will hold a reference to your argument during my lifetime (so you should NOT delete it while I'm stile alive)
- I will use your argument only during construction and won't hold a reference
Is there a common way to express these things simply using the method declaration? I think in the first case a std::auto_ptr would make sense. In the second case I usually take a pointer to avoid someone passing a value from the stack which would invalidate my reference quickly, or alternatively a shared_ptr. In the third case I take a reference to allow values from the stack.
How do you deal with this? Also is it necessary to rely on smart pointers here, or can one express such things simply by using naked references and pointers somehow?
While documentation is the only answer, it is unfortunately not a useful one. The research that I've been doing shows that many clients never read the documentation of methods that they are using.
I would recommend putting it into the nomenclature of your method names or the argument names, since at least that is visible in the autocomplete window. It's ugly, but it gets the message through :)
In my own tool (for Java/Eclipse) I have a tag for that that allows me to announce to the user when he needs to know something important about the parameter.
Our team has similar coding conventions to the ones you suggest:
1 - auto_ptr argument means that the class will take control of memory management for the object. (We don't use this much.)
2 - shared_ptr means that the class will probably use the argument for an extended period of time, and in particular may store off its own shared_ptr to the object.
3 - Plain reference means that the argument will only be used for the duration of the call.
We treat this as a coding standard. It isn't something we document for each and every call.
Maybe I'm missing the point of you question, but if your method prototypes in your header file are setup properly then that will do it.
"Implicitly" the user will know that your given method only accepts a reference to the parameter, or that a given parameter is read-only (const).. etc....
I don't work with code that uses boost or STL, as I don't work with systems that support them particularly well, but I have found the following standards useful...
I use const references and value parameters interchangeably. (I use references if the values are larger than a register -- this is simply an optimisation.) The callee will take a copy by value if it needs it, so the argument can be destroyed straight away. (Case 3.)
I use const pointers to indicate that the input is by-reference, and that the callee will take a copy by reference. The lifespan of the argument must exceed that of the callee. (Case 2.)
I use non-const pointers to indicate that the input is by-reference, and that the callee will modify the pointee if necessary. Lifespan of the argument is deliberately undefined in the general case. (i.e., case 2 or 3.)
I don't use non-const references myself, purely because it's not obvious syntactically at the call point what is going on, but this is simply my personal inclination. I have my reasons, but that doesn't mean anybody else has to agree! So one might usefully use the reference/pointer distinction to indicate lifespan, as I have described in the const case. In that case, maybe pointer = case 2, and reference = case 3.
I indicate case 1 (callee takes ownership) by a comment; this is rare enough in my code that I don't worry about it too much.
If you're looking for clarity, documentation is one good way to deal with this. Of course, that assumes that people pay attention to your docs.
Aside from using existing classes with known usage patterns, you could create your own class. With a class that the compiler could auto-generate from a normal parameter (using operators you would write, of course), you could more accurately specify the contract without making it more difficult to pass the parameters. There are drawbacks to this (extra classes, more layers for the user to see and understand, naming responsibilities), but it is an option to consider.
If you're in an environment where IntelliSense is available, make sure that the necessary information is present in the parameter description. That could save you from any of these other steps.
I don't know if there is a common idiom, but I do know that the one sure-fire way that I provide this information is comments in the interface header. The users of the class won't always read them, won't always remember them, and they'll screw them up quicker than you can blink, but the information WILL be there.
Now, that being said, I've also taken to being against keeping references to something that some other piece of the system owns. It's not always practical (or sensible) to restructure so that your class owns everything (or doesn't keep references after a method call returns), but it's the safest way to do things, and either one is easier for the caller to understand.
The big problem with retaining references is that your callers will never remember that they aren't allowed to destroy these things, and you'll eventually end up with 'use of deleted object' type failures.