Returning a “NULL reference” in C++?

前端 未结 8 553
旧巷少年郎
旧巷少年郎 2020-12-05 00:02

In dynamically typed languages like JavaScript or PHP, I often do functions such as:

function getSomething(name) {
    if (content_[name]) return content_[na         


        
相关标签:
8条回答
  • 2020-12-05 00:08

    unlike Java and C# in C++ reference object can't be null.
    so I would advice 2 methods I use in this case.

    1 - instead of reference use a type which have a null such as std::shared_ptr

    2 - get the reference as a out-parameter and return Boolean for success.

    bool SomeClass::getSomething(std::string name, SomeResource& outParam) {
        if (content_.find(name) != content_.end()) 
        {
            outParam = content_[name];
            return true;
        }
        return false;
    }
    
    0 讨论(0)
  • 2020-12-05 00:09

    I can think of a few ways to handle this:

    • As others suggested, use boost::optional
    • Make the object have a state that indicates it is not valid (Yuk!)
    • Use pointer instead of reference
    • Have a special instance of the class that is the null object
    • Throw an exception to indicate failure (not always applicable)
    0 讨论(0)
  • 2020-12-05 00:11

    Yet another option - one that I have used from time to time for when you don't really want a "null" object returned but instead an "empty/invalid" object will do:

    // List of things
    std::vector<some_struct> list_of_things;
    // An emtpy / invalid instance of some_struct
    some_struct empty_struct{"invalid"};
    
    const some_struct &get_thing(int index)
    {
        // If the index is valid then return the ref to the item index'ed
        if (index <= list_of_things.size())
        {
            return list_of_things[index];
        }
    
        // Index is out of range, return a reference to the invalid/empty instance
        return empty_struct; // doesn't exist
    }
    

    Its quite simple and (depending on what you are doing with it at the other end) can avoid the need to do null pointer checks on the other side. For example if you are generating some lists of thing, e.g:

    for (const auto &sub_item : get_thing(2).sub_list())
    {
        // If the returned item from get_thing is the empty one then the sub list will
        // be empty - no need to bother with nullptr checks etc... (in this case)
    }
    
    0 讨论(0)
  • 2020-12-05 00:15

    This code below demonstrates how to return "invalid" references; it is just a different way of using pointers (the conventional method).

    Not recommended that you use this in code that will be used by others, since the expectation is that functions that return references always return valid references.

    #include <iostream>
    #include <cstddef>
    
    #define Nothing(Type) *(Type*)nullptr
    //#define Nothing(Type) *(Type*)0
    
    struct A { int i; };
    struct B
    {
        A a[5];
        B() { for (int i=0;i<5;i++) a[i].i=i+1; }
        A& GetA(int n)
        {
            if ((n>=0)&&(n<5)) return a[n];
            else return Nothing(A);
        }
    };
    
    int main()
    {
        B b;
        for (int i=3;i<7;i++)
        {
            A &ra=b.GetA(i);
            if (!&ra) std::cout << i << ": ra=nothing\n";
            else std::cout << i << ": ra=" << ra.i << "\n";
        }
        return 0;
    }
    

    The macro Nothing(Type) returns a value, in this case that represented by nullptr - you can as well use 0, to which the reference's address is set. This address can now be checked as-if you have been using pointers.

    0 讨论(0)
  • 2020-12-05 00:17

    You cannot do this during references, as they should never be NULL. There are basically three options, one using a pointer, the others using value semantics.

    1. With a pointer (note: this requires that the resource doesn't get destructed while the caller has a pointer to it; also make sure the caller knows it doesn't need to delete the object):

      SomeResource* SomeClass::getSomething(std::string name) {
          std::map<std::string, SomeResource>::iterator it = content_.find(name);
          if (it != content_.end()) 
              return &(*it);  
          return NULL;  
      }
      
    2. Using std::pair with a bool to indicate if the item is valid or not (note: requires that SomeResource has an appropriate default constructor and is not expensive to construct):

      std::pair<SomeResource, bool> SomeClass::getSomething(std::string name) {
          std::map<std::string, SomeResource>::iterator it = content_.find(name);
          if (it != content_.end()) 
              return std::make_pair(*it, true);  
          return std::make_pair(SomeResource(), false);  
      }
      
    3. Using boost::optional:

      boost::optional<SomeResource> SomeClass::getSomething(std::string name) {
          std::map<std::string, SomeResource>::iterator it = content_.find(name);
          if (it != content_.end()) 
              return *it;  
          return boost::optional<SomeResource>();  
      }
      

    If you want value semantics and have the ability to use Boost, I'd recommend option three. The primary advantage of boost::optional over std::pair is that an unitialized boost::optional value doesn't construct the type its encapsulating. This means it works for types that have no default constructor and saves time/memory for types with a non-trivial default constructor.

    I also modified your example so you're not searching the map twice (by reusing the iterator).

    0 讨论(0)
  • 2020-12-05 00:22

    One nice and relatively non-intrusive approach, which avoids the problem if implementing special methods for all types, is that used with boost.optional. It is essentially a template wrapper which allows you to check whether the value held is "valid" or not.

    BTW I think this is well explained in the docs, but beware of boost::optional of bool, this is a construction which is hard to interpret.

    Edit: The question asks about "NULL reference", but the code snippet has a function that returns by value. If that function indeed returned a reference:

    const someResource& getSomething(const std::string& name) const ; // and possibly non-const version
    

    then the function would only make sense if the someResource being referred to had a lifetime at least as long as that of the object returning the reference (otherwise you woul dhave a dangling reference). In this case, it seems perfectly fine to return a pointer:

    const someResource* getSomething(const std::string& name) const; // and possibly non-const version
    

    but you have to make it absolutely clear that the caller does not take ownership of the pointer and should not attempt to delete it.

    0 讨论(0)
提交回复
热议问题