Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 66d0262

Browse files
committedApr 22, 2016
Fix type instability
Until the number of elements passed via varargs is inferred, we need to use a wrapper function. Also remove access to lev[i] from anonymous function, which is currently buggy on 0.5 (JuliaLang/julia#15276).
1 parent 059afbb commit 66d0262

File tree

1 file changed

+23
-18
lines changed

1 file changed

+23
-18
lines changed
 

‎src/freqtable.jl

+23-18
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,11 @@
11
import Base.ht_keyindex
22

3-
function freqtable(x::AbstractVector...;
4-
# Parametric unions are currently not supported for keyword arguments,
5-
# so weights are restricted to Float64 for now
6-
# https://github.com/JuliaLang/julia/issues/3738
7-
weights::Union{Void, AbstractVector{Float64}} = nothing,
8-
subset::Union{Void, AbstractVector{Int}, AbstractVector{Bool}} = nothing)
9-
n = length(x)
10-
3+
# Internal function needed for now so that n is inferred
4+
function _freqtable{n}(x::NTuple{n},
5+
weights::Union{Void, AbstractVector{Float64}} = nothing,
6+
subset::Union{Void, AbstractVector{Int}, AbstractVector{Bool}} = nothing)
117
if subset != nothing
12-
x = ntuple(i -> x[i][subset], n)
8+
x = map(y -> y[subset], x)
139

1410
if weights != nothing
1511
weights = weights[subset]
@@ -78,20 +74,27 @@ function freqtable(x::AbstractVector...;
7874
na
7975
end
8076

81-
function freqtable(x::PooledDataVector...; usena::Bool = false)
82-
n = length(x)
83-
len = [length(y) for y in x]
77+
freqtable(x::AbstractVector...;
78+
# Parametric unions are currently not supported for keyword arguments,
79+
# so weights are restricted to Float64 for now
80+
# https://github.com/JuliaLang/julia/issues/3738
81+
weights::Union{Void, AbstractVector{Float64}} = nothing,
82+
subset::Union{Void, AbstractVector{Int}, AbstractVector{Bool}} = nothing) =
83+
_freqtable(x, weights, subset)
84+
85+
# Internal function needed for now so that n is inferred
86+
function _freqtable{n}(x::NTuple{n, PooledDataVector}, usena = false)
87+
len = map(length, x)
88+
lev = map(levels, x)
8489

8590
for i in 1:n
8691
if len[1] != len[i]
8792
error(string("arguments are not of the same length: ", tuple(len...)))
8893
end
8994
end
9095

91-
lev = [levels(y) for y in x]
92-
9396
if usena
94-
dims = ntuple(i -> length(lev[i]) + 1, n)
97+
dims = map(l -> length(l) + 1, lev)
9598
sizes = cumprod([dims...])
9699
a = zeros(Int, dims)
97100

@@ -115,9 +118,9 @@ function freqtable(x::PooledDataVector...; usena::Bool = false)
115118
a[el] += 1
116119
end
117120

118-
NamedArray(a, ntuple(i -> [lev[i], "NA"], n), ntuple(i -> "Dim$i", n))
121+
NamedArray(a, map(l -> [l; "NA"], lev), ntuple(i -> "Dim$i", n))
119122
else
120-
dims = ntuple(i -> length(lev[i]), n)
123+
dims = map(length, lev)::NTuple{n,Int}
121124
sizes = cumprod([dims...])
122125
a = zeros(Int, dims)
123126

@@ -141,10 +144,12 @@ function freqtable(x::PooledDataVector...; usena::Bool = false)
141144
end
142145
end
143146

144-
NamedArray(a, ntuple(i -> lev[i], n), ntuple(i -> "Dim$i", n))
147+
NamedArray(a, lev, ntuple(i -> "Dim$i", n))
145148
end
146149
end
147150

151+
freqtable(x::PooledDataVector...; usena::Bool = false) = _freqtable(x, usena)
152+
148153
function freqtable(d::DataFrame, x::Symbol...; args...)
149154
a = freqtable([d[y] for y in x]...; args...)
150155
setdimnames!(a, x)

0 commit comments

Comments
 (0)
Please sign in to comment.