summaryrefslogtreecommitdiffhomepage
path: root/mrbgems/mruby-range-ext
diff options
context:
space:
mode:
authorYukihiro "Matz" Matsumoto <[email protected]>2020-10-16 18:29:29 +0900
committerYukihiro "Matz" Matsumoto <[email protected]>2020-10-29 10:43:08 +0900
commitd35677f213feb3d1a3fe38a763c3dd765e440549 (patch)
tree07341e458e531d85943ec2c92078de2b526b5152 /mrbgems/mruby-range-ext
parent76eafa46082e43b33b107e845ebbfaa6ebae3155 (diff)
downloadmruby-d35677f213feb3d1a3fe38a763c3dd765e440549.tar.gz
mruby-d35677f213feb3d1a3fe38a763c3dd765e440549.zip
Implement `Range#last` in `mruby-range-ext` in Ruby.
This reduce unnecessary calls of `mrb_funcall()`.
Diffstat (limited to 'mrbgems/mruby-range-ext')
-rw-r--r--mrbgems/mruby-range-ext/mrblib/range.rb26
-rw-r--r--mrbgems/mruby-range-ext/src/range.c36
2 files changed, 26 insertions, 36 deletions
diff --git a/mrbgems/mruby-range-ext/mrblib/range.rb b/mrbgems/mruby-range-ext/mrblib/range.rb
index 018bd7094..5eeb802c1 100644
--- a/mrbgems/mruby-range-ext/mrblib/range.rb
+++ b/mrbgems/mruby-range-ext/mrblib/range.rb
@@ -26,6 +26,32 @@ class Range
ary
end
+ ##
+ # call-seq:
+ # rng.last -> obj
+ # rng.last(n) -> an_array
+ #
+ # Returns the last object in the range,
+ # or an array of the last +n+ elements.
+ #
+ # Note that with no arguments +last+ will return the object that defines
+ # the end of the range even if #exclude_end? is +true+.
+ #
+ # (10..20).last #=> 20
+ # (10...20).last #=> 20
+ # (10..20).last(3) #=> [18, 19, 20]
+ # (10...20).last(3) #=> [17, 18, 19]
+ def last(*args)
+ raise RangeError, "cannot get the first element of beginless range" if self.end.nil?
+ return self.end if args.empty?
+
+ raise ArgumentError, "wrong number of arguments (given #{args.length}, expected 1)" unless args.length == 1
+ nv = args[0]
+ n = nv.__to_int
+ raise ArgumentError, "negative array size (or size too big)" unless 0 <= n
+ return self.to_a.last(nv)
+ end
+
def max(&block)
val = self.begin
last = self.end
diff --git a/mrbgems/mruby-range-ext/src/range.c b/mrbgems/mruby-range-ext/src/range.c
index a5651cdd9..7b919eb90 100644
--- a/mrbgems/mruby-range-ext/src/range.c
+++ b/mrbgems/mruby-range-ext/src/range.c
@@ -60,41 +60,6 @@ range_cover(mrb_state *mrb, mrb_value range)
/*
* call-seq:
- * rng.last -> obj
- * rng.last(n) -> an_array
- *
- * Returns the last object in the range,
- * or an array of the last +n+ elements.
- *
- * Note that with no arguments +last+ will return the object that defines
- * the end of the range even if #exclude_end? is +true+.
- *
- * (10..20).last #=> 20
- * (10...20).last #=> 20
- * (10..20).last(3) #=> [18, 19, 20]
- * (10...20).last(3) #=> [17, 18, 19]
- */
-static mrb_value
-range_last(mrb_state *mrb, mrb_value range)
-{
- mrb_value num;
- mrb_value array;
- struct RRange *r = mrb_range_ptr(mrb, range);
-
- if (mrb_nil_p(RANGE_END(r))) {
- mrb_raise(mrb, E_RANGE_ERROR, "cannot get the last element of endless range");
- }
-
- if (mrb_get_args(mrb, "|o", &num) == 0) {
- return mrb_range_end(mrb, range);
- }
-
- array = mrb_funcall_id(mrb, range, MRB_SYM(to_a), 0);
- return mrb_funcall_id(mrb, array, MRB_SYM(last), 1, mrb_to_int(mrb, num));
-}
-
-/*
- * call-seq:
* rng.size -> num
*
* Returns the number of elements in the range. Both the begin and the end of
@@ -194,7 +159,6 @@ mrb_mruby_range_ext_gem_init(mrb_state* mrb)
struct RClass * s = mrb_class_get(mrb, "Range");
mrb_define_method(mrb, s, "cover?", range_cover, MRB_ARGS_REQ(1));
- mrb_define_method(mrb, s, "last", range_last, MRB_ARGS_OPT(1));
mrb_define_method(mrb, s, "size", range_size, MRB_ARGS_NONE());
}