Struct vs test double in ruby-Collection of common programming errors

Test doubles are easier to set up

Slip = Struct.new(:id)
slip = Slip.new(:id => 1)

vs.

slip = double('slip', :id => 1)

Test doubles generate more informative error messages

require 'spec_helper'

class TooLongError < StandardError; end

class Boat
  def moor(slip)
    slip.moor!(self)
  rescue TooLongError
    false
  end
end

describe Boat do
  let(:boat) { subject }

  context "when slip won't accept boat" do
    it "can't be moored" do
      Slip = Struct.new('Slip')
      slip = Slip.new
      boat.moor(slip).should be_false
    end
  end
end
Failure/Error: slip.moor!(self)
 NoMethodError:
   undefined method `moor!' for #

vs.

it "can't be moored" do
  slip = double('slip')
  boat.moor(slip).should be_false
end
Failure/Error: slip.moor!(self)
   Double "slip" received unexpected message :moor! with (#)

Test doubles have better support for testing

class Boat
  def moor(slip)
    slip.dont_care
    slip.moor!(self)
  rescue TooLongError
    false
  end
end

it "can't be moored" do
  Slip = Struct.new('Slip')
  slip = Slip.new
  slip.should_receive(:moor!).and_raise(TooLongError)
  boat.moor(slip).should be_false
end
Failure/Error: slip.dont_care
 NoMethodError:
   undefined method `dont_care' for #

vs.

it "can't be moored" do
  slip = double('slip').as_null_object
  slip.should_receive(:moor!).and_raise(TooLongError)
  boat.moor(slip).should be_false
end

0 failures # passed!

That’s a few examples — I’m sure there are more reasons to prefer test doubles.