If you’re a Web developer who’s been curious about Ruby on Rails but has never gotten around to trying it out because you couldn’t find a suitable overview of its advantages, then this article is for you.
We want to bring Ruby on Rails closer to those who want to take a peek first, without going through an entire tutorial. So, this article is structured a little different from most other introductions out there; hopefully it is more useful because of this.
I assume you’re already familiar with some other form of Web development, whether PHP, Python, Perl or Java, and relational databases like MySQL. First, we’ll introduce Rails and Ruby and the basic ideas behind both. I’ll teach you just enough Ruby so that you understand the code samples. I’ll tell you how to get Ruby on Rails running on your computer, and I’ll give you an overview of the basic functionality of Rails and demonstrate how Rails’ main parts work together.
This tutorial consists of two articles: in the current, first article we get started with some basic concepts and essential components of Ruby on Rails. In the second part (it will be published next week) you will learn how to install the engine; you’ll also take a closer look at Rails’ inner workings and discover main advantages of Ruby on Rails. Please stay tuned.
After reading these parts, you should have an idea of whether Rails is for you. If you get the feeling that it is, I’ll point you to some good tutorials on the Web that you can use to learn Rails. I’ll also provide a lot of further reading recommendations so you can dig as deep into the topic as you like.
I’m taking this approach because Rails is almost 5 years old now and has become very complex. There are a lot of “Create-your-own-blog-in-5-minutes”-type tutorials out there already, and rather than adding another one, I wanted to provide this kind of rough overview to help you decide whether to take this adventure.
You may want to take a look at the following related posts:
The Idea Behind Rails
Ruby on Rails was created by David Heinemeier Hansson as a kind of byproduct of Basecamp’s development at 37signals in 2004. Basecamp was built in Ruby because Hansson found PHP and Java not powerful or flexible enough. It was quite an obscure language back then, without the large eco-system available today. To make development easier, Hansson rolled his own Web development framework, based on simple ideas that had proven successful elsewhere. Rails is founded on pragmatism and established paradigms instead of exotic new ideas. And that’s what made it so successful.
Rails is based on the Model-View-Controller pattern that splits your application into three parts:
- The Models are your business objects describing the structure and behavior of the problem your application is trying to solve. These are usually backed by an Object-Relational-Mapping framework that persists your objects to a database in the background.
- The Views are the templates that render data to the user and all the logic surrounding presentational aspects of your app.
- The Controller sits at the heart of everything, processing requests from clients, initiating changes in the models and triggering the rendering of the templates.
Rails is “opinionated software.” It doesn’t want to be everything for everyone. It focuses on one way of doing things and streamlines all its parts around that way. That’s not to say there’s no possibility of doing things differently if you need to, but you’ll definitely have it easier if you do things “the Rails way.” And that way happened to be exactly the right one not only for Hansson but for a lot of other developers, too, another important reason for Rails’ success.
Programmer productivity was the main goal during Rails’ development, not performance. This has led to a lot of controversy and claims that arise over and over about how Rails can’t scale. This is Rails’ own fault to a certain degree. In its early days, it had the image of a Web development framework messiah of hope and wonder that would lead us all to the promised land were applications wrote themselves. The Rails team didn’t do enough to keep expectations more realistic, and some people became disappointed.
While it’s true that Ruby on Rails is slower than comparable stacks on PHP or Python, it certainly does scale, as hundreds of successful deployments are proving. You’ll just need to scale sooner and put some thought into it. Remember also that Ruby’s current default implementation is terribly inefficient, but alternatives are on the way. There’s nothing inherently slow about the language, though, as blazing-fast implementations of Smalltalk (a language very similar to Ruby) prove. Ruby will only get faster. As the saying goes, you don’t have a performance problem until you have a performance problem, and all this talk should not scare you yet. You haven’t even started. ;)
Now, before I introduce you to the framework, let’s get started with Ruby.
A Gem From Japan
Ruby on Rails owes not only half its name but its entire feel and flexibility to “Ruby,” that neat little language from Japan.
Ruby came out in 1995 and was developed by Yukihiro Matsumoto, or “Matz” as he’s called in the community. Version 1.0 was released in 1999 and slowly gained recognition in the west from then on.
A key point in the spread of Ruby was the release of “Programming Ruby,” also called the “Pickaxe” (a reference to its cover illustration), by the Pragmatic Programmers. “Programming Ruby” was the first comprehensive English guide to the language and API.
Ruby was designed with simple principles in mind. Matz took the most successful and powerful elements from his favorite programming languages – Perl, Smalltak and Lisp – and combined them into one language with easy syntax. One goal was to make Ruby feel “natural, not simple” and to create a language “that was more powerful than Perl, and more object-oriented than Python.” This results in Ruby’s core principle: Everything is an object.
Objects
Let’s stop here and examine this. Really, everything is an object in Ruby.True
and false
are objects, literals are objects, classes are objects. You can call a method on a numeric literal:
>> 5.next
=> 6
Operators in Ruby are nothing but methods:
>> 5 * 10
=> 50
>> 5.*(10) # times-operator called as a method (dot-notation)
=> 50 # with a parameter (in parentheses)
Ruby is extremely flexible and open. Almost everything about it can be changed or manipulated at runtime:
- You can add and remove methods and variables to and from objects.
- You can add and remove methods and variables to and from classes.
- You can truly manipulate any class this way, even core classes like
String
andInteger
!
>> “hi”.repeat(4)
NoMethodError: undefined method `repeat’ for “hi”:String
>> class String # Open the string class and add the method
>> def repeat(i)
>> self * i
>> end
>> end
=> nil
>> “hi”.repeat(4) # Call it again on a fresh String literal
=> “hihihihi” # And there it is!
Here, I defined the method repeat
on the String core class, and it was immediately available on a string literal.
And he who giveth, taketh away:
>> class String # Open up the method again
>> undef_method :repeat # And remove the method
>> end
=> String
>> “hi”.repeat(4) # Try to call it
NoMethodError: undefined method `repeat’ for “hi”:String
I could have also done this with predefined methods. They are no more “special” than the methods we have defined.
Let’s review the definition of repeat
in the above example for some more interesting tidbits. Note that we’re not saying return
anywhere in the body. That is because in Ruby, methods always implicitly return the value of their last expression. You could of course always jump out of a method by using return
before reaching the last statement, but you don’t have to. The expression we’re returning is self * i
. Self
is equal to this
in Java and $this
in PHP and always refers to the current object. The times-operator on a string repeats the string as often as told by the second operand/parameter, i
in this case.
Loops
You rarely see manual iterations in Ruby, likefor
or while
loops. Instead, Collections come with their own iterators that you can pass blocks to, which are executed for every element in the collection:
a = “Hey “
[1, 2, 3].each do |num|
puts a * num
end
Outputs:
Hey
Hey Hey
Hey Hey Hey
What you see here is an array literal containing numbers. On that array, the each
method is called, an iterator that takes a block and calls the block for every element in the array. The block starts with the do
, followed by a list of its parameters enclosed in pipe symbols. Here we have one parameter called num
that will take on the value of the array element in each iteration. Inside the block, we’re simply outputting the result of a * num. The definition of *
on Strings is to repeat the string accordingly. We could have put the String inside the Block, but I wanted to demonstrate that blocks have access to their surrounding scope.
Syntax
Ruby likes to keep the syntax clean and friendly. You can see this in the above examples. Although heavily influenced by Perl, Ruby doesn’t have Perl’s excessive use of special characters. You can use semicolons to end lines, but you don’t have to (and no Ruby programmer does). You don’t need to surround method parameters with braces in unambiguous situations (although it is recommended you do so if they enhance readability), and you especially don’t need to provide empty braces around an empty parameter list. That’s what makes accessors look so much like native properties.Blocks are framed by do
and end
. You should only use equivalent curly braces if your blocks don’t span several lines. The only significant use of special characters is found at variable declaration. Variables in Ruby are prefixed with special characters to indicate their scope. Variables starting with a lowercase letter are local variables. Variables starting with an uppercase letter are constants. (This means that all classes are constants, too, since classes start with uppercase letters.) Instance variables start with an @
. Class variables that are shared among all instances of a class start with @@
. Finally, global variables all start with a $
.
You’ll often find methods ending in ?
or !
. These are not special characters. It is merely conventional in Ruby to use question marks for methods that query an object for a Boolean condition, like Array#empty?
, or exclamation marks for methods that are destructible:
>> a = [5, 1, 9, 2, 7] # Create an array and store it in a
=> [5, 1, 9, 2, 7]
>> a.sort # sort merely returns a new, sorted array
=> [1, 2, 5, 7, 9]
>> a
=> [5, 1, 9, 2, 7] # a still is in its original order
>> a.sort! # sort! instead sorts the original array
=> [1, 2, 5, 7, 9]
>> a
=> [1, 2, 5, 7, 9] # a was changed
Conditionals
Conditionals in Ruby are very similar to other programming languages, with two notable exceptions. First, it’s possible to put a conditional after the statement it protects to make the code more readable:execute_dangerous_operation() if user.is_authorized?
is equal to
if user.is_authorized?
execute_dangerous_operation()
end
Secondly, Ruby has not only an if
but an unless
. This is a syntactic nicety for when you want to check for the absence of a condition in a more readable manner:
unless user.is_admin?
user.delete
else
raise “Can’t delete admins”
end
Symbols
Sometimes you’ll see names starting with a:
(colon). These are a very special feature of Ruby called symbols. Symbols can be used to index hashes or mark states in a variable like you would with an ENUM in C. They are very similar to Strings but also very different. The point about symbols is that they don’t really occupy space in memory, and the same symbol literal always resolves to the exact same symbol:
>> “a”.object_id # object_id returns Ruby’s internal identifier for an object
=> 3477510
>> “a”.object_id
=> 3475550 # a new object on the heap
>> :a.object_id
=> 184178
>> :a.object_id
=> 184178 # the same literal refers to the exact same Symbol object
You’ll find them very often as parameters to methods, where they indicate how a method should work,
User.find(:all) #find all users
User.find(:first) #find the first user
or as pointers to methods and variables (see the undef_method
example in the “Objects” paragraph above).
Classes and Modules
Ruby supports single inheritance only, but for added flexibility it supports a feature called Mixins. In Ruby, it’s possible to define Modules that contain Methods and constants and to include these modules in a class via theinclude
method. This way, you can extend the functionality of a class very easily.
Many of Ruby’s core classes even use this mechanism.Array
and Hash
, for example, both include the Enumerable
module to provide a lot of convenience methods for iterating over their contents.
Often, Modules pose certain requirements to classes that include them. The Enumerable
Module, for example, requires classes to provide at least an each
method and an implementation of <=>
, too, if its sorting features are to be used.
Modules also serve other purposes. Most importantly, they can be used to organize code into namespaces. Because classes are constants (which means you can’t assign another class to the same name), they can be stored in modules. These modules can then be nested to form namespaces.
These paragraphs probably won’t enable you to write Ruby programs, but you should be able to understand the code samples in this article now. If you want to explore Ruby a little, try the great interactive tutorial at Try Ruby, or take a peek at one of the books listed at the end of this article. If you just want to see some more code samples, check out the Wikipedia page on Ruby.
In the second part of this tutorial we will get rolling with Ruby on Rails, install the engine, take a closer look at Rails’ inner workings and discover main advantages of Ruby on Rails. Please stay tuned.
(al)