Ruby 2.7.0 Publicado

Nos complace anunciar la publicación de Ruby 2.7.0.

Introduce una serie de características nuevas y mejoras de desempeño, las más notorias son:

  • Reconocimiento de patrones
  • Mejoras en REPL
  • Compactar en el Recolector de Basura (GC)
  • Separación de argumentos posicionales y de palabra clave

Reconocimiento de patrones [Experimental]

El reconocimiento de patrones, es una característica ampliamente usada en lenguajes de programación funcional, se introduce como característica experimental. [Característica #14912]

Puede recorrer un objeto dado y asignar su valor si concuerda con un patrón.

json ='{
  "nombre": "Alice",
  "edad": 30,
  "hijos": [
    {
      "nombre": "Bob",
      "edad": 2
    }
  ]
}'
case JSON.parse(json, symbolize_names: true)
in {nombre: "Alice", hijos: [{nombre: "Bob", edad: edad}]}
  p edad
  ...
end

Puede ver más detalles en Pattern matching - New feature in Ruby 2.7.

Mejoras a REPL

irb, el ambiente interactivo incluido con Ruby (REPL; Read-Eval-Print-Loop; Bucle-Leer-Evaluar-Presentar), ahora soporta edición de múltiples líneas. Es potenciado por la gema reline, implementanda en Ruby puro y compatible con la librería readline. También provee integración con rdoc. En irb puede presentar el manual de referencia para una clase dada, para un modulo o para un método. [Característica #14683], [Característica #14787], [Característica #14918]

Además, ahora se presentan con colores tanto el código fuente de su programa al iniciar una sesión IRB con Binding#irb como los resultados de inspeccionar los objetos de las clases del núcleo de ruby.

Compactar en el recolector de basura (GC)

Esta versión introduce compactar en el recolector de basura, que defragmenta un espacio de memoria fragmentado.

Algunos programas Ruby de múltiples hilos pueden causar fragmentación en la memoria, que conlleva a un alto uso de la memoria y a que se degrade la velocidad.

El método GC.compact se introduce para compactar el montón (heap). Esta función compacta los objetos vivos en el montón de forma que usen menos páginas y que el montón sea más amigable con la técnica de administración de recursos compartidos “Copiar ante Escritura” (Copy on Write o CoW). [Característica #15626]

Separación de argumentos posicionales y de palabra clave

Se desecha la conversión automática de argumentos de palabra clave a argumentos posicionales, y tal conversión se eliminará en Ruby 3. [Característica #14183]

Ver detalle en el artículo “Separación de argumentos de palabra clave y posicionales en Ruby 3.0”.

  • Si la llamada a un método pasa un diccionario (Hash) cómo último argumento y si no pasa palabras clave, y si el método llamado acepta palabras clave, se emitirá una advertencia. Para seguir tratando el diccionario como palabras clave, agregue un operador doble splat al llamar para evitar la advertencia y asegurar el comportamiento correcto en Ruby 3.

    def foo(key: 42); end; foo({key: 42})   # advertencia
    def foo(**kw);    end; foo({key: 42})   # advertencia
    def foo(key: 42); end; foo(**{key: 42}) # OK
    def foo(**kw);    end; foo(**{key: 42}) # OK
    
  • Si la llamada a un método pasa palabras clave a un método que acepta palabras clave, pero no pasa suficientes argumentos posicionales requeridos, las palabras clave se tratarán como los argumentos posicionales finales que requiera el método pero se emitirá una advertencia. Pase el argumento como un diccionario en lugar de palabras clave para evitar la advertencia y asegurar el comportamiento correcto en Ruby 3.

    def foo(h, **kw); end; foo(key: 42)      # warned
    def foo(h, key: 42); end; foo(key: 42)   # warned
    def foo(h, **kw); end; foo({key: 42})    # OK
    def foo(h, key: 42); end; foo({key: 42}) # OK
    
  • Si un método acepta palabras clave especificas, pero no una palabra clave splat, y si en la llamada al método se pasa un diccionario o palabra clave con unas llaves que sean símbolos y otras que no sean símbolos, el diccionario será dividido y se emitirá una advertencia. Tendrá que actualizar el código que hace la llamada para pasar diccionarios separados y asegurar el comportamiento correcto en Ruby 3.

    def foo(h={}, key: 42); end; foo("key" => 43, key: 42)   # warned
    def foo(h={}, key: 42); end; foo({"key" => 43, key: 42}) # warned
    def foo(h={}, key: 42); end; foo({"key" => 43}, key: 42) # OK
    
  • Si un método no acepta palabras clave, y se llama con palabras clave, las palabras clave se tratarán como un diccionario posicional, sin advetencias. Este comportamiento seguirá operando en Ruby 3.

    def foo(opt={});  end; foo( key: 42 )   # OK
    
  • Las cadenas que no sean símbolos se aceptarán como llaves en argumentos de palabra clave si el método acepta palabras clave arbitrarias. [Característica #14183]

    def foo(**kw); p kw; end; foo("str" => 1) #=> {"str"=>1}
    
  • **nil se permite en la definición de métodos para marcar explicitamente que el método no acepta palabras clave. Llamar a un método así con palabras clave resultará en un ArgumentError. [Característica #14183]

    def foo(h, **nil); end; foo(key: 1)       # ArgumentError
    def foo(h, **nil); end; foo(**{key: 1})   # ArgumentError
    def foo(h, **nil); end; foo("str" => 1)   # ArgumentError
    def foo(h, **nil); end; foo({key: 1})     # OK
    def foo(h, **nil); end; foo({"str" => 1}) # OK
    
  • Si se pasa una palabra clave splat vacía a un método que no acepta palabras clave, ya no pasará un diccionario vacío, a menos que el diccionario vacío sea necesario para una parámetro requerido, en cuyo caso se emite una advertencia. Elimine el doble splat para continuar pasando un diccionario posicional. [Característica #14183]

    h = {}; def foo(*a) a end; foo(**h) # []
    h = {}; def foo(a) a end; foo(**h)  # {} and warning
    h = {}; def foo(*a) a end; foo(h)   # [{}]
    h = {}; def foo(a) a end; foo(h)    # {}
    

Si desea deshabilitar las advertencias de obsolescencia , por favor use un argumento en la línea de ordenes -W:no-deprecated o añada Warning[:deprecated] = false a su código.

Otras caracerísticas nuevas y notables

  • Se introducen paramétros numerados como parámetros por omisión en bloques. [Característica #4475]

  • Un rango sin inicio se introduce de forma experimental. Podría no ser tan útil como un rango sin terminación, pero es bueno para lenguajes específicos para un domino (DSL). [Característica #14799]

    ary[..3]  # identico a ary[0..3]
    rel.where(ventas: ..100)
    
  • Se añade Enumerable#tally. Que cuenta las ocurrencias de cada elemento.

    ["a", "b", "c", "b"].tally
    #=> {"a"=>1, "b"=>2, "c"=>1}
    
  • Ahora se permite llamar un método privado con un literal self como receptor. [Característica #11297], [Característica #16123]

    def foo
    end
    private :foo
    self.foo
    
  • Se añade Enumerator::Lazy#eager que genera un enumerador no-perezoso a partir de un enumerador perezoso. [Característica #15901]

    a = %w(foo bar baz)
    e = a.lazy.map {|x| x.upcase }.map {|x| x + "!" }.eager
    p e.class               #=> Enumerator
    p e.map {|x| x + "?" }  #=> ["FOO!?", "BAR!?", "BAZ!?"]
    

Mejoras en desempeño

  • JIT [Experimental]

    • El código compilado con JIT es recompilado a uno menos optimizado cuando los supuestos de la optimización dejan de ser válidos.

    • Un método se puede ejecutar en línea (method inlining o inserción en lugar de llamado) cuando un método se considera puro. Esta optimización aún es experimental y muchos métodos aún NO se consideran puros.

    • El valor por omisión de --jit-min-calls cambió de 5 a 10,000

    • El valor por omisión de --jit-max-cache cambió de 1,000 a 100

  • Se cambia la estrategia del cache de fibras y se aumenta la velocidad de creación de fibras. GH-2224

  • Module#name, true.to_s, false.to_s y nil.to_s ahora siempre retornan una cadena congelada. La cadena retornada es siempre la misma para un objeto dado. [Experimental] [Característica#16150]

  • Se mejora el desempeño de CGI.escapeHTML. GH-2226

  • Se mejora el desempeño de `Monitor y MonitorMixin [Característica #16255]

  • El cache para llamada a métodos, que ha estado allí aproximadamente desde 1.9, se ha mejorado: la tasa de aciertos al cache ha aumentado de 89% a 94%. Ver GH-2583

  • El método RubyVM::InstructionSequence#to_binary genera un binario compilado. El tamaño del binario se ha reducido. [Característica #16163]

Otros cambios notables desde la versión 2.6

  • Se actualizaron algunas librerías estándar
  • Las siguientes librerías ya no son gemas incluidas. Instale las gemas correspondientes para contar con sus características.
    • CMath (cmath gem)
    • Scanf (scanf gem)
    • Shell (shell gem)
    • Synchronizer (sync gem)
    • ThreadsWait (thwait gem)
    • E2MM (e2mmap gem)
  • profile.rb se eliminó de la libería estándar.

  • Se promovieron de stdlib a gemas por omisión
    • Las siguientes gemas por omisión se publicaron en rubygems.org
      • benchmark
      • cgi
      • delegate
      • getoptlong
      • net-pop
      • net-smtp
      • open3
      • pstore
      • singleton
    • Las siguientes gemas por omisión se promovieron en ruby-core pero aún no se han publicado en rubygems.org
      • monitor
      • observer
      • timeout
      • tracer
      • uri
      • yaml
  • Proc.new y proc sin bloque en un método llamado con un bloque ahora produce una advertencia.

  • lambda sin un bloque en un método llamado con un bloque lanza una excepción.

  • Actualizada la versión de Unicode y de Emoji de 11.0.0 a 12.0.0. [Característica #15321]

  • Actualizada la versión de Unicode a 12.1.0, añadiendo soporte para U+32FF SQUARE ERA NAME REIWA. [Característica #15195]

  • Date.jisx0301, Date#jisx0301, y Date.parse soportan la nueva era japonesa. [Característica#15742]

  • Requiere compiladores que soporten C99. [Misc #15347]

Vea más detalles en el archivo NEWS o en la bitácora de cambios.

Con estos cambios, 4190 archivos modificados, 227498 inserciones(+), 99979 eliminaciones (-) desde Ruby 2.6.0!

!Feliz Navidad, Felices Fiestas y disfrute programando con Ruby 2.7!

Descargas

  • https://cache.ruby-lang.org/pub/ruby/2.7/ruby-2.7.0.tar.bz2

    SIZE: 14703381
    SHA1: b54f4633174dbc55db77d9fd6d0ef90cc35503af
    SHA256: 7aa247a19622a803bdd29fdb28108de9798abe841254fe8ea82c31d125c6ab26
    SHA512: 8b8dd0ceba65bdde53b7c59e6a84bc6bf634c676bfeb2ff0b3604c362c663b465397f31ff6c936441b3daabb78fb7a619be5569480c95f113dd0453488761ce7
    
  • https://cache.ruby-lang.org/pub/ruby/2.7/ruby-2.7.0.tar.gz

    SIZE: 16799684
    SHA1: 6f4e99b5556010cb27e236873cb8c09eb8317cd5
    SHA256: 8c99aa93b5e2f1bc8437d1bbbefd27b13e7694025331f77245d0c068ef1f8cbe
    SHA512: 973fc29b7c19e96c5299817d00fbdd6176319468abfca61c12b5e177b0fb0d31174a5a5525985122a7a356091a709f41b332454094940362322d1f42b77c9927
    
  • https://cache.ruby-lang.org/pub/ruby/2.7/ruby-2.7.0.tar.xz

    SIZE: 11990900
    SHA1: 943c767cec037529b8e2d3cc14fc880cad5bad8d
    SHA256: 27d350a52a02b53034ca0794efe518667d558f152656c2baaf08f3d0c8b02343
    SHA512: dd5690c631bf3a2b76cdc06902bcd76a89713a045e136debab9b8a81ff8c433bbb254aa09e4014ca1cf85a69ff4bcb13de11da5e40c224e7268be43ef2194af7
    
  • https://cache.ruby-lang.org/pub/ruby/2.7/ruby-2.7.0.zip

    SIZE: 20571744
    SHA1: fbebdd3a2a641f9a81f7d8db5abd926acea27e80
    SHA256: 8bf2050fa1fc76882f878fd526e4184dc54bd402e385efa80ef5fd3b810522e0
    SHA512: 5060f2dd3bfd271ef255b17589d6d014260d7ec2d97b48112b717ee01c62fe125c3fe04f813e02d607cea3f0a2a812b14eb3a28d06c2551354dfeff5f4c3dd6b
    

¿Qué es Ruby?

Ruby fue desarrollado primero por Matz (Yukihiro Matsumoto) en 1993, y ahora es desarrollado como Código Abierto. Corre en múltiples plataformas y se usa en todo el mundo especialmente para desarrollo web.