我不喜欢在Ruby中使用抽象类(几乎总是有更好的方法)。如果您确实认为这是针对这种情况的最佳技术,则可以使用以下代码段来更明确地声明哪些方法是抽象的:
module Abstract
def abstract_methods(*args)
args.each do |name|
class_eval(<<-END, __FILE__, __LINE__)
def #{name}(*args)
raise NotImplementedError.new("You must implement #{name}.")
end
END
# important that this END is capitalized, since it marks the end of <<-END
end
end
end
require 'rubygems'
require 'rspec'
describe "abstract methods" do
before(:each) do
@klass = Class.new do
extend Abstract
abstract_methods :foo, :bar
end
end
it "raises NoMethodError" do
proc {
@klass.new.foo
}.should raise_error(NoMethodError)
end
it "can be overridden" do
subclass = Class.new(@klass) do
def foo
:overridden
end
end
subclass.new.foo.should == :overridden
end
end
基本上,您只需abstract_methods
使用抽象方法的列表进行调用,并且当它们被抽象类的实例调用时,NotImplementedError
将引发异常。