Class alias in Ruby

20,572

Solution 1

Classes don't have names in Ruby. They are just objects assigned to variables, just like any other object. If you want to refer to a class via a different variable, assign it to a different variable:

Foo = String

Solution 2

in file coupon.rb:

class Coupon 
  #...
end

# add this line of code to make alias for class names
# option1. if you haven't defined a "Ticket" class: 
Ticket = Coupon   

# option2. if Ticket has been defined, you have to redefine it: 
Object.send :remove_const, "Ticket"
const_set "Ticket", Coupon

"Any referrence that begins with an uppercase letter, including the names of classes and modules, is a constant" -- << metaprogramming ruby>>, page38, Constant section

Solution 3

Anyone coming here looking for how to alias a rails model class to have a new name:

I was able to simply do Foo = Bar, but had to put Foo inside it's own model file so that I wouldn't get a Uninitialized Constant Error. e.g.

# models/foo.rb
Foo = Bar

Also you may find weirdness trying to use the alias in associations like has_many, has_one etc. I've found you can usually get around those by using the root namespace (or appropriate namespace depending on how your models are structured) to make sure Rails is trying to autoload the right constant:

has_many :foo, class_name: '::Foo'

Solution 4

You've got to be careful with this, because if your class undergoes any state change (added functions, changed constants, class variables, etc) the state that your class was in when the alias was instantiated will not reflect the updated changes in your class.

In order to avoid carpal tunnel without sacrificing readability, you can store a lambda in your alias object rather than the actual class. Of course, the lambda contains the class but this assures your alias will call up the latest version of your class.

I put this in my supermanpatches.rb rails initializer (inside of config/initializers/) ‡

LAP = lambda { LosAngelesParcel }

Now you can call this using LAP[] and a freshly minted version of your class will be loaded. (Allowing you to create instances, for example, by l = LAP[].new)

runs once when rails is loaded & then is pervasive through your app, callable anywhere kind of like a global variable but 'read-only', so to speak.

Solution 5

I agree with warhog, more or less - but I would subclass ticket from your coupon class - that way if you need to do any data munging, you can put the code in your ticket class

Share:
20,572

Related videos on Youtube

AdamNYC
Author by

AdamNYC

Updated on July 09, 2022

Comments

  • AdamNYC
    AdamNYC almost 2 years

    I am developing a new Rails app based on a similar existing one. In my old app, I have Coupon class, which is very similar to Ticket in my new app. I want to reuse all code in Coupon, but with a new class name.

    Since refactoring is cumbersome in Rails, I wonder if there is a way to create alias for a class in Ruby (similar to alias for attributes and methods).

    • WarHog
      WarHog over 12 years
      What about inheritance? Something like this: class Coupon < Ticket; end
  • p4010
    p4010 over 12 years
    Jörg, thanks for that gem: it is perfectly obvious, still I never thought about it before!
  • AdamNYC
    AdamNYC over 12 years
    Thanks, Jorg. It's totally logical, yet surprising solution. :)
  • Wand Maker
    Wand Maker almost 11 years
    Subclassing means IS-A relationship. I feel unless it is true in the problem domain, we should not do it in software.
  • Hauleth
    Hauleth over 10 years
    It's not variable. It's constant (all identifiers that begins with capital letter are constants).
  • boulder_ruby
    boulder_ruby about 10 years
    This really is not a good option because if String changes, Foo will not...Foo doesn't point to String, it clones it.
  • Jörg W Mittag
    Jörg W Mittag about 10 years
    @boulder_ruby: what are you talking about? Foo = []; Bar = Foo; Foo << 1; p Bar # [1]
  • boulder_ruby
    boulder_ruby about 10 years
    It might just be a rails thing or even superstition on my part. I just ran this test and got similar results. class Demo; def hey; puts "text1"; end; end;; Test = Demo;; class Demo; def hey; puts "text 2"; end; end;;Demo.new.hey #=> "text 2"
  • Moustafa Samir
    Moustafa Samir over 9 years
    This isn't correct 'class A end; B= A; A.class_eval do def foo puts("test"); end end; B.foo' This will print test because both A and B refer to same Object in memory. also B.object_id is equal to A.object_id So you don't have to use lambda
  • Moustafa Samir
    Moustafa Samir over 9 years
    I already updated it using A.class_eval above, is there another way to update it ?
  • Wayne Conrad
    Wayne Conrad almost 9 years
    "If you update your classes' methods you do" -- That's not true for plain old Ruby. Are referring to Rails and its ability to reload changed classes during development?
  • Cyril Duchon-Doris
    Cyril Duchon-Doris over 7 years
    Long live Ruby.
  • RAJ
    RAJ over 5 years
    From rails console, I tried running Foo which thrown error NameError: uninitialized constant Foo. If I run String and then Foo, it works.