-
Notifications
You must be signed in to change notification settings - Fork 0
/
active_record_querying.html
2035 lines (1973 loc) · 216 KB
/
active_record_querying.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!doctype html>
<html dir="ltr" lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Interfaz de Consulta de Active Record — Ruby on Rails Guides</title>
<link rel="stylesheet" type="text/css" href="stylesheets/style-v2.css" data-turbo-track="reload">
<link rel="stylesheet" type="text/css" href="stylesheets/print-v2.css" media="print">
<link rel="stylesheet" type="text/css" href="stylesheets/highlight-v2.css" data-turbo-track="reload">
<link rel="icon" href="images/favicon.ico" sizes="any">
<link rel="apple-touch-icon" href="images/icon.png">
<script src="javascripts/@hotwired--turbo.js" data-turbo-track="reload"></script>
<script src="javascripts/clipboard.js" data-turbo-track="reload"></script>
<script src="javascripts/guides.js" data-turbo-track="reload"></script>
<meta property="og:title" content="Interfaz de Consulta de Active Record — Ruby on Rails Guides" />
<meta name="description" content="Interfaz de Consulta de Active RecordEsta guía cubre diferentes formas de recuperar datos de la base de datos utilizando Active Record.Después de leer esta guía, sabrás: Cómo encontrar registros usando una variedad de métodos y condiciones. Cómo especificar el orden, los atributos recuperados, la agrupación y otras propiedades de los registros encontrados. Cómo usar la carga anticipada para reducir el número de consultas a la base de datos necesarias para la recuperación de datos. Cómo usar métodos de búsqueda dinámica. Cómo encadenar métodos para usar múltiples métodos de Active Record juntos. Cómo verificar la existencia de registros particulares. Cómo realizar varios cálculos en modelos de Active Record. Cómo ejecutar EXPLAIN en relaciones." />
<meta property="og:description" content="Interfaz de Consulta de Active RecordEsta guía cubre diferentes formas de recuperar datos de la base de datos utilizando Active Record.Después de leer esta guía, sabrás: Cómo encontrar registros usando una variedad de métodos y condiciones. Cómo especificar el orden, los atributos recuperados, la agrupación y otras propiedades de los registros encontrados. Cómo usar la carga anticipada para reducir el número de consultas a la base de datos necesarias para la recuperación de datos. Cómo usar métodos de búsqueda dinámica. Cómo encadenar métodos para usar múltiples métodos de Active Record juntos. Cómo verificar la existencia de registros particulares. Cómo realizar varios cálculos en modelos de Active Record. Cómo ejecutar EXPLAIN en relaciones." />
<meta property="og:locale" content="en_US" />
<meta property="og:site_name" content="Ruby on Rails Guides" />
<meta property="og:image" content="https://avatars.githubusercontent.com/u/4223" />
<meta property="og:type" content="website" />
<link rel="preconnect" href="https://fonts.googleapis.com">
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
<link href="https://fonts.googleapis.com/css2?family=Noto+Sans+Arabic:[email protected]&display=swap" rel="stylesheet">
<link href="https://fonts.googleapis.com/css2?family=Heebo:[email protected]&family=Noto+Sans+Arabic:[email protected]&display=swap" rel="stylesheet">
<meta name="theme-color" content="#C81418">
</head>
<body class="guide">
<nav id="topNav" aria-label="Secondary">
<div class="wrapper">
<strong class="more-info-label">Más en <a href="https://rubyonrails.org/">rubyonrails.org:</a> </strong>
<span class="red-button more-info-button">
Más Ruby on Rails
</span>
<ul class="more-info-links s-hidden">
<li class="more-info"><a href="https://rubyonrails.org/blog">Blog</a></li>
<li class="more-info"><a href="https://guides.rubyonrails.org/">Guías</a></li>
<li class="more-info"><a href="https://api.rubyonrails.org/">API</a></li>
<li class="more-info"><a href="https://discuss.rubyonrails.org/">Foro</a></li>
<li class="more-info"><a href="https://github.com/rails/rails">Contribuir en GitHub</a></li>
</ul>
</div>
</nav>
<header id="page_header">
<div class="wrapper clearfix">
<nav id="feature_nav">
<div class="header-logo">
<a href="index.html" title="Regresar a la página principal de Guías para Edge">Guías</a>
<span id="version_switcher">
Versión:
<select class="guides-version">
<option value="https://edgeguides.rubyonrails.org/" selected>Edge</option>
<option value="https://guides.rubyonrails.org/v7.2/">7.2</option>
<option value="https://guides.rubyonrails.org/v7.1/">7.1</option>
<option value="https://guides.rubyonrails.org/v7.0/">7.0</option>
<option value="https://guides.rubyonrails.org/v6.1/">6.1</option>
<option value="https://guides.rubyonrails.org/v6.0/">6.0</option>
<option value="https://guides.rubyonrails.org/v5.2/">5.2</option>
<option value="https://guides.rubyonrails.org/v5.1/">5.1</option>
<option value="https://guides.rubyonrails.org/v5.0/">5.0</option>
<option value="https://guides.rubyonrails.org/v4.2/">4.2</option>
<option value="https://guides.rubyonrails.org/v4.1/">4.1</option>
<option value="https://guides.rubyonrails.org/v4.0/">4.0</option>
<option value="https://guides.rubyonrails.org/v3.2/">3.2</option>
<option value="https://guides.rubyonrails.org/v3.1/">3.1</option>
<option value="https://guides.rubyonrails.org/v3.0/">3.0</option>
<option value="https://guides.rubyonrails.org/v2.3/">2.3</option>
</select>
</span>
</div>
<ul class="nav">
<li><a class="nav-item" id="home_nav" href="https://rubyonrails.org/">Inicio</a></li>
<li class="guides-index guides-index-large">
<a href="index.html" id="guidesMenu" class="guides-index-item nav-item">Índice de Guías</a>
<div id="guides" class="clearfix" style="display: none;">
<hr />
<dl class="guides-section-container">
<div class="guides-section">
<dt>Comienza Aquí</dt>
<dd><a href="getting_started.html">Primeros Pasos con Rails</a></dd>
</div>
<div class="guides-section">
<dt>Modelos</dt>
<dd><a href="active_record_basics.html">Conceptos Básicos de Active Record</a></dd>
<dd><a href="active_record_migrations.html">Migraciones de Active Record</a></dd>
<dd><a href="active_record_validations.html">Validaciones de Active Record</a></dd>
</div>
<div class="guides-section">
<dt>Vistas</dt>
<dd><a href="action_view_overview.html">Resumen de Action View</a></dd>
<dd><a href="layouts_and_rendering.html">Diseños y Renderizado en Rails</a></dd>
</div>
<div class="guides-section">
<dt>Controladores</dt>
<dd><a href="action_controller_overview.html">Resumen de Action Controller</a></dd>
<dd><a href="routing.html">Enrutamiento en Rails desde el Exterior</a></dd>
</div>
<div class="guides-section">
<dt>Otros Componentes</dt>
<dd><a href="active_support_core_extensions.html">Extensiones Básicas de Active Support</a></dd>
<dd><a href="action_mailer_basics.html">Conceptos Básicos de Action Mailer</a></dd>
<dd><a href="action_mailbox_basics.html">Conceptos Básicos de Action Mailbox</a></dd>
<dd><a href="action_text_overview.html">Resumen de Action Text</a></dd>
<dd><a href="active_job_basics.html">Conceptos Básicos de Active Job</a></dd>
</div>
<div class="guides-section">
<dt>Políticas</dt>
<dd><a href="maintenance_policy.html">Política de Mantenimiento</a></dd>
</div>
<div class="guides-section">
<dt>Notas de Lanzamiento</dt>
<dd><a href="upgrading_ruby_on_rails.html">Actualizando Ruby on Rails</a></dd>
<dd><a href="7_2_release_notes.html">Versión 7.2 - ?</a></dd>
<dd><a href="7_1_release_notes.html">Versión 7.1 - Octubre 2023</a></dd>
<dd><a href="7_0_release_notes.html">Versión 7.0 - Diciembre 2021</a></dd>
<dd><a href="6_1_release_notes.html">Versión 6.1 - Diciembre 2020</a></dd>
</div>
</dl>
</div>
</li>
<li><a class="nav-item" href="contributing_to_ruby_on_rails.html">Contribuir</a></li>
<li class="guides-index guides-index-small">
<select class="guides-index-item nav-item">
<option value="index.html">Índice de Guías</option>
<optgroup label="Comienza Aquí">
<option value="getting_started.html">Primeros Pasos con Rails</option>
</optgroup>
<optgroup label="Modelos">
<option value="active_record_basics.html">Conceptos Básicos de Active Record</option>
<option value="active_record_migrations.html">Migraciones de Active Record</option>
<option value="active_record_validations.html">Validaciones de Active Record</option>
</optgroup>
<optgroup label="Vistas">
<option value="action_view_overview.html">Resumen de Action View</option>
<option value="layouts_and_rendering.html">Diseños y Renderizado en Rails</option>
</optgroup>
<optgroup label="Controladores">
<option value="action_controller_overview.html">Resumen de Action Controller</option>
<option value="routing.html">Enrutamiento en Rails desde el Exterior</option>
</optgroup>
<optgroup label="Otros Componentes">
<option value="active_support_core_extensions.html">Extensiones Básicas de Active Support</option>
<option value="action_mailer_basics.html">Conceptos Básicos de Action Mailer</option>
<option value="action_mailbox_basics.html">Conceptos Básicos de Action Mailbox</option>
<option value="action_text_overview.html">Resumen de Action Text</option>
<option value="active_job_basics.html">Conceptos Básicos de Active Job</option>
</optgroup>
<optgroup label="Políticas">
<option value="maintenance_policy.html">Política de Mantenimiento</option>
</optgroup>
<optgroup label="Notas de Lanzamiento">
<option value="upgrading_ruby_on_rails.html">Actualizando Ruby on Rails</option>
<option value="7_2_release_notes.html">Versión 7.2 - ?</option>
<option value="7_1_release_notes.html">Versión 7.1 - Octubre 2023</option>
<option value="7_0_release_notes.html">Versión 7.0 - Diciembre 2021</option>
<option value="6_1_release_notes.html">Versión 6.1 - Diciembre 2020</option>
</optgroup>
</select>
</li>
</ul>
</nav>
</div>
</header>
<hr class="hide" />
<section id="feature">
<div class="wrapper">
<h1>Interfaz de Consulta de Active Record</h1><p>Esta guía cubre diferentes formas de recuperar datos de la base de datos utilizando Active Record.</p><p>Después de leer esta guía, sabrás:</p>
<ul>
<li>Cómo encontrar registros usando una variedad de métodos y condiciones.</li>
<li>Cómo especificar el orden, los atributos recuperados, la agrupación y otras propiedades de los registros encontrados.</li>
<li>Cómo usar la carga anticipada para reducir el número de consultas a la base de datos necesarias para la recuperación de datos.</li>
<li>Cómo usar métodos de búsqueda dinámica.</li>
<li>Cómo encadenar métodos para usar múltiples métodos de Active Record juntos.</li>
<li>Cómo verificar la existencia de registros particulares.</li>
<li>Cómo realizar varios cálculos en modelos de Active Record.</li>
<li>Cómo ejecutar EXPLAIN en relaciones.</li>
</ul>
<nav id="subCol">
<h3 class="chapter">
<picture>
<!-- Using the `source` HTML tag to set the dark theme image -->
<source
srcset="images/icon_book-close-bookmark-1-wht.svg"
media="(prefers-color-scheme: dark)"
/>
<img src="images/icon_book-close-bookmark-1.svg" alt="Chapter Icon" />
</picture>
Chapters
</h3>
<ol class="chapters">
<li><a href="#¿qué-es-la-interfaz-de-consulta-de-active-record-questionmark">¿Qué es la Interfaz de Consulta de Active Record?</a></li>
<li><a href="#recuperación-de-objetos-de-la-base-de-datos">Recuperación de Objetos de la Base de Datos</a>
<ul>
<li><a href="#recuperación-de-un-solo-objeto">Recuperación de un Solo Objeto</a></li>
<li><a href="#recuperación-de-múltiples-objetos-en-lotes">Recuperación de Múltiples Objetos en Lotes</a></li>
</ul></li>
<li><a href="#condiciones">Condiciones</a>
<ul>
<li><a href="#condiciones-de-cadena-pura">Condiciones de Cadena Pura</a></li>
<li><a href="#condiciones-de-array">Condiciones de Array</a></li>
<li><a href="#condiciones-de-hash">Condiciones de Hash</a></li>
<li><a href="#condiciones-not">Condiciones NOT</a></li>
<li><a href="#condiciones-or">Condiciones OR</a></li>
<li><a href="#condiciones-and">Condiciones AND</a></li>
</ul></li>
<li><a href="#ordenamiento">Ordenamiento</a></li>
<li><a href="#seleccionar-campos-específicos">Seleccionar Campos Específicos</a></li>
<li><a href="#límite-y-desplazamiento">Límite y Desplazamiento</a></li>
<li><a href="#agrupación">Agrupación</a>
<ul>
<li><a href="#total-de-elementos-agrupados">Total de Elementos Agrupados</a></li>
<li><a href="#condiciones-having">Condiciones HAVING</a></li>
</ul></li>
<li><a href="#sobrescribir-condiciones">Sobrescribir Condiciones</a>
<ul>
<li><a href="#unscope"><code>unscope</code></a></li>
<li><a href="#only"><code>only</code></a></li>
<li><a href="#reselect"><code>reselect</code></a></li>
<li><a href="#reorder"><code>reorder</code></a></li>
<li><a href="#reverse-order"><code>reverse_order</code></a></li>
<li><a href="#rewhere"><code>rewhere</code></a></li>
<li><a href="#regroup"><code>regroup</code></a></li>
</ul></li>
<li><a href="#relación-nula">Relación Nula</a></li>
<li><a href="#objetos-de-solo-lectura">Objetos de Solo Lectura</a></li>
<li><a href="#bloqueo-de-registros-para-actualización">Bloqueo de Registros para Actualización</a>
<ul>
<li><a href="#bloqueo-optimista">Bloqueo Optimista</a></li>
<li><a href="#bloqueo-pesimista">Bloqueo Pesimista</a></li>
</ul></li>
<li><a href="#unión-de-tablas">Unión de Tablas</a>
<ul>
<li><a href="#joins"><code>joins</code></a></li>
<li><a href="#left-outer-joins"><code>left_outer_joins</code></a></li>
<li><a href="#where-associated-y-where-missing"><code>where.associated</code> y <code>where.missing</code></a></li>
</ul></li>
<li><a href="#carga-anticipada-de-asociaciones">Carga Anticipada de Asociaciones</a>
<ul>
<li><a href="#problema-de-consultas-n-1">Problema de Consultas N + 1</a></li>
<li><a href="#includes"><code>includes</code></a></li>
<li><a href="#preload"><code>preload</code></a></li>
<li><a href="#eager-load"><code>eager_load</code></a></li>
<li><a href="#strict-loading"><code>strict_loading</code></a></li>
<li><a href="#strict-loading-bang"><code>strict_loading!</code></a></li>
<li><a href="#opción-strict-loading-en-una-asociación">Opción <code>strict_loading</code> en una asociación</a></li>
</ul></li>
<li><a href="#alcances">Alcances</a>
<ul>
<li><a href="#pasar-argumentos">Pasar Argumentos</a></li>
<li><a href="#usar-condicionales">Usar Condicionales</a></li>
<li><a href="#aplicar-un-alcance-predeterminado">Aplicar un Alcance Predeterminado</a></li>
</ul></li>
</ol>
</nav>
<hr>
</div>
</section>
<main id="container">
<div class="wrapper">
<div id="mainCol">
<h2 id="¿qué-es-la-interfaz-de-consulta-de-active-record-questionmark"><a class="anchorlink" href="#¿qué-es-la-interfaz-de-consulta-de-active-record-questionmark"><span>1</span> ¿Qué es la Interfaz de Consulta de Active Record?</a></h2><p>Si estás acostumbrado a usar SQL en bruto para encontrar registros de la base de datos, generalmente encontrarás que hay mejores formas de llevar a cabo las mismas operaciones en Rails. Active Record te aísla de la necesidad de usar SQL en la mayoría de los casos.</p><p>Active Record realizará consultas en la base de datos por ti y es compatible con la mayoría de los sistemas de bases de datos, incluidos MySQL, MariaDB, PostgreSQL y SQLite. Independientemente del sistema de base de datos que estés utilizando, el formato del método de Active Record siempre será el mismo.</p><p>Los ejemplos de código a lo largo de esta guía se referirán a uno o más de los siguientes modelos:</p><p>CONSEJO: Todos los modelos siguientes usan <code>id</code> como clave primaria, a menos que se especifique lo contrario.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Author</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="n">has_many</span> <span class="ss">:books</span><span class="p">,</span> <span class="o">-></span> <span class="p">{</span> <span class="n">order</span><span class="p">(</span><span class="ss">year_published: :desc</span><span class="p">)</span> <span class="p">}</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Author < ApplicationRecord
has_many :books, -> { order(year_published: :desc) }
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Book</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="n">belongs_to</span> <span class="ss">:supplier</span>
<span class="n">belongs_to</span> <span class="ss">:author</span>
<span class="n">has_many</span> <span class="ss">:reviews</span>
<span class="n">has_and_belongs_to_many</span> <span class="ss">:orders</span><span class="p">,</span> <span class="ss">join_table: </span><span class="s1">'books_orders'</span>
<span class="n">scope</span> <span class="ss">:in_print</span><span class="p">,</span> <span class="o">-></span> <span class="p">{</span> <span class="n">where</span><span class="p">(</span><span class="ss">out_of_print: </span><span class="kp">false</span><span class="p">)</span> <span class="p">}</span>
<span class="n">scope</span> <span class="ss">:out_of_print</span><span class="p">,</span> <span class="o">-></span> <span class="p">{</span> <span class="n">where</span><span class="p">(</span><span class="ss">out_of_print: </span><span class="kp">true</span><span class="p">)</span> <span class="p">}</span>
<span class="n">scope</span> <span class="ss">:old</span><span class="p">,</span> <span class="o">-></span> <span class="p">{</span> <span class="n">where</span><span class="p">(</span><span class="ss">year_published: </span><span class="o">...</span><span class="mi">50</span><span class="p">.</span><span class="nf">years</span><span class="p">.</span><span class="nf">ago</span><span class="p">.</span><span class="nf">year</span><span class="p">)</span> <span class="p">}</span>
<span class="n">scope</span> <span class="ss">:out_of_print_and_expensive</span><span class="p">,</span> <span class="o">-></span> <span class="p">{</span> <span class="n">out_of_print</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s1">'price > 500'</span><span class="p">)</span> <span class="p">}</span>
<span class="n">scope</span> <span class="ss">:costs_more_than</span><span class="p">,</span> <span class="o">-></span><span class="p">(</span><span class="n">amount</span><span class="p">)</span> <span class="p">{</span> <span class="n">where</span><span class="p">(</span><span class="s1">'price > ?'</span><span class="p">,</span> <span class="n">amount</span><span class="p">)</span> <span class="p">}</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Book < ApplicationRecord
belongs_to :supplier
belongs_to :author
has_many :reviews
has_and_belongs_to_many :orders, join_table: 'books_orders'
scope :in_print, -> { where(out_of_print: false) }
scope :out_of_print, -> { where(out_of_print: true) }
scope :old, -> { where(year_published: ...50.years.ago.year) }
scope :out_of_print_and_expensive, -> { out_of_print.where('price > 500') }
scope :costs_more_than, ->(amount) { where('price > ?', amount) }
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Customer</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="n">has_many</span> <span class="ss">:orders</span>
<span class="n">has_many</span> <span class="ss">:reviews</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Customer < ApplicationRecord
has_many :orders
has_many :reviews
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Order</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="n">belongs_to</span> <span class="ss">:customer</span>
<span class="n">has_and_belongs_to_many</span> <span class="ss">:books</span><span class="p">,</span> <span class="ss">join_table: </span><span class="s1">'books_orders'</span>
<span class="n">enum</span> <span class="ss">:status</span><span class="p">,</span> <span class="p">[</span><span class="ss">:shipped</span><span class="p">,</span> <span class="ss">:being_packed</span><span class="p">,</span> <span class="ss">:complete</span><span class="p">,</span> <span class="ss">:cancelled</span><span class="p">]</span>
<span class="n">scope</span> <span class="ss">:created_before</span><span class="p">,</span> <span class="o">-></span><span class="p">(</span><span class="n">time</span><span class="p">)</span> <span class="p">{</span> <span class="n">where</span><span class="p">(</span><span class="ss">created_at: </span><span class="o">...</span><span class="n">time</span><span class="p">)</span> <span class="p">}</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Order < ApplicationRecord
belongs_to :customer
has_and_belongs_to_many :books, join_table: 'books_orders'
enum :status, [:shipped, :being_packed, :complete, :cancelled]
scope :created_before, ->(time) { where(created_at: ...time) }
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Review</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="n">belongs_to</span> <span class="ss">:customer</span>
<span class="n">belongs_to</span> <span class="ss">:book</span>
<span class="n">enum</span> <span class="ss">:state</span><span class="p">,</span> <span class="p">[</span><span class="ss">:not_reviewed</span><span class="p">,</span> <span class="ss">:published</span><span class="p">,</span> <span class="ss">:hidden</span><span class="p">]</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Review < ApplicationRecord
belongs_to :customer
belongs_to :book
enum :state, [:not_reviewed, :published, :hidden]
end
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Supplier</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="n">has_many</span> <span class="ss">:books</span>
<span class="n">has_many</span> <span class="ss">:authors</span><span class="p">,</span> <span class="ss">through: :books</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Supplier < ApplicationRecord
has_many :books
has_many :authors, through: :books
end
">Copy</button>
</div>
<p><img src="images/active_record_querying/bookstore_models.png" alt="Diagrama de todos los modelos de la librería"></p><h2 id="recuperación-de-objetos-de-la-base-de-datos"><a class="anchorlink" href="#recuperación-de-objetos-de-la-base-de-datos"><span>2</span> Recuperación de Objetos de la Base de Datos</a></h2><p>Para recuperar objetos de la base de datos, Active Record proporciona varios métodos de búsqueda. Cada método de búsqueda te permite pasar argumentos para realizar ciertas consultas en tu base de datos sin escribir SQL en bruto.</p><p>Los métodos son:</p>
<ul>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-annotate"><code>annotate</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-find"><code>find</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-create_with"><code>create_with</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-distinct"><code>distinct</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-eager_load"><code>eager_load</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-extending"><code>extending</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-extract_associated"><code>extract_associated</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-from"><code>from</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-group"><code>group</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-having"><code>having</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-includes"><code>includes</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-joins"><code>joins</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-left_outer_joins"><code>left_outer_joins</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-limit"><code>limit</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-lock"><code>lock</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-none"><code>none</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-offset"><code>offset</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-optimizer_hints"><code>optimizer_hints</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-order"><code>order</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-preload"><code>preload</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-readonly"><code>readonly</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-references"><code>references</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-reorder"><code>reorder</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-reselect"><code>reselect</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-regroup"><code>regroup</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-reverse_order"><code>reverse_order</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-select"><code>select</code></a></li>
<li><a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-where"><code>where</code></a></li>
</ul>
<p>Los métodos de búsqueda que devuelven una colección, como <code>where</code> y <code>group</code>, devuelven una instancia de <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/Relation.html"><code>ActiveRecord::Relation</code></a>. Los métodos que encuentran una sola entidad, como <code>find</code> y <code>first</code>, devuelven una única instancia del modelo.</p><p>La operación principal de <code>Model.find(options)</code> se puede resumir como:</p>
<ul>
<li>Convertir las opciones suministradas en una consulta SQL equivalente.</li>
<li>Ejecutar la consulta SQL y recuperar los resultados correspondientes de la base de datos.</li>
<li>Instanciar el objeto Ruby equivalente del modelo apropiado para cada fila resultante.</li>
<li>Ejecutar <code>after_find</code> y luego las devoluciones de llamada <code>after_initialize</code>, si las hay.</li>
</ul>
<h3 id="recuperación-de-un-solo-objeto"><a class="anchorlink" href="#recuperación-de-un-solo-objeto"><span>2.1</span> Recuperación de un Solo Objeto</a></h3><p>Active Record proporciona varias formas diferentes de recuperar un solo objeto.</p><h4 id="find"><a class="anchorlink" href="#find"><span>2.1.1</span> <code>find</code></a></h4><p>Usando el método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-find"><code>find</code></a> puedes recuperar el objeto correspondiente a la <em>clave primaria</em> especificada que coincida con cualquier opción suministrada. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="c"># Encuentra el cliente con clave primaria (id) 10.
</span><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">find</span><span class="p">(</span><span class="mi">10</span><span class="p">)</span>
<span class="gp">=> #<Customer id: 10, first_name: "Ryan"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.find(10)
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="o">=</span> <span class="mi">10</span><span class="p">)</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.id = 10) LIMIT 1
">Copy</button>
</div>
<p>El método <code>find</code> generará una excepción <code>ActiveRecord::RecordNotFound</code> si no se encuentra un registro coincidente.</p><p>También puedes usar este método para consultar múltiples objetos. Llama al método <code>find</code> y pasa un array de claves primarias. La devolución será un array que contiene todos los registros coincidentes para las <em>claves primarias</em> suministradas. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="c"># Encuentra los clientes con claves primarias 1 y 10.
</span><span class="gp">irb></span><span class="w"> </span><span class="n">customers</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">find</span><span class="p">([</span><span class="mi">1</span><span class="p">,</span> <span class="mi">10</span><span class="p">])</span> <span class="c1"># O Customer.find(1, 10)</span>
<span class="gp">=> [#<Customer id: 1, first_name: "Lifo"></span><span class="p">,</span> <span class="c1">#<Customer id: 10, first_name: "Ryan">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="customers = Customer.find([1, 10]) # O Customer.find(1, 10)
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">1</span><span class="p">,</span><span class="mi">10</span><span class="p">))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.id IN (1,10))
">Copy</button>
</div>
<p>ADVERTENCIA: El método <code>find</code> generará una excepción <code>ActiveRecord::RecordNotFound</code> a menos que se encuentre un registro coincidente para <strong>todas</strong> las claves primarias suministradas.</p><p>Si tu tabla usa una clave primaria compuesta, necesitarás pasar un array a find para encontrar un solo elemento. Por ejemplo, si los clientes se definieron con <code>[:store_id, :id]</code> como clave primaria:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="c"># Encuentra el cliente con store_id 3 y id 17
</span><span class="gp">irb></span><span class="w"> </span><span class="n">customers</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">find</span><span class="p">([</span><span class="mi">3</span><span class="p">,</span> <span class="mi">17</span><span class="p">])</span>
<span class="gp">=> #<Customer store_id: 3, id: 17, first_name: "Magda"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customers = Customer.find([3, 17])
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="n">store_id</span> <span class="o">=</span> <span class="mi">3</span> <span class="k">AND</span> <span class="n">id</span> <span class="o">=</span> <span class="mi">17</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE store_id = 3 AND id = 17
">Copy</button>
</div>
<p>Para encontrar múltiples clientes con IDs compuestos, pasarías un array de arrays:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="c"># Encuentra los clientes con claves primarias [1, 8] y [7, 15].
</span><span class="gp">irb></span><span class="w"> </span><span class="n">customers</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">find</span><span class="p">([[</span><span class="mi">1</span><span class="p">,</span> <span class="mi">8</span><span class="p">],</span> <span class="p">[</span><span class="mi">7</span><span class="p">,</span> <span class="mi">15</span><span class="p">]])</span> <span class="c1"># O Customer.find([1, 8], [7, 15])</span>
<span class="gp">=> [#<Customer store_id: 1, id: 8, first_name: "Pat"></span><span class="p">,</span> <span class="c1">#<Customer store_id: 7, id: 15, first_name: "Chris">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="customers = Customer.find([[1, 8], [7, 15]]) # O Customer.find([1, 8], [7, 15])
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">store_id</span> <span class="o">=</span> <span class="mi">1</span> <span class="k">AND</span> <span class="n">id</span> <span class="o">=</span> <span class="mi">8</span> <span class="k">OR</span> <span class="n">store_id</span> <span class="o">=</span> <span class="mi">7</span> <span class="k">AND</span> <span class="n">id</span> <span class="o">=</span> <span class="mi">15</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (store_id = 1 AND id = 8 OR store_id = 7 AND id = 15)
">Copy</button>
</div>
<h4 id="take"><a class="anchorlink" href="#take"><span>2.1.2</span> <code>take</code></a></h4><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-take"><code>take</code></a> recupera un registro sin ningún orden implícito. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">take</span>
<span class="gp">=> #<Customer id: 1, first_name: "Lifo"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.take
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers LIMIT 1
">Copy</button>
</div>
<p>El método <code>take</code> devuelve <code>nil</code> si no se encuentra ningún registro y no se generará ninguna excepción.</p><p>Puedes pasar un argumento numérico al método <code>take</code> para devolver hasta ese número de resultados. Por ejemplo</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customers</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">take</span><span class="p">(</span><span class="mi">2</span><span class="p">)</span>
<span class="gp">=> [#<Customer id: 1, first_name: "Lifo"></span><span class="p">,</span> <span class="c1">#<Customer id: 220, first_name: "Sara">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="customers = Customer.take(2)
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">LIMIT</span> <span class="mi">2</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers LIMIT 2
">Copy</button>
</div>
<p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-take-21"><code>take!</code></a> se comporta exactamente como <code>take</code>, excepto que generará <code>ActiveRecord::RecordNotFound</code> si no se encuentra un registro coincidente.</p><p>CONSEJO: El registro recuperado puede variar dependiendo del motor de la base de datos.</p><h4 id="first"><a class="anchorlink" href="#first"><span>2.1.3</span> <code>first</code></a></h4><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-first"><code>first</code></a> encuentra el primer registro ordenado por clave primaria (por defecto). Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">first</span>
<span class="gp">=> #<Customer id: 1, first_name: "Lifo"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.first
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">ASC</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.id ASC LIMIT 1
">Copy</button>
</div>
<p>El método <code>first</code> devuelve <code>nil</code> si no se encuentra un registro coincidente y no se generará ninguna excepción.</p><p>Si tu <a href="active_record_querying.html#applying-a-default-scope">alcance predeterminado</a> contiene un método de orden, <code>first</code> devolverá el primer registro según este ordenamiento.</p><p>Puedes pasar un argumento numérico al método <code>first</code> para devolver hasta ese número de resultados. Por ejemplo</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customers</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">first</span><span class="p">(</span><span class="mi">3</span><span class="p">)</span>
<span class="gp">=> [#<Customer id: 1, first_name: "Lifo"></span><span class="p">,</span> <span class="c1">#<Customer id: 2, first_name: "Fifo">, #<Customer id: 3, first_name: "Filo">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="customers = Customer.first(3)
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">ASC</span> <span class="k">LIMIT</span> <span class="mi">3</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.id ASC LIMIT 3
">Copy</button>
</div>
<p>Los modelos con claves primarias compuestas usarán la clave primaria compuesta completa para el ordenamiento.
Por ejemplo, si los clientes se definieron con <code>[:store_id, :id]</code> como clave primaria:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">first</span>
<span class="gp">=> #<Customer id: 2, store_id: 1, first_name: "Lifo"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.first
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">store_id</span> <span class="k">ASC</span><span class="p">,</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">ASC</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.store_id ASC, customers.id ASC LIMIT 1
">Copy</button>
</div>
<p>En una colección que está ordenada usando <code>order</code>, <code>first</code> devolverá el primer registro ordenado por el atributo especificado para <code>order</code>.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">order</span><span class="p">(</span><span class="ss">:first_name</span><span class="p">).</span><span class="nf">first</span>
<span class="gp">=> #<Customer id: 2, first_name: "Fifo"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.order(:first_name).first
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">first_name</span> <span class="k">ASC</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.first_name ASC LIMIT 1
">Copy</button>
</div>
<p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-first-21"><code>first!</code></a> se comporta exactamente como <code>first</code>, excepto que generará <code>ActiveRecord::RecordNotFound</code> si no se encuentra un registro coincidente.</p><h4 id="last"><a class="anchorlink" href="#last"><span>2.1.4</span> <code>last</code></a></h4><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-last"><code>last</code></a> encuentra el último registro ordenado por clave primaria (por defecto). Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">last</span>
<span class="gp">=> #<Customer id: 221, first_name: "Russel"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.last
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">DESC</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.id DESC LIMIT 1
">Copy</button>
</div>
<p>El método <code>last</code> devuelve <code>nil</code> si no se encuentra un registro coincidente y no se generará ninguna excepción.</p><p>Los modelos con claves primarias compuestas usarán la clave primaria compuesta completa para el ordenamiento.
Por ejemplo, si los clientes se definieron con <code>[:store_id, :id]</code> como clave primaria:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">last</span>
<span class="gp">=> #<Customer id: 221, store_id: 1, first_name: "Lifo"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.last
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">store_id</span> <span class="k">DESC</span><span class="p">,</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">DESC</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.store_id DESC, customers.id DESC LIMIT 1
">Copy</button>
</div>
<p>Si tu <a href="active_record_querying.html#applying-a-default-scope">alcance predeterminado</a> contiene un método de orden, <code>last</code> devolverá el último registro según este ordenamiento.</p><p>Puedes pasar un argumento numérico al método <code>last</code> para devolver hasta ese número de resultados. Por ejemplo</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customers</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">last</span><span class="p">(</span><span class="mi">3</span><span class="p">)</span>
<span class="gp">=> [#<Customer id: 219, first_name: "James"></span><span class="p">,</span> <span class="c1">#<Customer id: 220, first_name: "Sara">, #<Customer id: 221, first_name: "Russel">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="customers = Customer.last(3)
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">DESC</span> <span class="k">LIMIT</span> <span class="mi">3</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.id DESC LIMIT 3
">Copy</button>
</div>
<p>En una colección que está ordenada usando <code>order</code>, <code>last</code> devolverá el último registro ordenado por el atributo especificado para <code>order</code>.</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">order</span><span class="p">(</span><span class="ss">:first_name</span><span class="p">).</span><span class="nf">last</span>
<span class="gp">=> #<Customer id: 220, first_name: "Sara"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.order(:first_name).last
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">ORDER</span> <span class="k">BY</span> <span class="n">customers</span><span class="p">.</span><span class="n">first_name</span> <span class="k">DESC</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers ORDER BY customers.first_name DESC LIMIT 1
">Copy</button>
</div>
<p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-last-21"><code>last!</code></a> se comporta exactamente como <code>last</code>, excepto que generará <code>ActiveRecord::RecordNotFound</code> si no se encuentra un registro coincidente.</p><h4 id="find-by"><a class="anchorlink" href="#find-by"><span>2.1.5</span> <code>find_by</code></a></h4><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-find_by"><code>find_by</code></a> encuentra el primer registro que coincide con algunas condiciones. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="no">Customer</span><span class="p">.</span><span class="nf">find_by</span> <span class="ss">first_name: </span><span class="s1">'Lifo'</span>
<span class="gp">=> #<Customer id: 1, first_name: "Lifo"></span><span class="w">
</span><span class="err">
</span><span class="gp">irb></span><span class="w"> </span><span class="no">Customer</span><span class="p">.</span><span class="nf">find_by</span> <span class="ss">first_name: </span><span class="s1">'Jon'</span>
<span class="p">=></span> <span class="kp">nil</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_by first_name: 'Lifo'
Customer.find_by first_name: 'Jon'
">Copy</button>
</div>
<p>Es equivalente a escribir:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">first_name: </span><span class="s1">'Lifo'</span><span class="p">).</span><span class="nf">take</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(first_name: 'Lifo').take
">Copy</button>
</div>
<p>El equivalente SQL de lo anterior es:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">first_name</span> <span class="o">=</span> <span class="s1">'Lifo'</span><span class="p">)</span> <span class="k">LIMIT</span> <span class="mi">1</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.first_name = 'Lifo') LIMIT 1
">Copy</button>
</div>
<p>Ten en cuenta que no hay <code>ORDER BY</code> en el SQL anterior. Si tus condiciones de <code>find_by</code> pueden coincidir con múltiples registros, deberías <a href="#ordenamiento">aplicar un orden</a> para garantizar un resultado determinista.</p><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-find_by-21"><code>find_by!</code></a> se comporta exactamente como <code>find_by</code>, excepto que generará <code>ActiveRecord::RecordNotFound</code> si no se encuentra un registro coincidente. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="no">Customer</span><span class="p">.</span><span class="nf">find_by!</span> <span class="ss">first_name: </span><span class="s1">'does not exist'</span>
<span class="go">ActiveRecord::RecordNotFound
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_by! first_name: 'does not exist'
">Copy</button>
</div>
<p>Esto es equivalente a escribir:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">first_name: </span><span class="s1">'does not exist'</span><span class="p">).</span><span class="nf">take!</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(first_name: 'does not exist').take!
">Copy</button>
</div>
<h5 id="condiciones-con-id"><a class="anchorlink" href="#condiciones-con-id"><span>2.1.5.1</span> Condiciones con <code>:id</code></a></h5><p>Al especificar condiciones en métodos como <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-find_by"><code>find_by</code></a> y <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-where"><code>where</code></a>, el uso de <code>id</code> coincidirá con
un atributo <code>:id</code> en el modelo. Esto es diferente de <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/FinderMethods.html#method-i-find"><code>find</code></a>, donde el ID pasado debe ser un valor de clave primaria.</p><p>Ten precaución al usar <code>find_by(id:)</code> en modelos donde <code>:id</code> no es la clave primaria, como modelos de clave primaria compuesta.
Por ejemplo, si los clientes se definieron con <code>[:store_id, :id]</code> como clave primaria:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">last</span>
<span class="gp">=> #<Customer id: 10, store_id: 5, first_name: "Joe"></span><span class="w">
</span><span class="gp">irb></span><span class="w"> </span><span class="no">Customer</span><span class="p">.</span><span class="nf">find_by</span><span class="p">(</span><span class="ss">id: </span><span class="n">customer</span><span class="p">.</span><span class="nf">id</span><span class="p">)</span> <span class="c1"># Customer.find_by(id: [5, 10])</span>
<span class="gp">=> #<Customer id: 5, store_id: 3, first_name: "Bob"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.last
Customer.find_by(id: customer.id) # Customer.find_by(id: [5, 10])
">Copy</button>
</div>
<p>Aquí, podríamos intentar buscar un solo registro con la clave primaria compuesta <code>[5, 10]</code>, pero Active Record buscará un registro con una columna <code>:id</code> de <em>ya sea</em> 5 o 10, y puede devolver el registro incorrecto.</p><p>CONSEJO: El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/ModelSchema.html#method-i-id_value"><code>id_value</code></a> se puede usar para obtener el valor de la columna <code>:id</code> para un registro, para su uso en métodos de búsqueda como <code>find_by</code> y <code>where</code>. Ver ejemplo a continuación:</p><div class="interstitial code">
<pre><code class="highlight irb"><span class="gp">irb></span><span class="w"> </span><span class="n">customer</span> <span class="o">=</span> <span class="no">Customer</span><span class="p">.</span><span class="nf">last</span>
<span class="gp">=> #<Customer id: 10, store_id: 5, first_name: "Joe"></span><span class="w">
</span><span class="gp">irb></span><span class="w"> </span><span class="no">Customer</span><span class="p">.</span><span class="nf">find_by</span><span class="p">(</span><span class="ss">id: </span><span class="n">customer</span><span class="p">.</span><span class="nf">id_value</span><span class="p">)</span> <span class="c1"># Customer.find_by(id: 10)</span>
<span class="gp">=> #<Customer id: 10, store_id: 5, first_name: "Joe"></span><span class="w">
</span></code></pre>
<button class="clipboard-button" data-clipboard-text="customer = Customer.last
Customer.find_by(id: customer.id_value) # Customer.find_by(id: 10)
">Copy</button>
</div>
<h3 id="recuperación-de-múltiples-objetos-en-lotes"><a class="anchorlink" href="#recuperación-de-múltiples-objetos-en-lotes"><span>2.2</span> Recuperación de Múltiples Objetos en Lotes</a></h3><p>A menudo necesitamos iterar sobre un gran conjunto de registros, como cuando enviamos un boletín a un gran conjunto de clientes, o cuando exportamos datos.</p><p>Esto puede parecer sencillo:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="c1"># Esto puede consumir demasiada memoria si la tabla es grande.</span>
<span class="no">Customer</span><span class="p">.</span><span class="nf">all</span><span class="p">.</span><span class="nf">each</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="# Esto puede consumir demasiada memoria si la tabla es grande.
Customer.all.each do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<p>Pero este enfoque se vuelve cada vez más impráctico a medida que aumenta el tamaño de la tabla, ya que <code>Customer.all.each</code> instruye a Active Record a recuperar <em>toda la tabla</em> en una sola pasada, construir un objeto de modelo por fila y luego mantener todo el array de objetos de modelo en memoria. De hecho, si tenemos un gran número de registros, toda la colección puede exceder la cantidad de memoria disponible.</p><p>Rails proporciona dos métodos que abordan este problema dividiendo los registros en lotes amigables con la memoria para su procesamiento. El primer método, <code>find_each</code>, recupera un lote de registros y luego cede <em>cada</em> registro al bloque individualmente como un modelo. El segundo método, <code>find_in_batches</code>, recupera un lote de registros y luego cede <em>todo el lote</em> al bloque como un array de modelos.</p><p>CONSEJO: Los métodos <code>find_each</code> y <code>find_in_batches</code> están destinados a ser utilizados en el procesamiento por lotes de un gran número de registros que no cabrían en memoria a la vez. Si solo necesitas recorrer mil registros, los métodos de búsqueda regulares son la opción preferida.</p><h4 id="find-each"><a class="anchorlink" href="#find-each"><span>2.2.1</span> <code>find_each</code></a></h4><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/Batches.html#method-i-find_each"><code>find_each</code></a> recupera registros en lotes y luego cede <em>cada</em> uno al bloque. En el siguiente ejemplo, <code>find_each</code> recupera clientes en lotes de 1000 y los cede al bloque uno por uno:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_each</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_each do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<p>Este proceso se repite, recuperando más lotes según sea necesario, hasta que todos los registros hayan sido procesados.</p><p><code>find_each</code> funciona en clases de modelo, como se ve arriba, y también en relaciones:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">weekly_subscriber: </span><span class="kp">true</span><span class="p">).</span><span class="nf">find_each</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(weekly_subscriber: true).find_each do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<p>siempre que no tengan ordenación, ya que el método necesita forzar un orden internamente para iterar.</p><p>Si hay un orden presente en el receptor, el comportamiento depende de la bandera <a href="configuring.html#config-active-record-error-on-ignored-order"><code>config.active_record.error_on_ignored_order</code></a>. Si es verdadero, se genera <code>ArgumentError</code>, de lo contrario, el orden se ignora y se emite una advertencia, que es el valor predeterminado. Esto se puede anular con la opción <code>:error_on_ignore</code>, explicada a continuación.</p><h5 id="opciones-para-find-each"><a class="anchorlink" href="#opciones-para-find-each"><span>2.2.1.1</span> Opciones para <code>find_each</code></a></h5><p><strong><code>:batch_size</code></strong></p><p>La opción <code>:batch_size</code> te permite especificar el número de registros que se recuperarán en cada lote, antes de ser pasados individualmente al bloque. Por ejemplo, para recuperar registros en lotes de 5000:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_each</span><span class="p">(</span><span class="ss">batch_size: </span><span class="mi">5000</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_each(batch_size: 5000) do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<p><strong><code>:start</code></strong></p><p>Por defecto, los registros se recuperan en orden ascendente de la clave primaria. La opción <code>:start</code> te permite configurar el primer ID de la secuencia siempre que el ID más bajo no sea el que necesitas. Esto sería útil, por ejemplo, si quisieras reanudar un proceso por lotes interrumpido, siempre que hayas guardado el último ID procesado como punto de control.</p><p>Por ejemplo, para enviar boletines solo a clientes con la clave primaria a partir de 2000:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_each</span><span class="p">(</span><span class="ss">start: </span><span class="mi">2000</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_each(start: 2000) do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<p><strong><code>:finish</code></strong></p><p>Similar a la opción <code>:start</code>, <code>:finish</code> te permite configurar el último ID de la secuencia siempre que el ID más alto no sea el que necesitas.
Esto sería útil, por ejemplo, si quisieras ejecutar un proceso por lotes utilizando un subconjunto de registros basado en <code>:start</code> y <code>:finish</code>.</p><p>Por ejemplo, para enviar boletines solo a clientes con la clave primaria a partir de 2000 hasta 10000:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_each</span><span class="p">(</span><span class="ss">start: </span><span class="mi">2000</span><span class="p">,</span> <span class="ss">finish: </span><span class="mi">10000</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_each(start: 2000, finish: 10000) do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<p>Otro ejemplo sería si quisieras que múltiples trabajadores manejen la misma cola de procesamiento. Podrías hacer que cada trabajador maneje 10000 registros configurando las opciones <code>:start</code> y <code>:finish</code> apropiadas en cada trabajador.</p><p><strong><code>:error_on_ignore</code></strong></p><p>Anula la configuración de la aplicación para especificar si se debe generar un error cuando hay un orden presente en la relación.</p><p><strong><code>:order</code></strong></p><p>Especifica el orden de la clave primaria (puede ser <code>:asc</code> o <code>:desc</code>). El valor predeterminado es <code>:asc</code>.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_each</span><span class="p">(</span><span class="ss">order: :desc</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customer</span><span class="o">|</span>
<span class="no">NewsMailer</span><span class="p">.</span><span class="nf">weekly</span><span class="p">(</span><span class="n">customer</span><span class="p">).</span><span class="nf">deliver_now</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_each(order: :desc) do |customer|
NewsMailer.weekly(customer).deliver_now
end
">Copy</button>
</div>
<h4 id="find-in-batches"><a class="anchorlink" href="#find-in-batches"><span>2.2.2</span> <code>find_in_batches</code></a></h4><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/Batches.html#method-i-find_in_batches"><code>find_in_batches</code></a> es similar a <code>find_each</code>, ya que ambos recuperan lotes de registros. La diferencia es que <code>find_in_batches</code> cede <em>lotes</em> al bloque como un array de modelos, en lugar de individualmente. El siguiente ejemplo cederá al bloque proporcionado un array de hasta 1000 clientes a la vez, con el bloque final conteniendo cualquier cliente restante:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="c1"># Da add_customers un array de 1000 clientes a la vez.</span>
<span class="no">Customer</span><span class="p">.</span><span class="nf">find_in_batches</span> <span class="k">do</span> <span class="o">|</span><span class="n">customers</span><span class="o">|</span>
<span class="n">export</span><span class="p">.</span><span class="nf">add_customers</span><span class="p">(</span><span class="n">customers</span><span class="p">)</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="# Da add_customers un array de 1000 clientes a la vez.
Customer.find_in_batches do |customers|
export.add_customers(customers)
end
">Copy</button>
</div>
<p><code>find_in_batches</code> funciona en clases de modelo, como se ve arriba, y también en relaciones:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="c1"># Da add_customers un array de 1000 clientes recientemente activos a la vez.</span>
<span class="no">Customer</span><span class="p">.</span><span class="nf">recently_active</span><span class="p">.</span><span class="nf">find_in_batches</span> <span class="k">do</span> <span class="o">|</span><span class="n">customers</span><span class="o">|</span>
<span class="n">export</span><span class="p">.</span><span class="nf">add_customers</span><span class="p">(</span><span class="n">customers</span><span class="p">)</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="# Da add_customers un array de 1000 clientes recientemente activos a la vez.
Customer.recently_active.find_in_batches do |customers|
export.add_customers(customers)
end
">Copy</button>
</div>
<p>siempre que no tengan ordenación, ya que el método necesita forzar un orden internamente para iterar.</p><h5 id="opciones-para-find-in-batches"><a class="anchorlink" href="#opciones-para-find-in-batches"><span>2.2.2.1</span> Opciones para <code>find_in_batches</code></a></h5><p>El método <code>find_in_batches</code> acepta las mismas opciones que <code>find_each</code>:</p><p><strong><code>:batch_size</code></strong></p><p>Al igual que para <code>find_each</code>, <code>batch_size</code> establece cuántos registros se recuperarán en cada grupo. Por ejemplo, recuperar lotes de 2500 registros se puede especificar como:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_in_batches</span><span class="p">(</span><span class="ss">batch_size: </span><span class="mi">2500</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customers</span><span class="o">|</span>
<span class="n">export</span><span class="p">.</span><span class="nf">add_customers</span><span class="p">(</span><span class="n">customers</span><span class="p">)</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_in_batches(batch_size: 2500) do |customers|
export.add_customers(customers)
end
">Copy</button>
</div>
<p><strong><code>:start</code></strong></p><p>La opción <code>start</code> permite especificar el ID inicial desde donde se seleccionarán los registros. Como se mencionó antes, por defecto los registros se recuperan en orden ascendente de la clave primaria. Por ejemplo, para recuperar clientes comenzando en ID: 5000 en lotes de 2500 registros, se puede usar el siguiente código:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_in_batches</span><span class="p">(</span><span class="ss">batch_size: </span><span class="mi">2500</span><span class="p">,</span> <span class="ss">start: </span><span class="mi">5000</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customers</span><span class="o">|</span>
<span class="n">export</span><span class="p">.</span><span class="nf">add_customers</span><span class="p">(</span><span class="n">customers</span><span class="p">)</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_in_batches(batch_size: 2500, start: 5000) do |customers|
export.add_customers(customers)
end
">Copy</button>
</div>
<p><strong><code>:finish</code></strong></p><p>La opción <code>finish</code> permite especificar el ID final de los registros a recuperar. El código a continuación muestra el caso de recuperar clientes en lotes, hasta el cliente con ID: 7000:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">find_in_batches</span><span class="p">(</span><span class="ss">finish: </span><span class="mi">7000</span><span class="p">)</span> <span class="k">do</span> <span class="o">|</span><span class="n">customers</span><span class="o">|</span>
<span class="n">export</span><span class="p">.</span><span class="nf">add_customers</span><span class="p">(</span><span class="n">customers</span><span class="p">)</span>
<span class="k">end</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.find_in_batches(finish: 7000) do |customers|
export.add_customers(customers)
end
">Copy</button>
</div>
<p><strong><code>:error_on_ignore</code></strong></p><p>La opción <code>error_on_ignore</code> anula la configuración de la aplicación para especificar si se debe generar un error cuando hay un orden específico presente en la relación.</p><h2 id="condiciones"><a class="anchorlink" href="#condiciones"><span>3</span> Condiciones</a></h2><p>El método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-where"><code>where</code></a> te permite especificar condiciones para limitar los registros devueltos, representando la parte <code>WHERE</code> de la declaración SQL. Las condiciones se pueden especificar como una cadena, un array o un hash.</p><h3 id="condiciones-de-cadena-pura"><a class="anchorlink" href="#condiciones-de-cadena-pura"><span>3.1</span> Condiciones de Cadena Pura</a></h3><p>Si deseas agregar condiciones a tu búsqueda, podrías simplemente especificarlas allí, como <code>Book.where("title = 'Introduction to Algorithms'")</code>. Esto encontrará todos los libros donde el valor del campo <code>title</code> sea 'Introduction to Algorithms'.</p><p>ADVERTENCIA: Construir tus propias condiciones como cadenas puras puede dejarte vulnerable a exploits de inyección SQL. Por ejemplo, <code>Book.where("title LIKE '%#{params[:title]}%'")</code> no es seguro. Consulta la siguiente sección para conocer la forma preferida de manejar condiciones usando un array.</p><h3 id="condiciones-de-array"><a class="anchorlink" href="#condiciones-de-array"><span>3.2</span> Condiciones de Array</a></h3><p>Ahora, ¿qué pasa si ese título podría variar, digamos como un argumento de algún lugar? La búsqueda entonces tomaría la forma:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"title = ?"</span><span class="p">,</span> <span class="n">params</span><span class="p">[</span><span class="ss">:title</span><span class="p">])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("title = ?", params[:title])
">Copy</button>
</div>
<p>Active Record tomará el primer argumento como la cadena de condiciones y cualquier argumento adicional reemplazará los signos de interrogación <code>(?)</code> en ella.</p><p>Si deseas especificar múltiples condiciones:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"title = ? AND out_of_print = ?"</span><span class="p">,</span> <span class="n">params</span><span class="p">[</span><span class="ss">:title</span><span class="p">],</span> <span class="kp">false</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("title = ? AND out_of_print = ?", params[:title], false)
">Copy</button>
</div>
<p>En este ejemplo, el primer signo de interrogación será reemplazado con el valor en <code>params[:title]</code> y el segundo será reemplazado con la representación SQL de <code>false</code>, que depende del adaptador.</p><p>Este código es altamente preferible:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"title = ?"</span><span class="p">,</span> <span class="n">params</span><span class="p">[</span><span class="ss">:title</span><span class="p">])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("title = ?", params[:title])
">Copy</button>
</div>
<p>a este código:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"title = </span><span class="si">#{</span><span class="n">params</span><span class="p">[</span><span class="ss">:title</span><span class="p">]</span><span class="si">}</span><span class="s2">"</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("title = #{params[:title]}")
">Copy</button>
</div>
<p>debido a la seguridad de los argumentos. Poner la variable directamente en la cadena de condiciones pasará la variable a la base de datos <strong>tal cual</strong>. Esto significa que será una variable no escapada directamente de un usuario que podría tener intenciones maliciosas. Si haces esto, pones en riesgo toda tu base de datos porque una vez que un usuario descubre que puede explotar tu base de datos, puede hacer casi cualquier cosa con ella. Nunca pongas tus argumentos directamente dentro de la cadena de condiciones.</p><p>CONSEJO: Para obtener más información sobre los peligros de la inyección SQL, consulta la <a href="security.html#sql-injection">Guía de Seguridad de Ruby on Rails</a>.</p><h4 id="condiciones-de-marcadores-de-posición"><a class="anchorlink" href="#condiciones-de-marcadores-de-posición"><span>3.2.1</span> Condiciones de Marcadores de Posición</a></h4><p>Similar al estilo de reemplazo <code>(?)</code> de params, también puedes especificar claves en tu cadena de condiciones junto con un hash de claves/valores correspondiente:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"created_at >= :start_date AND created_at <= :end_date"</span><span class="p">,</span>
<span class="p">{</span> <span class="ss">start_date: </span><span class="n">params</span><span class="p">[</span><span class="ss">:start_date</span><span class="p">],</span> <span class="ss">end_date: </span><span class="n">params</span><span class="p">[</span><span class="ss">:end_date</span><span class="p">]</span> <span class="p">})</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("created_at >= :start_date AND created_at <= :end_date",
{ start_date: params[:start_date], end_date: params[:end_date] })
">Copy</button>
</div>
<p>Esto hace que la legibilidad sea más clara si tienes un gran número de condiciones variables.</p><h4 id="condiciones-que-usan-like"><a class="anchorlink" href="#condiciones-que-usan-like"><span>3.2.2</span> Condiciones que Usan <code>LIKE</code></a></h4><p>Aunque los argumentos de condición se escapan automáticamente para prevenir inyecciones SQL, los comodines SQL <code>LIKE</code> (es decir, <code>%</code> y <code>_</code>) <strong>no</strong> se escapan. Esto puede causar un comportamiento inesperado si se utiliza un valor no sanitizado en un argumento. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"title LIKE ?"</span><span class="p">,</span> <span class="n">params</span><span class="p">[</span><span class="ss">:title</span><span class="p">]</span> <span class="o">+</span> <span class="s2">"%"</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("title LIKE ?", params[:title] + "%")
">Copy</button>
</div>
<p>En el código anterior, la intención es coincidir con títulos que comienzan con una cadena especificada por el usuario. Sin embargo, cualquier aparición de <code>%</code> o <code>_</code> en <code>params[:title]</code> será tratada como comodines, lo que lleva a resultados de consulta sorprendentes. En algunas circunstancias, esto también puede prevenir que la base de datos use un índice previsto, lo que lleva a una consulta mucho más lenta.</p><p>Para evitar estos problemas, utiliza <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/Sanitization/ClassMethods.html#method-i-sanitize_sql_like"><code>sanitize_sql_like</code></a> para escapar los caracteres comodín en la parte relevante del argumento:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s2">"title LIKE ?"</span><span class="p">,</span>
<span class="no">Book</span><span class="p">.</span><span class="nf">sanitize_sql_like</span><span class="p">(</span><span class="n">params</span><span class="p">[</span><span class="ss">:title</span><span class="p">])</span> <span class="o">+</span> <span class="s2">"%"</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where("title LIKE ?",
Book.sanitize_sql_like(params[:title]) + "%")
">Copy</button>
</div>
<h3 id="condiciones-de-hash"><a class="anchorlink" href="#condiciones-de-hash"><span>3.3</span> Condiciones de Hash</a></h3><p>Active Record también te permite pasar condiciones en hash, lo que puede aumentar la legibilidad de tu sintaxis de condiciones. Con condiciones de hash, pasas un hash con claves de los campos que deseas calificar y los valores de cómo deseas calificarlos:</p><p>NOTA: Solo son posibles la igualdad, el rango y la verificación de subconjuntos con condiciones de Hash.</p><h4 id="condiciones-de-igualdad"><a class="anchorlink" href="#condiciones-de-igualdad"><span>3.3.1</span> Condiciones de Igualdad</a></h4><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">out_of_print: </span><span class="kp">true</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where(out_of_print: true)
">Copy</button>
</div>
<p>Esto generará SQL como este:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">books</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">books</span><span class="p">.</span><span class="n">out_of_print</span> <span class="o">=</span> <span class="mi">1</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM books WHERE (books.out_of_print = 1)
">Copy</button>
</div>
<p>El nombre del campo también puede ser una cadena:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="s1">'out_of_print'</span> <span class="o">=></span> <span class="kp">true</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where('out_of_print' => true)
">Copy</button>
</div>
<p>En el caso de una relación belongs_to, se puede usar una clave de asociación para especificar el modelo si se utiliza un objeto de Active Record como valor. Este método también funciona con relaciones polimórficas.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="n">author</span> <span class="o">=</span> <span class="no">Author</span><span class="p">.</span><span class="nf">first</span>
<span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">author: </span><span class="n">author</span><span class="p">)</span>
<span class="no">Author</span><span class="p">.</span><span class="nf">joins</span><span class="p">(</span><span class="ss">:books</span><span class="p">).</span><span class="nf">where</span><span class="p">(</span><span class="ss">books: </span><span class="p">{</span> <span class="ss">author: </span><span class="n">author</span> <span class="p">})</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="author = Author.first
Book.where(author: author)
Author.joins(:books).where(books: { author: author })
">Copy</button>
</div>
<p>Las condiciones de hash también se pueden especificar en una sintaxis similar a una tupla, donde la clave es un array de columnas y el valor es un array de tuplas:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">([</span><span class="ss">:author_id</span><span class="p">,</span> <span class="ss">:id</span><span class="p">]</span> <span class="o">=></span> <span class="p">[[</span><span class="mi">15</span><span class="p">,</span> <span class="mi">1</span><span class="p">],</span> <span class="p">[</span><span class="mi">15</span><span class="p">,</span> <span class="mi">2</span><span class="p">]])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where([:author_id, :id] => [[15, 1], [15, 2]])
">Copy</button>
</div>
<p>Esta sintaxis puede ser útil para consultar relaciones donde la tabla usa una clave primaria compuesta:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="k">class</span> <span class="nc">Book</span> <span class="o"><</span> <span class="no">ApplicationRecord</span>
<span class="nb">self</span><span class="p">.</span><span class="nf">primary_key</span> <span class="o">=</span> <span class="p">[</span><span class="ss">:author_id</span><span class="p">,</span> <span class="ss">:id</span><span class="p">]</span>
<span class="k">end</span>
<span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="no">Book</span><span class="p">.</span><span class="nf">primary_key</span> <span class="o">=></span> <span class="p">[[</span><span class="mi">2</span><span class="p">,</span> <span class="mi">1</span><span class="p">],</span> <span class="p">[</span><span class="mi">3</span><span class="p">,</span> <span class="mi">1</span><span class="p">]])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="class Book < ApplicationRecord
self.primary_key = [:author_id, :id]
end
Book.where(Book.primary_key => [[2, 1], [3, 1]])
">Copy</button>
</div>
<h4 id="condiciones-de-rango"><a class="anchorlink" href="#condiciones-de-rango"><span>3.3.2</span> Condiciones de Rango</a></h4><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">created_at: </span><span class="p">(</span><span class="no">Time</span><span class="p">.</span><span class="nf">now</span><span class="p">.</span><span class="nf">midnight</span> <span class="o">-</span> <span class="mi">1</span><span class="p">.</span><span class="nf">day</span><span class="p">)</span><span class="o">..</span><span class="no">Time</span><span class="p">.</span><span class="nf">now</span><span class="p">.</span><span class="nf">midnight</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where(created_at: (Time.now.midnight - 1.day)..Time.now.midnight)
">Copy</button>
</div>
<p>Esto encontrará todos los libros creados ayer usando una declaración SQL <code>BETWEEN</code>:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">books</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">books</span><span class="p">.</span><span class="n">created_at</span> <span class="k">BETWEEN</span> <span class="s1">'2008-12-21 00:00:00'</span> <span class="k">AND</span> <span class="s1">'2008-12-22 00:00:00'</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM books WHERE (books.created_at BETWEEN '2008-12-21 00:00:00' AND '2008-12-22 00:00:00')
">Copy</button>
</div>
<p>Esto demuestra una sintaxis más corta para los ejemplos en <a href="#condiciones-de-array">Condiciones de Array</a></p><p>Los rangos sin inicio y sin fin son compatibles y se pueden usar para construir condiciones de menor/mayor que.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">created_at: </span><span class="p">(</span><span class="no">Time</span><span class="p">.</span><span class="nf">now</span><span class="p">.</span><span class="nf">midnight</span> <span class="o">-</span> <span class="mi">1</span><span class="p">.</span><span class="nf">day</span><span class="p">)</span><span class="o">..</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.where(created_at: (Time.now.midnight - 1.day)..)
">Copy</button>
</div>
<p>Esto generaría SQL como:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">books</span> <span class="k">WHERE</span> <span class="n">books</span><span class="p">.</span><span class="n">created_at</span> <span class="o">>=</span> <span class="s1">'2008-12-21 00:00:00'</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM books WHERE books.created_at >= '2008-12-21 00:00:00'
">Copy</button>
</div>
<h4 id="condiciones-de-subconjunto"><a class="anchorlink" href="#condiciones-de-subconjunto"><span>3.3.3</span> Condiciones de Subconjunto</a></h4><p>Si deseas encontrar registros usando la expresión <code>IN</code>, puedes pasar un array al hash de condiciones:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">orders_count: </span><span class="p">[</span><span class="mi">1</span><span class="p">,</span> <span class="mi">3</span><span class="p">,</span> <span class="mi">5</span><span class="p">])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(orders_count: [1, 3, 5])
">Copy</button>
</div>
<p>Este código generará SQL como este:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">orders_count</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">1</span><span class="p">,</span><span class="mi">3</span><span class="p">,</span><span class="mi">5</span><span class="p">))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.orders_count IN (1,3,5))
">Copy</button>
</div>
<h3 id="condiciones-not"><a class="anchorlink" href="#condiciones-not"><span>3.4</span> Condiciones NOT</a></h3><p>Las consultas SQL <code>NOT</code> se pueden construir con <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods/WhereChain.html#method-i-not"><code>where.not</code></a>:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">.</span><span class="nf">not</span><span class="p">(</span><span class="ss">orders_count: </span><span class="p">[</span><span class="mi">1</span><span class="p">,</span> <span class="mi">3</span><span class="p">,</span> <span class="mi">5</span><span class="p">])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where.not(orders_count: [1, 3, 5])
">Copy</button>
</div>
<p>En otras palabras, esta consulta se puede generar llamando a <code>where</code> sin argumento, luego encadenando inmediatamente con <code>not</code> pasando condiciones de <code>where</code>. Esto generará SQL como este:</p><div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">orders_count</span> <span class="k">NOT</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">1</span><span class="p">,</span><span class="mi">3</span><span class="p">,</span><span class="mi">5</span><span class="p">))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.orders_count NOT IN (1,3,5))
">Copy</button>
</div>
<p>Si una consulta tiene una condición de hash con valores no nulos en una columna nullable, los registros que tienen valores <code>nil</code> en la columna nullable no serán devueltos. Por ejemplo:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">create!</span><span class="p">(</span><span class="ss">nullable_country: </span><span class="kp">nil</span><span class="p">)</span>
<span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">.</span><span class="nf">not</span><span class="p">(</span><span class="ss">nullable_country: </span><span class="s2">"UK"</span><span class="p">)</span>
<span class="c1"># => []</span>
<span class="c1"># Pero</span>
<span class="no">Customer</span><span class="p">.</span><span class="nf">create!</span><span class="p">(</span><span class="ss">nullable_country: </span><span class="s2">"UK"</span><span class="p">)</span>
<span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">.</span><span class="nf">not</span><span class="p">(</span><span class="ss">nullable_country: </span><span class="kp">nil</span><span class="p">)</span>
<span class="c1"># => [#<Customer id: 2, nullable_country: "UK">]</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.create!(nullable_country: nil)
Customer.where.not(nullable_country: "UK")
# => []
# Pero
Customer.create!(nullable_country: "UK")
Customer.where.not(nullable_country: nil)
# => [#<Customer id: 2, nullable_country: "UK">]
">Copy</button>
</div>
<h3 id="condiciones-or"><a class="anchorlink" href="#condiciones-or"><span>3.5</span> Condiciones OR</a></h3><p>Las condiciones <code>OR</code> entre dos relaciones se pueden construir llamando a <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-or"><code>or</code></a> en la primera
relación, y pasando la segunda como argumento.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">last_name: </span><span class="s1">'Smith'</span><span class="p">).</span><span class="nf">or</span><span class="p">(</span><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">orders_count: </span><span class="p">[</span><span class="mi">1</span><span class="p">,</span> <span class="mi">3</span><span class="p">,</span> <span class="mi">5</span><span class="p">]))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(last_name: 'Smith').or(Customer.where(orders_count: [1, 3, 5]))
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">last_name</span> <span class="o">=</span> <span class="s1">'Smith'</span> <span class="k">OR</span> <span class="n">customers</span><span class="p">.</span><span class="n">orders_count</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">1</span><span class="p">,</span><span class="mi">3</span><span class="p">,</span><span class="mi">5</span><span class="p">))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.last_name = 'Smith' OR customers.orders_count IN (1,3,5))
">Copy</button>
</div>
<h3 id="condiciones-and"><a class="anchorlink" href="#condiciones-and"><span>3.6</span> Condiciones AND</a></h3><p>Las condiciones <code>AND</code> se pueden construir encadenando condiciones <code>where</code>.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">last_name: </span><span class="s1">'Smith'</span><span class="p">).</span><span class="nf">where</span><span class="p">(</span><span class="ss">orders_count: </span><span class="p">[</span><span class="mi">1</span><span class="p">,</span> <span class="mi">3</span><span class="p">,</span> <span class="mi">5</span><span class="p">])</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(last_name: 'Smith').where(orders_count: [1, 3, 5])
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="n">customers</span><span class="p">.</span><span class="n">last_name</span> <span class="o">=</span> <span class="s1">'Smith'</span> <span class="k">AND</span> <span class="n">customers</span><span class="p">.</span><span class="n">orders_count</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">1</span><span class="p">,</span><span class="mi">3</span><span class="p">,</span><span class="mi">5</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE customers.last_name = 'Smith' AND customers.orders_count IN (1,3,5)
">Copy</button>
</div>
<p>Las condiciones <code>AND</code> para la intersección lógica entre relaciones se pueden construir llamando a <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-and"><code>and</code></a> en la primera relación, y pasando la segunda como argumento.</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">id: </span><span class="p">[</span><span class="mi">1</span><span class="p">,</span> <span class="mi">2</span><span class="p">]).</span><span class="nf">and</span><span class="p">(</span><span class="no">Customer</span><span class="p">.</span><span class="nf">where</span><span class="p">(</span><span class="ss">id: </span><span class="p">[</span><span class="mi">2</span><span class="p">,</span> <span class="mi">3</span><span class="p">]))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Customer.where(id: [1, 2]).and(Customer.where(id: [2, 3]))
">Copy</button>
</div>
<div class="interstitial code">
<pre><code class="highlight sql"><span class="k">SELECT</span> <span class="o">*</span> <span class="k">FROM</span> <span class="n">customers</span> <span class="k">WHERE</span> <span class="p">(</span><span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">1</span><span class="p">,</span> <span class="mi">2</span><span class="p">)</span> <span class="k">AND</span> <span class="n">customers</span><span class="p">.</span><span class="n">id</span> <span class="k">IN</span> <span class="p">(</span><span class="mi">2</span><span class="p">,</span> <span class="mi">3</span><span class="p">))</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="SELECT * FROM customers WHERE (customers.id IN (1, 2) AND customers.id IN (2, 3))
">Copy</button>
</div>
<h2 id="ordenamiento"><a class="anchorlink" href="#ordenamiento"><span>4</span> Ordenamiento</a></h2><p>Para recuperar registros de la base de datos en un orden específico, puedes usar el método <a href="https://edgeapi.rubyonrails.org/classes/ActiveRecord/QueryMethods.html#method-i-order"><code>order</code></a>.</p><p>Por ejemplo, si estás obteniendo un conjunto de registros y deseas ordenarlos en orden ascendente por el campo <code>created_at</code> en tu tabla:</p><div class="interstitial code">
<pre><code class="highlight ruby"><span class="no">Book</span><span class="p">.</span><span class="nf">order</span><span class="p">(</span><span class="ss">:created_at</span><span class="p">)</span>
<span class="c1"># O</span>
<span class="no">Book</span><span class="p">.</span><span class="nf">order</span><span class="p">(</span><span class="s2">"created_at"</span><span class="p">)</span>
</code></pre>
<button class="clipboard-button" data-clipboard-text="Book.order(:created_at)
# O
Book.order("created_at")
">Copy</button>