There are many situations where you need to express that something is “optional” - an object that might contain a value or not. You have several options to implement such case, but with C++17 there’s probably the most helpful way: std::optional
.
For today I’ve prepared one refactoring case where you can learn how to apply this new C++17 feature.
Intro
Let’s dive into the code quickly.
There’s a function that takes ObjSelection
representing for example current mouse selection. The function scans the selection and finds out the number of animating objects, if there any civil units and if there are any combat units.
The existing code looks like that:
classObjSelection
{
public:
boolIsValid()const{returntrue;}
// more code...
};
boolCheckSelectionVer1(constObjSelection&objList,
bool*pOutAnyCivilUnits,
bool*pOutAnyCombatUnits,
int*pOutNumAnimating);
As you can see above, there are mostly output parameters (in the form of raw pointers), and the function returns true/false
to indicate success (for example the input selection might be invalid).
I’ll skip the implementation for now, but here’s an example code that calls this function:
ObjSelection sel;
bool anyCivilUnits {false};
bool anyCombatUnits {false};
int numAnimating {0};
if(CheckSelectionVer1(sel,&anyCivilUnits,&anyCombatUnits,&numAnimating))
{
// ...
}
Why is this function not perfect?
There might be several things:
- Look at the caller’s code: we have to create all the variables that will hold the outputs. For sure it looks like a code duplication if you call the function is many places.
- Output parameters: Core guidelines suggests not to use them.
- If you have raw pointers you have to check if they are valid.
- What about extending the function? What if you need to add another output param?
Anything else?
How would you refactor this?
Motivated by Core Guidelines and new C++17 features, I plan to use the following refactoring steps:
- Refactor output parameters into a tuple that will be returned.
- Refactor tuple into a separate struct and reduce the tuple to a pair.
- Use
std::optional
to express possible errors.
Let’s start!
Tuple
The first step is to convert the output parameters into a tuple and return it from the function.
According to F.21: To return multiple “out” values, prefer returning a tuple or struct:
A return value is self-documenting as an “output-only” value. Note that C++ does have multiple return values, by convention of using a
tuple
(includingpair
), possibly with the extra convenience oftie
at the call site.
After the change the code might look like this:
std::tuple<bool,bool,bool,int>
CheckSelectionVer2(constObjSelection&objList)
{
if(!objList.IsValid())
return{false,false,false,0};
// local variables:
int numCivilUnits =0;
int numCombat =0;
int numAnimating =0;
// scan...
return{true, numCivilUnits >0, numCombat >0, numAnimating };
}
A bit better… isn’t it?
- No need to check raw pointers
- Code is quite expressive
What’s more on the caller site, you can use Structured Bindings to wrap the returned tuple:
auto[ok, anyCivil, anyCombat, numAnim]=CheckSelectionVer2(sel);
if(ok)
{
// ...
}
Unfortunately, I don’t see this version as the best one. I think that it’s easy to forget the order of outputs from the tuple. There was even an article on that at SimplifyC++: Smelly std::pair and std::tuple.
What’s more, the problem of function extensions is still present. So when you’d like to add another output value, you have to extend this tuple and the caller site.
That’s why I propose another step: a structure (as it’s also suggested by Core Guidelines).
A separate structure
The outputs seem to represent related data. That’s why it’s probably a good idea to wrap them into a struct
called SelectionData
.
structSelectionData
{
bool anyCivilUnits {false};
bool anyCombatUnits {false};
int numAnimating {0};
};
And then you can rewrite the function into:
std::pair<bool,SelectionData>CheckSelectionVer3(constObjSelection&objList)
{
SelectionData out;
if(!objList.IsValid())
return{false, out};
// scan...
return{true, out};
}
And the caller site:
if(auto[ok, selData]=CheckSelectionVer3(sel); ok)
{
// ...
}
I’ve used std::pair
so we still preserve the success flag, it’s not the part of the new struct.
The main advantage that we got here is that the code is the logical structure and extensibility. If you want to add a new parameter then just extend the structure.
But isn’t std::pair<bool, MyType>
not similar to std::optional
?
std::optional
From cppreference - std::optional
:
The class template
std::optional
manages an optional contained value, i.e. a value that may or may not be present.
A common use case foroptional
is the return value of a function that may fail. As opposed to other approaches, such asstd::pair
<T,bool>
,optional
handles expensive-to-construct objects well and is more readable, as the intent is expressed explicitly.
That seems to be the perfect choice for out code. We can remove ok
and rely on the semantics of the optional.
Just for the reference std::optional
was added in C++17 (see my description), but before you could also leverage boost::optional
as they are mostly the same type.
The new version of the code:
std::optional<SelectionData>CheckSelection(constObjSelection&objList)
{
if(!objList.IsValid())
return{};
SelectionData out;
// scan...
return{out};
}
And the caller site:
if(auto ret =CheckSelection(sel); ret.has_value())
{
// access via *ret or even ret->
// ret->numAnimating
}
What are the advantages of the optional version?
- Clean and expressive form
- Efficient: Implementations of optional are not permitted to use additional storage, such as dynamic memory, to allocate its contained value. The contained value shall be allocated in a region of the optional storage suitably aligned for the type T.
- Don’t worry about extra memory allocations.
The `optional` version looks best to me.
Sorry for a little interruption in the flow :)
I've prepared a little bonus if you're interested in C++17, check it out here:
The code
You can play with the code below, compile and experiment:
Wrap up
In this post, you’ve seen how to refactor lots of ugly-looking output parameters to a nicer std::optional
version. The optional wrapper clearly expresses that the computed value might be not present. Also, I’ve shown how to wrap several function parameters into a separate struct. Having one separate type lets you easily extend the code while keeping the logical structure at the same time.
How would you refactor the first version of the code?
Do you return tuples or try to create structs from them?
Here’s some more articles that helped me with this post: