The Ruby Programming Language COMP 205 - Week

The Ruby Programming Language COMP 205 - Week

The Ruby Programming Language COMP 205 - Week 7 Dr. Chunbo Chu Courtesy: Matthew Hearn What is Ruby? A programming language invented by Yukihiro Matsumoto in 1995 Designed to be a combination of the creators favorite languages Perl SmallTalk Eiffel Ada Lisp Why Use Ruby?

According to its creators Ruby Is flexible Has simple syntax Is easy to read Is free Runs on Windows, Mac OS X, UNIX, Linux, and numerous other operating systems Overview of Rubys Features Interpreted langauge Object Oriented Everything

is an object All types have methods and instance variables Single inheritance of classes Overview of Rubys Features Parts of the language can be redefined and programmers can add to it Blocks of code can be passed as

parameters to methods Variables can be created at any time and dont need declarations Ruby on Windows The one-click installer from rubylang.org the Ruby language itself dozens of popular extensions and packages a syntax-highlighting editor and execution environment a Windows help file that contains the full text of the book, Programming Ruby: The Pragmatic Programmer's Guide. Hello World in Ruby

First, launch SciTE on your VM: Start All ProgrammsRuby-186-26 Type: puts "Hello World!" Save as hello.rb Run it: ToolsGo (F5) Syntax Rubys creators take pride in its brief and simple syntax Ruby is a case-sensitive language

Comments begin with a # character and continue to the end of the line ## print print aa string string in in 33 ways ways print hello world print hello world puts puts hello hello world world ## pp is is aa little little different, different, though though pp hello hello world

world Syntax No semicolons at the end of statements No curly braces to group code Method calls dont need parentheses if if xx == == 55 print print xx end end Basic syntax rules

02/29/2020 Local variables, method parameters, and method names should all start with a lowercase letter or _ Global variables are prefixed with a $ sign Class names, module names, and constants should start with an uppercase letter Class instance variables begin Ruby Recitation with an 15-441 @ sign

10 Ruby is Object Oriented In Ruby, everything is an object Objects are often used to model the realworld objects An object has attributes and methods belonging to it A bicycle: Ruby is Object Oriented A software object:

Methods operate on an object's internal state and serve as the primary mechanism for object-to-object communication Hiding internal state and requiring all interaction to be performed through an object's methods is known as data encapsulation a fundamental principle of object-oriented programming. Why OOP? Modularity: The source code for an object can be

written and maintained independently of the source code for other objects. Information-hiding: By interacting only with an object's methods, the details of its internal implementation remain hidden from the outside world. Code re-use: If an object already exists, you can use that object in your program. Pluggability and debugging ease: If a particular object turns out to be problematic, you can simply remove it from your application and plug in a different object as its replacement. Classes Try the following puts 5.next puts (5 + 7).to_s Classes

Ruby has classes to represent objects Classes have the following format class class ClassName ClassName ## attributes attributes ## methods methods end end Methods are defined in classes like this def def do_stuff do_stuff ## stuff

stuff end end Numbers Numbers may be written in decimal, hexadecimal, octal, or binary Decimal: 3405691582 Hex: 0xCAFEBABE or 0XCAFEBABE Octal: 031277535276 or 0o31277535276 Binary: 0b11001010111111101011101010111110 or 0B etc.

For readability, numbers may contain (but not begin or end with) underscores Examples: 3_405_691_582, 0b_111_101_101 Other data types Strings can be singly quoted or doubly quoted Doubly quoted strings can interpolate values for #{expression} and for #@variable, and allow

the usual escape characters In a singly quoted string, \' is the only recognized escape character Strings are not immutable Arrays are untyped and expandable, for example [1, 2, "hi"] Ruby has hashes: { :banana => 'yellow', :cherry => 'red' } Operators Almost all the Java operators, except ++ and Ruby uses =~ and !~ for Perl-style regular expressions ** is the exponentiation operator + is used to concatenate strings

There is a to_s method for converting other things to strings Operators .. is an inclusive range, ... is an exclusive range So 1..4 and 1...5 both mean 1, 2, 3, 4 0..k is a range object, but [0..k] is an array with one element, a range defined? is a prefix operator that returns a true value or nil

Ruby has both !, &&, || and also not, and, or Precedence is ! > && > ||, but not > and == or Other data types Ruby has regular expressions, dates, and times Ruby has ranges, such as 1..10 Ruby has symbols, which stand for themselves You can think of them as immutable strings Examples are :banana and :cherry

Since they are immutable, they make good keys for hashes They are also often used to refer to methods and variables: attr :do_it Assignment and alias statements Assignment statements use =, +=, *=, etc. You can have multiple (parallel) assignment: x, y= 1, 2 # same as x=1; y=2 x, y = y, x swaps the values of x and y x,y,z = [1,2,3] # Array elements automatically assigned to variables Ruby does not have the ++ and -operators

You can create a new name for a method, operator, global variable, or regular expression backreference Syntax: alias new_name original_name Example: alias display puts if statements Multi-line version: if condition then code elsif condition then

code else code end The then is optional at the end of a line Single-line versions: if condition then code elsif condition then code else code end then is required to separate condition from code statement if condition Statement Modifier if radiation > 3000 puts "Danger, Will Robinson" end

#is the same as: puts "Danger, Will Robinson" if radiation > 3000 In Ruby, everything is an expression The return value of an if "statement is the value of the last expression in the code that was executed, or nil if no block of code was executed if x == 1 name = "one" elsif x == 2 name = "two" elsif x == 3 then name = "three" elsif x == 4; name = "four" else name = "many" end name = if x == 1 then "one"

elsif x == 2 then "two" elsif x == 3 then "three" elsif x == 4 then "four" else "many" end unless statements unless means if not Multi-line version: unless condition then code else code

end The then is optional at the end of a line Single-line versions: unless condition then code else code end then is required to separate condition from code statement unless condition case and === case expr_1 expr_n === expr_1 has many meanings expr_2

thentest when Simple equality code expr_1 =~ expr_n when expr_3, ..., expr_n then expr_n.kindof? expr_1 code expr_n.include? expr_1 else Thecode === operator has the misleading name case end equality operator

Cases do not fall through; no break is needed Comparisons use expr_n === expr_1 and not the other way around name = case when x == 1 then "one when x == 2 then "two" when x == 3 then "three" when x == 4 then "four" else "many" end name = if x == 1 then "one elsif x == 2 then "two elsif x == 3 then "three" elsif x == 4 then "four else "many end tax = case income when 0..7550 income * 0.1

when 7550..30650 755 + (income-7550)*0.15 when 30650..74200 4220 + (income-30655)*0.25 when 74200..154800 15107.5 + (income-74201)*0.28 when 154800..336550 37675.5 + (income-154800)*0.33 else 97653 + (income-336550)*0.35 end puts case x when String then "string" when Numeric then "number" when TrueClass, FalseClass then "boolean" else "other" end Loops in Ruby Ruby has several loops

while condition do statements end begin statements end while condition until condition statements end begin statements

end until condition Loops in Ruby for variable in range do statements end loop do statements end statement while condition

statement until condition loop { statements } However, loops are not used as often in Ruby as in other languages Instead, Ruby programmers use iterator methods Blocks A block is a set of statements enclosed by begin..end, by do..end, or by {..}

begin introduces a new scope; variables seen for the first time inside a begin..end block are local to that scope do..end and {..} are interchangeable Defining methods Defining an instance method: Defining a class method: def method_name(parameter1, ..., parameterN) statements end

def Class_name.method_name(parameter1, ..., parameterN) statements end Parentheses are optional Defining methods A method may contain explicit return and/or return value statements A method may return multiple values: return x, 2*x, x**2 the values are collected into an array, and the array becomes the single return value of the method

A method which reaches the end returns the value of the last statement executed (often nil) # Convert polar coordinates to Cartesian coordinates def cartesian(magnitude, angle) [magnitude*Math.cos(angle), magnitude*Math.sin(angle)] end Methods of this form are typically intended for use with parallel assignment so that each return value is assigned to a separate variable: x,y = cartesian(distance,theta) A method may also have a block as an (invisible) parameter The block is not mentioned in the def line, but is executed when the method evaluates the yield

statement Calling methods Methods may be called as in Java Parentheses are usually unnecessary puts "x = #{x}" All values are objects, and may have methods

puts("x = #{x}") x = y.abs n = "abcdefg".length A method may be passed a block as an invisible parameter def twice n yield 2 * n # uses the block 99 end p = twice(7) { |y| puts "y = #{y}" } puts "p = #{p}" y = 14 p = 99

Activity Write a method that takes three numbers and return the max and min def max_min(num1, num2, num3) if num1>num2 if num1>num3 max=num1 else max=num3 elseif num2>num3 max=num2 else max=num3 #... return max, min end Iterators An iterator returns values one at a time

The syntax is object.iterator { |value| statement } or object.iterator do |value| statements end The object is typically an array, a range, or a hash, but it can be any object with a coroutine Iterators In Ruby, loops are considered low-level, to be used only when there is no appropriate iterator collection.each step through every element

n.times do a block n times n.downto(limit) step from n down to and including limit n.upto(limit) step from n up to and including limit string.each_line get each line from a string string.each_char get each character (as an integer) from a string Example use of an iterator a = [1, 1, 2, 3, 5, 8, 13] a.each { |i| print " #{i}" } a.each do |i| print " #{i}" end

Output: 1 1 2 3 5 8 13 Output: 1 1 2 3 5 8 13 In the above, each is a method A block is a chunk of code enclosed by {...} or by do...end By convention, braces are used for single-line blocks, do...end for multi-line blocks Activity you have the array: names = [ "daniel", john", alex", eric" ] Use String#capitalize to print these names capitalized. Then use String#upcase..

Blocks again A block is a chunk of code that can be passed as a parameter to a method A block isnt a statementit cant be used alone Its passed as an invisible parameter, and executed with the yield statement Blocks Blocks of code can be passed as parameters numbers numbers == [1, [1, 2, 2, 3] 3]

numbers.each numbers.each do do |num| |num| puts puts num num end end Blocks Blocks can be used for purposes other than iterators too arr arr == [1, [1, 2, 2, 3, 3, 4] 4] ## Returns

Returns the the first first number number >> 33 arr.find arr.find do do |num| |num| num num >> 33 end end Simplest use of yield ---------def three_times three_times hello puts "---------- three_times hello

yield hello yield yield end three_times { puts "hello" } My version of loop def my_loop yield while true end a = [1, 2, 3, 5, 8, 13, "hi"] my_loop do break if a.empty?

print a.pop end puts "Done" hi1385321Done Activity Write a method to calculate the first N Fibonacci numbers: 1, 1, 2, 3, 5, 8, , and output these numbers. Next, move the output out of the method. Fibonacci numbers def fibonacci_upto n

i1, i2 = 1, 1 while i1 < n yield i1 i1, i2 = i2, i1 + i2 end end fibonacci_upto(100) { |f| print " ", f } 1 1 2 3 5 8 13 21 34 55 89 Passing a parameter to the block ---------- count_to 3 defcount_to n putshello

"---------- count_to hello #{n}" for ihello in 1..n yield end end count_to 3 { puts "hello" } Returning a value from a coroutine ---------- count_to_3 def count_to_3 puts 1"---------- count_to_3" yield21

yield32 yield 3 end count_to_3 { |result| puts result } Regular expressions Ruby has regular expressions, almost identical to the way they are done in Perl Example: hamlet = "The slings and arrows of outrageous fortune." hamlet.scan(/w+/) ["The", "slings", "and", "arrows", "of", "outrageous", "fortune"]

Arrays Arrays are created like this ## Note: Note: Objects Objects are are created created with with the the new new method. method. ## Create Create aa 10 10 element element array array arr arr == Array.new(10)

Array.new(10) Arrays have a lot of built in features Arrays Operators like + and - will perform various functions ## Create Create 22 arrays arrays and and combine combine them them arr1 arr1 == [1, [1, 2, 2, 3,

3, 4, 4, 5] 5] arr2 = [6, 7, 8, 9, 10] arr2 = [6, 7, 8, 9, 10] arr3 arr3 == arr1 arr1 ++ arr2 arr2 ## arr3 arr3 is is [1, [1, 2, 2, 3, 3, 4, 4, 5, 5, 6,

6, 7, 7, 8, 8, ## 9, 9, 10] 10] ## Remove Remove all all values values in in arr4 arr4 from from arr3 arr3 arr4 arr4 == [1, [1, 3, 3, 9] 9] arr5 arr5 == arr3

arr3 arr4 arr4 ## arr5 arr5 is is [2, [2, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 10] 10] Arrays Arrays can also be treated like a stack using the push and pop methods

The sort method will sort the array Union and Intersection can be accomplished with the | and & operators respectively Arrays An array literal can be written with brackets and commas a = [1, 1, 2, 3, 5, 8, 13, "hi"] Arrays are zero based: a[2] == 2 Arrays can be expanded a = a + [21, 34] pa

[1, 2, 3, 5, 8, 13, "hi", 21, 34] Arrays can be treated as stacks, with a.push(v) and v = a.pop Arrays The join(string) method creates a string of the elements of the array, with the given string between each pair of elements You can take slices of arrays, sort them, find unique elements, perform set operations, transpose 2-dimensional arrays, etc.

Before you write methods to manipulate arrays, you should look to see whether the method you want has already been written Hashes A hash can use any object as an index in a list hsh hsh == {one {one => => 1, 1, two => 2, two => 2, three three => => 3}

3} ## Assigns Assigns num num to to 22 num num == hsh[two] hsh[two] Hashes A hash (hash table) literal can be written with braces, commas, and the => arrow h = {:apple => :red, :banana => :yellow, :cherry => :red} Element access is similar to that for arrays:

h[:banana] h[:apple] = :green ph :yellow {:banana=>:yellow, :cherry=>:red, :apple=>:green} You can use any types for keys and values, but the characteristics of symbols make them especially useful as keys Adding and removing methods

def adds a method; undef removes a method To add an instance method, first open the class Example: class String; def nchars; length; end; end There are several ways to add a class method to a class The only parameter to undef is the method name

def Person.species; 'human'; end Within the Person class, you can use def self.species You can add a method to individual objects def oscar.mood; 'grouchy' ; end Attributes (instance variables) Attributes (instance variables) of an object are written with an @ prefix: @name, @age, @hobbies, @favorite_language By default, attributes are private You can write getters:

def name @name end You can write setters: def name=(new_name) @name = new_name end When you define the setter, there is no space before the = in the header When you call the setter, you can use a space: teacher.name = "Dave" Yes, we are calling the method

name= ! Shorthand for getters and setters Writing code for routine getters and setters is tedious, so in Ruby we dont have to do it Shorthand for creating getters: attr_reader :name, :age, :hobbies

Note the use of symbols, not variables or strings Shorthand for creating setters: attr_writer :name, :hobbies Shorthand for creating both at once: attr_accessor :name, :favorite_language By the way, these arent special Ruby syntax; they are methods that write the getters and setters for you Ruby uses lots of metaprogramming: programs that write programs Access controls Public methods can be called from anywhere Protected methods can be called only within the class and its subclasses

Private methods cannot be called with an explicit receiver, only with an implicit self In Ruby, methods are public by default The functions public, protected, and private can be called with or without arguments With arguments, they set the access of the named methods Example: private :dump, :swear With no arguments, they set the default access for all subsequent methods Some File < IO methods gets get a line of text getc get a character of text (as ASCII; use .chr) ungetc put back a character pos the current character position in the input stream lineno the number of times gets has been called pos= move to the given position in the file rewind move to the beginning of the file readlines read the stream as an array of strings write(string), print(string), <<(string) write at the current position eof? test if at the end of file closed? test if the file has been closed Some File methods rename(oldname, newname) rename a file read(filename) read the entire file as a single string readlines(filename) read the entire file as an array of strings open(filename, mode)

with no block, a synonym for File.new with a block, the file is passed to the block, and automatically closed when the block finishes Modes: "r" read, "r+" read and write, "w" write, "a" append exists?(filename) test if a file with that name exists writable?(filename) test if the file can be written directory?(filename) test if the file is a directory zero?(filename) test if the file is empty size(filename) returns the size of the file mtime(filename) returns the modification time of the file Some String methods ljust(length), center(length), rjust(length) left justify, center, or right justify the string by padding with spaces downcase, upcase, swap, capitalize modify capitalization

include?(s_or_c) tests whether the string includes the given string or character index(s_or_c [, offset]) returns the index after offset(or nil) at which the gives string starts rindex(s_or_c [, limit]) returns the last index (before limit), or nil, at which the string starts Some more String methods strip remove leading and trailing spaces chop remove the last character chomp remove the last character if it is a newline tr(chars, replacement) replace the characters in chars with the corresponding characters in replacement

Some Array methods min, max return the smallest or largest element uniq return an array with no duplicate elements compact return an array with no nil elements sort return a sorted array & perform an intersection (only elements in both) | perform a union (elements in either) grep(regexp) return elements matching the pattern

push(element) add the element to the end of the array pop remove and return the last element shift remove and return the first element Activity Get 5 numbers from the user and display the numbers in ascending order. Chaining Nondestructive methods can usually be chained Example: x= gets.chomp.strip.downcase

Many destructive methods return nil if they make no changes in the receiver, hence cannot be chained Example: x= gets.chomp!.strip!.downcase! will result in a runtime error Context def do_it

a = [1, 2, 3, 5, 8, 13, "hi"] x = 4 # local variable a.my_each { |v| print v * x, "; " } end do_it 4; 8; 12; 20; 32; 52; hihihihi; Notice that the print v*x statement is being executed in the my_each method, not in the do_it method However, x is local to the do_it method How can this be? Answer: The block carries its context along with it A block is a closure More iterators

collection.each_index iterate over the indices of a collection collection.each_with_index iterate over the values in a collection, along with their indices Example: lineup.each_with_index { |man, pos| print pos, man } hash.each_key iterate over keys hash.each_value iterate over values hash.each_pair iterate over key-value pairs collection.select { |v| condition } choose only items that meet the condition collection.map { |v| transformation } create a

new collection with the transformation applied to each item Proc Objects Ruby can create methods instantly using blocks and the Proc object These can then be passed as parameters in methods my_method1 my_method1 == Proc.new Proc.new do do puts puts Hi!! Hi!! end end ## Prints Prints Hi!! Hi!!

my_method1.call my_method1.call my_method2 my_method2 == Proc.new Proc.new do do |x, |x, y| y| return return xx ++ yy end end ## Returns Returns 66 my_method2.call(5, my_method2.call(5, 1) 1) Procs A proc is a procedure that is stored in a variable

p = Proc.new { |x, y, z| puts 100 * x + 10 * y+z} Call a proc by using the variables call method p.call 14, 9, 2 1492 max = Proc.new do |a, b| if a > b then a else b end end

puts max.call(0.8, 0.12) 0.8 Procs are closures, too def scoper p x=3 p.call end x = 77 p = Proc.new { puts x } x = 19 scoper p

19 Procs as parameters A block passed as a parameter to a function becomes a Proc The formal parameter must be last, and is prefixed with an ampersand def foo &b b.call end foo { puts "Hi!" } Hi! Adding methods to a class

To add (or replace) a method to a class, just open up the class definition again and define the method You can even do this with Rubys built-in classes class Array def every_other i=0 while i < self.length yield self[i] i += 2 end end end [1, 2, 3, 5, 8, 13].every_other {|v| print v, "; "} The command line irb starts an interactive Ruby interpreter ruby starts Ruby, with input from the command line ruby filename.rb executes the given file ruby -e quotedString executes the quoted string End with an end-of-file character (^D or F6)

Not actually very useful Example: ruby -e 'puts "The time is #{Time.now}" ' ruby -v tells you Rubys version number On Unix, if the first line of a Ruby file is #!/usr/bin/ruby (or wherever ruby is located), the file can be executed by just entering the file name Looking ahead Ruby on Rails (or just Rails) is the killer app that catapulted Ruby to prominence Web applications are extremely complex serverside programs that communicate with the users browser using HTML, XML, and CSS, do session management, and handle a server-side database Rails uses metaprogramming to write your web

application for you Its hard to convey just how much work this saves The downside? You still have to understand the programs that it writes Tutorials There are many Ruby tutorials on the web--here are some of special interest http://www.tutorialspoint.com/ruby/ http://www.math.umd.edu/~dcarrera/ruby/0.3/

Pretty complete http://www.infoq.com/minibooks/ruby Short enough to read through http://www.troubleshooters.com/codecorn/ruby/basictutorial.htm Very similar in style to the w3schools tutorials A favorite free online book http://poignantguide.net/ruby/

For something just a little bit different :) Conclusion Ruby is a versatile language The fact that everything is an object makes using it more predictable Blocks of code are very useful and greatly expand the language Any questions? How to run Ruby on Linux

Create a file name>.rb my_file.rb Or place this at the top of the file and make the file executable #!/usr/bin/ruby Thats it!!

Recently Viewed Presentations

  • All About Assessment

    All About Assessment

    Eastern Mennonite University. Introduction Raised in a Pastor's Home. Work with Mennonite Conciliation Service (1986-1988) First congregational conflict case in 1987. ... make proactive adaptations to be. successful in those environments. For Discussion.
  • www.calsaws.org

    www.calsaws.org

    Technical Assessment. IAPDU. Prepare IAPDU Framework. Incorporate Alternatives Analysis. Incorporate Cost Benefit Analysis. Update Budget. Prepare IAPDU Package (Narrative & Appendices) Facilitate Review & Approval. Assess Impact on Requirements. Based on technical assessment results, requirements will be reviewed & updated...
  • The Diagnosis and Management ofNonalcoholic Fatty Liver Disease:

    The Diagnosis and Management ofNonalcoholic Fatty Liver Disease:

    The most important histological feature of NAFLD associated with long-term mortality is fibrosis; specifically, zone 3 sinusoidal fibrosis plus periportal fibrosis (stage 2) to advanced (bridging fibrosis [stage 3] or cirrhosis [stage 4]). Patients with NAFLD have increased overall mortality...
  • Predation - Southeast Missouri State University
  • Describing Bioinformatic Metadata at EBI James Malone malone@ebi.ac.uk

    Describing Bioinformatic Metadata at EBI James Malone [email protected]

    Describing Bioinformatic Metadata at EBI James Malone [email protected] Master headline * Cross-Domain Data available from EBI Genomes DNA & RNA sequence Gene expression Protein sequence Protein families, motifs and domains Protein structure Protein interactions Chemical entities Pathways Systems Literature and...
  • Increase Your Productivity by Doing Less

    Increase Your Productivity by Doing Less

    Most Frequent Posters, 1987-2012 1987. Nelson Pardee
  • AUR_DWH-BI - euroCRIS

    AUR_DWH-BI - euroCRIS

    The case for interoperability in CRIS solutions euroCRIS membership meeting Bologna May 26-27 2011 Susanna Mornati, CILEA Main dimensions Over 27M € annual revenue (red: financial contribution from the Ministry, green: revenues from activities) Over 120 employees (high degree of...
  • Issue Y2K The Great War for Talent!

    Issue Y2K The Great War for Talent!

    Two Weeks in July 2004: Not Your Father's World! "China's size does not merely enable low-cost manufacturing; it forces it. Increasingly, it is what Chinese ...