On Writing Exception Safe Code - Part 3

Now let's write exception safe member functions of the Stack class.


std::size_t Count() const;
void Push(const T&);
T Pop(); // if empty, throws exception

Count()

Count() is easiest to implement and also exception safe because builtins never throw.


template <class T>
std::size_t Stack::Count() const {
    return vused_; // safe, builtins don't throw
}

Push()

For Push(), if NewCopy() throws then state is unchanged, and exception is propagated back to the client cleanly.


template <class T>
void Stack::Push( const T& t ) {
 if( vused_ == vsize_ ) { // grow if necessary by some grow factor
     std::size_t vsize_new = vsize_*2+1;
     T* v_new = NewCopy( v_, vsize_, vsize_new );
     delete[] v_; // this can't throw
     v_ = v_new; // take ownership
     vsize_ = vsize_new;
 }
 v_[vused_] = t;
 ++vused_;
}

After any required grow operation, we attempt to copy the new value before incrementing our vused_ count. This way, if the assignment throws, the increment is not performed and our Stack's state is unchanged. If the assignment succeeds, the Stack's state is changed to recognize the presence of the new value.

Guidelines: Observe the canonical exception-safety rules: In each function, take all the code that might emit an exception and do all that work safely off to the side; only then, when you know that the real work has succeeded, should you modify the program state (and clean up) using only nonthrowing operations.

Pop()

Now comes the challenging part of writing Pop() correctly.

Implementing Pop() - 1st Try


template <class T>
T Stack::Pop() {
    if( vused_ == 0) {
        throw "pop from empty stack";
    } else {
         T result = v_[vused_-1];
         --vused_;
         return result;
    }
}

If the initial copy from v_[vused_-1] fails, the exception is propagated and the state of the Stack is unchanged.

Pop() has a flaw. Consider the client code,


std::string s1(s.Pop());
std::string s2;
s2 = s.Pop();

The flaw arises from the fact that the Pop function returns a value by value. If an exception occurs during the construction of the returned copy (e.g., during the copy constructor or copy assignment operator), the state of the stack might be modified, but the popped value is lost.

Implementing Pop() - 2nd Try


template <class T> T Stack<T>::Pop(T &result) {
  if (vused_ == 0) {
      throw "stack is empty!";
  } else {
      result = v_[vused_ - 1];
      --vused_;
  }
}

In the revised implementation, the Pop function doesn't return a value anymore. Instead, it takes an additional parameter result by reference. The popped element is directly assigned to this result parameter. The Stack's state is not changed until the popped element safely arrives in the caller's hand.

But Pop() now has two responsibilities, to pop the top-most element and to return the just-popped value. The breaks cohesion principle.

Guideline: Prefer cohesion. Always endeavor to give each piece of code—each module, each class, each function—a single, well-defined responsibility.

So other preferable option is to separate the functions of "querying the top-most value" and "popping the top-most value off the stack." We do this by having one function for eac

Implementing Pop() - 3rd Try


template <class T> T &Stack<T>::Top() {
    if (vused_ == 0) {
        throw "empty stack";
    } else {
        return v_[vused_ - 1];
    }
}

template <class T> void Stack<T>::Pop() {
    if (vused_ == 0) {
        throw "pop from empty stack";
    } else {
        --vused_;
    }
}

The above separated Top() and Pop() now match the signatures of the top and pop members of the standard library's stack<> adapter. That's no coincidence. The reason why standard library containers' pop functions (for example, list::pop_back, stack::pop, etc.) don't return the popped value is to avoid weakening exception safety.

Common Mistakes: "Exception-unsafe" and "poor design" go hand in hand. If a piece of code isn't exception-safe, that's generally okay and can simply be fixed. But if a piece of code cannot be made exception-safe because of its underlying design, that almost always is a signal of its poor design.