[前][次][番号順一覧][スレッド一覧]

ruby-changes:65233

From: Chris <ko1@a...>
Date: Thu, 11 Feb 2021 19:14:47 +0900 (JST)
Subject: [ruby-changes:65233] c3b2bb0969 (master): The Queue constructor should take an initial set of objects

https://git.ruby-lang.org/ruby.git/commit/?id=c3b2bb0969

From c3b2bb0969cc47dcfb1f624c94a46cdf1e2cc2ad Mon Sep 17 00:00:00 2001
From: Chris Seaton <chris.seaton@s...>
Date: Thu, 11 Feb 2021 10:14:18 +0000
Subject: The Queue constructor should take an initial set of objects

Co-authored-by: Nobuyoshi Nakada <nobu@r...>
---
 NEWS.md                                 |  6 +++++
 spec/ruby/core/queue/initialize_spec.rb | 44 +++++++++++++++++++++++++++++++++
 thread_sync.c                           | 18 +++++++++++---
 3 files changed, 65 insertions(+), 3 deletions(-)
 create mode 100644 spec/ruby/core/queue/initialize_spec.rb

diff --git a/NEWS.md b/NEWS.md
index c12f3e8..10623a1 100644
--- a/NEWS.md
+++ b/NEWS.md
@@ -34,6 +34,11 @@ Outstanding ones only. https://github.com/ruby/ruby/blob/trunk/NEWS.md#L34
       You need to use a Hash literal to set a Hash to a first member.
       [[Feature #16806]]
 
+* Queue
+
+    * Queue#initialize now accepts an Enumerable of initial values.
+      [[Feature #17327]]
+
 ## Stdlib updates
 
 Outstanding ones only.
@@ -67,4 +72,5 @@ Excluding feature bug fixes. https://github.com/ruby/ruby/blob/trunk/NEWS.md#L72
 [Feature #14256]: https://bugs.ruby-lang.org/issues/14256
 [Feature #16806]: https://bugs.ruby-lang.org/issues/16806
 [Feature #17312]: https://bugs.ruby-lang.org/issues/17312
+[Feature #17327]: https://bugs.ruby-lang.org/issues/17327
 [Bug #17423]: https://bugs.ruby-lang.org/issues/17423
diff --git a/spec/ruby/core/queue/initialize_spec.rb b/spec/ruby/core/queue/initialize_spec.rb
new file mode 100644
index 0000000..5cecfe5
--- /dev/null
+++ b/spec/ruby/core/queue/initialize_spec.rb
@@ -0,0 +1,44 @@ https://github.com/ruby/ruby/blob/trunk/spec/ruby/core/queue/initialize_spec.rb#L1
+require_relative '../../spec_helper'
+
+describe "Queue#initialize" do
+  it "can be passed no arguments for an empty Queue" do
+    q = Queue.new
+    q.size.should == 0
+    q.should.empty?
+  end
+
+  ruby_version_is '3.1' do
+    it "adds all elements of the passed Enumerable to self" do
+      q = Queue.new([1, 2, 3])
+      q.size.should == 3
+      q.should_not.empty?
+      q.pop.should == 1
+      q.pop.should == 2
+      q.pop.should == 3
+      q.should.empty?
+    end
+
+    it "uses #to_ary on the provided Enumerable" do
+      enumerable = MockObject.new('mock-enumerable')
+      enumerable.should_receive(:to_ary).and_return([1, 2, 3])
+      q = Queue.new(enumerable)
+      q.size.should == 3
+      q.should_not.empty?
+      q.pop.should == 1
+      q.pop.should == 2
+      q.pop.should == 3
+      q.should.empty?
+    end
+
+    it "raises if the provided Enumerable does not respond to #to_ary" do
+      enumerable = MockObject.new('mock-enumerable')
+      -> { Queue.new(enumerable) }.should raise_error(TypeError, "no implicit conversion of MockObject into Array")
+    end
+
+    it "raises if the provided Enumerable #to_ary does not return an Array" do
+      enumerable = MockObject.new('mock-enumerable')
+      enumerable.should_receive(:to_ary).and_return(14)
+      -> { Queue.new(enumerable) }.should raise_error(TypeError, "can't convert MockObject to Array (MockObject#to_ary gives Integer)")
+    end
+  end
+end
diff --git a/thread_sync.c b/thread_sync.c
index 9932abd..131ace2 100644
--- a/thread_sync.c
+++ b/thread_sync.c
@@ -839,15 +839,27 @@ queue_closed_result(VALUE self, struct rb_queue *q) https://github.com/ruby/ruby/blob/trunk/thread_sync.c#L839
 /*
  * Document-method: Queue::new
  *
- * Creates a new queue instance.
+ * Creates a new queue instance, optionally using the contents of an Enumerable
+ * for its initial state.
+ *
+ *  Example:
+ *
+ *    	q = Queue.new
+ *    	q = Queue.new([a, b, c])
+ *    	q = Queue.new(items)
  */
 
 static VALUE
-rb_queue_initialize(VALUE self)
+rb_queue_initialize(int argc, VALUE *argv, VALUE self)
 {
+    VALUE initial;
     struct rb_queue *q = queue_ptr(self);
     RB_OBJ_WRITE(self, &q->que, ary_buf_new());
     list_head_init(queue_waitq(q));
+    rb_scan_args(argc, argv, "01", &initial);
+    if (argc == 1) {
+        rb_ary_concat(q->que, rb_convert_type(initial, T_ARRAY, "Array", "to_ary"));
+    }
     return self;
 }
 
@@ -1570,7 +1582,7 @@ Init_thread_sync(void) https://github.com/ruby/ruby/blob/trunk/thread_sync.c#L1582
 
     rb_eClosedQueueError = rb_define_class("ClosedQueueError", rb_eStopIteration);
 
-    rb_define_method(rb_cQueue, "initialize", rb_queue_initialize, 0);
+    rb_define_method(rb_cQueue, "initialize", rb_queue_initialize, -1);
     rb_undef_method(rb_cQueue, "initialize_copy");
     rb_define_method(rb_cQueue, "marshal_dump", undumpable, 0);
     rb_define_method(rb_cQueue, "close", rb_queue_close, 0);
-- 
cgit v1.1


--
ML: ruby-changes@q...
Info: http://www.atdot.net/~ko1/quickml/

[前][次][番号順一覧][スレッド一覧]