Ruby: How to handle a Failed or Invalid Initialization
What is the ruby best practice for handling a situation in which an object should fail to initialize owing to being passed invalid initialize arguments?
I realize that in ruby, duck typing means that we're not supposed to be overly concerned with what variable/parameters types but rather concern ourselves with how they behave. However, I am working in MacRuby which is bridged over the Cocoa Objective-C API and some of the Cocoa methods expect typed parameters.
For example, I have a ruby class that calls into the Objective-C API and must pass it an object of the NSURL class. It looks somethin开发者_Go百科g like this:
class Alpha
attr_accessor :model
def initialize(hopefully_a_NSURL)
# bridged from Objective-C API
@model=NSManagedObjectModel.alloc.initWithContentsOfURL(hopefully_a_NSURL)
end # initialize
end
... and I would call it like so:
#bridged from Objective-C API
u=NSURL.fileURLWithPath(p)
a=Alpha.new(u)
puts "a=#{a.model}" # => a=#<NSManagedObjectModel:0x2004970e0
>
... which works nicely.
However, if I were to slip up:
a=Alpha.new("Whoops, a string not a NSURL" )
... it explodes messily with errors coming from the depths of the Objective-C API.
I can, of course, put in test that will prevent bad parameter for reaching the bridged objects:
class Alpha
attr_accessor :model
def initialize(hopefully_a_NSURL)
if hopefully_a_NSURL.class==NSURL
@model=NSManagedObjectModel.alloc.initWithContentsOfURL(hopefully_a_NSURL)
end
end # initialize
end
u=NSURL.fileURLWithPath(p)
a=Alpha.new("")
puts "a=#{a}" # => a=#<Alpha:0x200399160>
... but I still get a live instance back. I even tried returning nil from initialize but it seems that ruby insist on always returning a live instance.
Everything I've read says that type checking is heavily frowned upon in ruby but perhaps I will have to make an exception in the case of MacRuby. Would this be a good use of exceptions in ruby or is there a more elegant solution? I'm a noob in ruby so assume I am approaching the problem from the wrong perspective.
I'd try to convert the argument and raise a TypeError
if no conversion was possible:
Raised when encountering an object that is not of the expected type.
[1, 2, 3].first("two")
raises the exception:
TypeError: can't convert String into Integer
The Ruby core and standard libraries do it so there's no reason you can't do it too. The Ruby core will raise exceptions when you do something you're not supposed to (calling an unsupported method, calling a method with the wrong number of arguments, ...) so throwing a TypeError
would make sense. And, if TypeError
isn't quite appropriate, there's always ArgumentError
.
In your specific case, try to convert the argument to an NSURL
by calling to_s
and then instantiating an NSURL
using that string if they don't give you an NSURL
. I don't know my way around MacRuby or the corresponding Mac APIs so I'm sort of guessing on the sensible behavior in this specific case but I think the "convert or raise an exception" idea is sound and sensible.
Of course, you should document the behavior you're going to use in your API documentation too.
精彩评论