Looking for some cool Ruby tricks?
You found them!
In this article I want to share with you some of my favorites.
Contents
- 1 Deep copy
- 2 Different ways to call a lambda
- 3 Creating a pre-filled array
- 4 True, false and nil are objects
- 5 Lambdas are strict about arguments, but Procs don’t care
- 6 Execute code directly without irb or files
- 7 Your own mini-irb in one command
- 8 Unfreeze an object (danger!)
- 9 Objects with special identity
- 10 Avoid big output in irb or pry
- 11 Using the caller method to get the current call stack
- 12 Bonus! Convert any value into a boolean
- 13 Bonus! Use A Keyword As A Variable Name
- 14 Summary
- 14.1 Related
Deep copy
When you copy
an object that contains other objects, like an Array
, only a reference to these objects is copied.
You can see that in action here:
food = %w( bread milk orange ) food.map(&:object_id) # [35401044, 35401020, 35400996] food.clone.map(&:object_id) # [35401044, 35401020, 35400996]
Using the Marshal
class, which is normally used for serialization, you can create a ‘deep copy’ of an object.
def deep_copy(obj) Marshal.load(Marshal.dump(obj)) end
The results:
deep_copy(food).map(&:object_id) # [42975648, 42975624, 42975612]
Different ways to call a lambda
my_lambda = -> { puts 'Hello' } my_lambda.call my_lambda[] my_lambda.() my_lambda.===
If possible, you should stick with the first one (call
), because it’s the one most people know.
Creating a pre-filled array
The Array class can take an argument + a block, which lets you create an array with n
elements. By default these elements are nil
, but if you have a block, the values will come from it.
Example:
Array.new(10) { rand 300 }
This will generate an array with 10 random numbers which are between 0 and 299.
True, false and nil are objects
true.class # TrueClass false.class # FalseClass nil.class # NilClass
There is only one copy of these objects, and you can’t create more even if you wanted.
This is the singleton pattern in action.
Lambdas are strict about arguments, but Procs don’t care
my_lambda = ->(a, b) { a + b } my_proc = Proc.new { |a, b| a + b } my_lambda.call(2) # ArgumentError: wrong number of arguments (1 for 2) my_proc.call(2) # TypeError: nil can't be coerced into Fixnum
Execute code directly without irb or files
The ruby
command has a number of interesting options you can use.
For example, with the -e
flag you can pass in a snippet of code to be executed.
ruby -e '5.times { puts "Fun with Ruby" }'
You can find more by using the -h
flag.
Your own mini-irb in one command
Ever wanted to know how irb
works? Well, this is a super-simple version of it.
Remember what ‘REPL’ stands for: Read-Eval-Print Loop.
ruby -n -e 'p eval($_)'
You won’t get a prompt, but go ahead and type some Ruby code.
"A" * 5 "AAAAA"
This works because the -n
flag does this:
-n assume 'while gets(); ... end' loop around your script
And $_
is a global variable. Which contains the following:
The last input line of string by gets or readline.
Unfreeze an object (danger!)
There isn’t any Ruby method to unfreeze an object, but using the Fiddle
class you can reach into Ruby internals to make it happen.
require 'fiddle' str = 'water'.freeze str.frozen? # true memory_address = str.object_id * 2 Fiddle::Pointer.new(memory_address)[1] &= ~8 str.frozen? # false
Don’t try this at home!
Objects with special identity
Ruby objects have an identifier or ‘id’ number you can access using the object_id
method. Some objects have a fixed id: Fixnums, true, false & nil.
false.object_id # 0 true.object_id # 2 nil.object_id # 4 1.object_id # 3 2.object_id # 5
Fixnum ids use this formula: (number * 2) + 1.
Bonus: The maximum Fixnum is 1073741823
, after that you get a Bignum object.
Avoid big output in irb or pry
If you are working in irb
and want to avoid filling your screen with the contents of some really big array or string you can just append ;
at the end of your code.
Example:
require 'rest-client' RestClient.get('www.rubyguides.com');
Try again without the ;
to see the difference 🙂
Using the caller method to get the current call stack
Here is a code example:
def foo bar end def bar puts caller end foo
Output:
-:3:in 'foo' -:10:in '<main>'
If you need the current method name you can use __method__
or __callee__
.
Bonus! Convert any value into a boolean
!!(1) # true !!(nil) # false
Bonus! Use A Keyword As A Variable Name
def foo (if: nil) binding.local_variable_get(:if) end foo(if: true)
Summary
I hope you enjoyed these Ruby tricks!
Share them with you friends so they can enjoy them too & subscribe to my blog in the form below so you won’t miss my next post. 🙂