summaryrefslogtreecommitdiff
path: root/array.c
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2009-02-05 03:01:55 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2009-02-05 03:01:55 +0000
commite3c8cdca4413219dccc57eec6d898cac6851a5db (patch)
tree07ce2b808ca92d81693aa3c17222ecda614d89f6 /array.c
parent8b093ebc3c091edf3bf15ae7061ea3bba753f2fc (diff)
* array.c (ary_add_hash): split from ary_make_hash().
* array.c (ary_recycle_hash): clears internally used hash. this name came from [ruby-dev:37908]. * array.c (rb_ary_diff, rb_ary_and, rb_ary_or, rb_ary_uniq_bang): recycle hashes. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@22072 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'array.c')
-rw-r--r--array.c44
1 files changed, 30 insertions, 14 deletions
diff --git a/array.c b/array.c
index ddfe3c2b2b..be05f0a6b7 100644
--- a/array.c
+++ b/array.c
@@ -2864,23 +2864,35 @@ rb_ary_cmp(VALUE ary1, VALUE ary2)
}
static VALUE
-ary_make_hash(VALUE ary1, VALUE ary2)
+ary_add_hash(VALUE hash, VALUE ary)
{
- VALUE hash = rb_hash_new();
long i;
- RBASIC(hash)->klass = 0;
- for (i=0; i<RARRAY_LEN(ary1); i++) {
- rb_hash_aset(hash, RARRAY_PTR(ary1)[i], Qtrue);
- }
- if (ary2) {
- for (i=0; i<RARRAY_LEN(ary2); i++) {
- rb_hash_aset(hash, RARRAY_PTR(ary2)[i], Qtrue);
- }
+ for (i=0; i<RARRAY_LEN(ary); i++) {
+ rb_hash_aset(hash, RARRAY_PTR(ary)[i], Qtrue);
}
return hash;
}
+static VALUE
+ary_make_hash(VALUE ary)
+{
+ VALUE hash = rb_hash_new();
+
+ RBASIC(hash)->klass = 0;
+ return ary_add_hash(hash, ary);
+}
+
+static inline void
+ary_recycle_hash(VALUE hash)
+{
+ if (RHASH(hash)->ntbl) {
+ st_table *tbl = RHASH(hash)->ntbl;
+ RHASH(hash)->ntbl = 0;
+ st_free_table(tbl);
+ }
+}
+
/*
* call-seq:
* array - other_array -> an_array
@@ -2900,13 +2912,14 @@ rb_ary_diff(VALUE ary1, VALUE ary2)
volatile VALUE hash;
long i;
- hash = ary_make_hash(to_ary(ary2), 0);
+ hash = ary_make_hash(to_ary(ary2));
ary3 = rb_ary_new();
for (i=0; i<RARRAY_LEN(ary1); i++) {
if (st_lookup(RHASH_TBL(hash), RARRAY_PTR(ary1)[i], 0)) continue;
rb_ary_push(ary3, rb_ary_elt(ary1, i));
}
+ ary_recycle_hash(hash);
return ary3;
}
@@ -2930,7 +2943,7 @@ rb_ary_and(VALUE ary1, VALUE ary2)
ary2 = to_ary(ary2);
ary3 = rb_ary_new2(RARRAY_LEN(ary1) < RARRAY_LEN(ary2) ?
RARRAY_LEN(ary1) : RARRAY_LEN(ary2));
- hash = ary_make_hash(ary2, 0);
+ hash = ary_make_hash(ary2);
if (RHASH_EMPTY_P(hash))
return ary3;
@@ -2941,6 +2954,7 @@ rb_ary_and(VALUE ary1, VALUE ary2)
rb_ary_push(ary3, v);
}
}
+ ary_recycle_hash(hash);
return ary3;
}
@@ -2965,7 +2979,7 @@ rb_ary_or(VALUE ary1, VALUE ary2)
ary2 = to_ary(ary2);
ary3 = rb_ary_new2(RARRAY_LEN(ary1)+RARRAY_LEN(ary2));
- hash = ary_make_hash(ary1, ary2);
+ hash = ary_add_hash(ary_make_hash(ary1), ary2);
for (i=0; i<RARRAY_LEN(ary1); i++) {
v = vv = rb_ary_elt(ary1, i);
@@ -2979,6 +2993,7 @@ rb_ary_or(VALUE ary1, VALUE ary2)
rb_ary_push(ary3, v);
}
}
+ ary_recycle_hash(hash);
return ary3;
}
@@ -3002,7 +3017,7 @@ rb_ary_uniq_bang(VALUE ary)
VALUE hash, v, vv;
long i, j;
- hash = ary_make_hash(ary, 0);
+ hash = ary_make_hash(ary);
if (RARRAY_LEN(ary) == RHASH_SIZE(hash)) {
return Qnil;
@@ -3014,6 +3029,7 @@ rb_ary_uniq_bang(VALUE ary)
}
}
ARY_SET_LEN(ary, j);
+ ary_recycle_hash(hash);
return ary;
}