Skip to content

Commit e8d02ea

Browse files
committed
Add docstrings for accumulation methods on vectors
Fix bug in accumulate when initial value is provided since the implementation doesn't support multidimentional arrays. Move the accumulate methods with initial values to the end
1 parent 491b18d commit e8d02ea

File tree

1 file changed

+138
-62
lines changed

1 file changed

+138
-62
lines changed

base/multidimensional.jl

+138-62
Original file line numberDiff line numberDiff line change
@@ -719,6 +719,27 @@ function cumsum(A::AbstractArray{T}, axis::Integer) where T
719719
cumsum!(out, A, axis)
720720
end
721721

722+
"""
723+
cumsum(x::AbstractVector)
724+
725+
Cumulative sum a vector. See also [`cumsum!`](@ref)
726+
to use a preallocated output array, both for performance and to control the precision of the
727+
output (e.g. to avoid overflow).
728+
729+
```jldoctest
730+
julia> cumsum([1, 1, 1])
731+
3-element Array{Int64,1}:
732+
1
733+
2
734+
3
735+
736+
julia> cumsum([fill(1, 2) for i in 1:3])
737+
3-element Array{Array{Int64,1},1}:
738+
[1, 1]
739+
[2, 2]
740+
[3, 3]
741+
```
742+
"""
722743
cumsum(x::AbstractVector) = cumsum(x, 1)
723744

724745
"""
@@ -728,6 +749,11 @@ Cumulative sum of `A` along a dimension, storing the result in `B`. See also [`c
728749
"""
729750
cumsum!(B, A, axis::Integer) = accumulate!(+, B, A, axis)
730751

752+
"""
753+
cumsum!(y::AbstractVector, x::AbstractVector)
754+
755+
Cumulative sum of a vector `x`, storing the result in `y`. See also [`cumsum`](@ref).
756+
"""
731757
cumsum!(y::AbstractVector, x::AbstractVector) = cumsum!(y, x, 1)
732758

733759
"""
@@ -756,6 +782,27 @@ julia> cumprod(a,2)
756782
"""
757783
cumprod(A::AbstractArray, axis::Integer) = accumulate(*, A, axis)
758784

785+
"""
786+
cumprod(x::AbstractVector)
787+
788+
Cumulative product of a vector. See also
789+
[`cumprod!`](@ref) to use a preallocated output array, both for performance and
790+
to control the precision of the output (e.g. to avoid overflow).
791+
792+
```jldoctest
793+
julia> cumprod(fill(1//2, 3))
794+
3-element Array{Rational{Int64},1}:
795+
1//2
796+
1//4
797+
1//8
798+
799+
julia> cumprod([fill(1//3, 2, 2) for i in 1:3])
800+
3-element Array{Array{Rational{Int64},2},1}:
801+
Rational{Int64}[1//3 1//3; 1//3 1//3]
802+
Rational{Int64}[2//9 2//9; 2//9 2//9]
803+
Rational{Int64}[4//27 4//27; 4//27 4//27]
804+
```
805+
"""
759806
cumprod(x::AbstractVector) = cumprod(x, 1)
760807

761808
"""
@@ -766,6 +813,12 @@ See also [`cumprod`](@ref).
766813
"""
767814
cumprod!(B, A, axis::Integer) = accumulate!(*, B, A, axis)
768815

816+
"""
817+
cumprod!(y::AbstractVector, x::AbstractVector)
818+
819+
Cumulative product of a vector `x`, storing the result in `y`.
820+
See also [`cumprod`](@ref).
821+
"""
769822
cumprod!(y::AbstractVector, x::AbstractVector) = cumprod!(y, x, 1)
770823

771824
"""
@@ -778,83 +831,48 @@ there are specialized variants of `accumulate`, see:
778831
[`cumsum`](@ref), [`cumprod`](@ref)
779832
780833
```jldoctest
781-
julia> accumulate(+, [1,2,3])
782-
3-element Array{Int64,1}:
783-
1
784-
3
785-
6
786-
787-
julia> accumulate(*, [1,2,3])
788-
3-element Array{Int64,1}:
789-
1
790-
2
791-
6
834+
julia> accumulate(+, fill(1, 3, 3), 1)
835+
3×3 Array{Int64,2}:
836+
1 1 1
837+
2 2 2
838+
3 3 3
839+
840+
julia> accumulate(+, fill(1, 3, 3), 2)
841+
3×3 Array{Int64,2}:
842+
1 2 3
843+
1 2 3
844+
1 2 3
792845
```
793846
"""
794847
function accumulate(op, A, axis::Integer)
795848
out = similar(A, rcum_promote_type(op, eltype(A)))
796849
accumulate!(op, out, A, axis)
797850
end
798851

799-
accumulate(op, x::AbstractVector) = accumulate(op, x, 1)
800-
801852
"""
802-
accumulate(op, v0, A)
853+
accumulate(op, x::AbstractVector)
803854
804-
Like `accumulate`, but using a starting element `v0`. The first entry of the result will be
805-
`op(v0, first(A))`.
855+
Cumulative operation `op` on a vector. See also
856+
[`accumulate!`](@ref) to use a preallocated output array, both for performance and
857+
to control the precision of the output (e.g. to avoid overflow). For common operations
858+
there are specialized variants of `accumulate`, see:
859+
[`cumsum`](@ref), [`cumprod`](@ref)
806860
807-
# Examples
808861
```jldoctest
809-
julia> accumulate(+, 100, [1,2,3])
862+
julia> accumulate(+, [1,2,3])
810863
3-element Array{Int64,1}:
811-
101
812-
103
813-
106
864+
1
865+
3
866+
6
814867
815-
julia> accumulate(min, 0, [1,2,-1])
868+
julia> accumulate(*, [1,2,3])
816869
3-element Array{Int64,1}:
817-
0
818-
0
819-
-1
870+
1
871+
2
872+
6
820873
```
821874
"""
822-
function accumulate(op, v0, A, axis::Integer)
823-
T = rcum_promote_type(op, typeof(v0), eltype(A))
824-
out = similar(A, T)
825-
accumulate!(op, out, v0, A, 1)
826-
end
827-
828-
accumulate(op, v0, x::AbstractVector) = accumulate(op, v0, x, 1)
829-
830-
function accumulate!(op::Op, B, A::AbstractVector, axis::Integer) where Op
831-
isempty(A) && return B
832-
v1 = first(A)
833-
_accumulate1!(op, B, v1, A, axis)
834-
end
835-
836-
function accumulate!(op, B, v0, A::AbstractVector, axis::Integer)
837-
isempty(A) && return B
838-
v1 = op(v0, first(A))
839-
_accumulate1!(op, B, v1, A, axis)
840-
end
841-
842-
accumulate!(op, y::AbstractVector, v0, x::AbstractVector) = accumulate!(op, y, v0, x, 1)
843-
844-
function _accumulate1!(op, B, v1, A::AbstractVector, axis::Integer)
845-
axis > 0 || throw(ArgumentError("axis must be a positive integer"))
846-
inds = linearindices(A)
847-
inds == linearindices(B) || throw(DimensionMismatch("linearindices of A and B don't match"))
848-
axis > 1 && return copy!(B, A)
849-
i1 = inds[1]
850-
cur_val = v1
851-
B[i1] = cur_val
852-
@inbounds for i in inds[2:end]
853-
cur_val = op(cur_val, A[i])
854-
B[i] = cur_val
855-
end
856-
return B
857-
end
875+
accumulate(op, x::AbstractVector) = accumulate(op, x, 1)
858876

859877
"""
860878
accumulate!(op, B, A, dim)
@@ -888,7 +906,17 @@ function accumulate!(op, B, A, axis::Integer)
888906
return B
889907
end
890908

891-
accumulate!(op, y::AbstractVector, x::AbstractVector) = accumulate!(op, y, x, 1)
909+
"""
910+
accumulate!(op, y, x::AbstractVector)
911+
912+
Cumulative operation `op` on a vector `x`, storing the result in `y`.
913+
See also [`accumulate`](@ref).
914+
"""
915+
function accumulate!(op::Op, y, x::AbstractVector) where Op
916+
isempty(x) && return y
917+
v1 = first(x)
918+
_accumulate1!(op, y, v1, x, 1)
919+
end
892920

893921
@noinline function _accumulate!(op, B, A, R1, ind, R2)
894922
# Copy the initial element in each 1d vector along dimension `axis`
@@ -903,6 +931,54 @@ accumulate!(op, y::AbstractVector, x::AbstractVector) = accumulate!(op, y, x, 1)
903931
B
904932
end
905933

934+
"""
935+
accumulate(op, v0, x::AbstractVector)
936+
937+
Like `accumulate`, but using a starting element `v0`. The first entry of the result will be
938+
`op(v0, first(A))`.
939+
940+
# Examples
941+
```jldoctest
942+
julia> accumulate(+, 100, [1,2,3])
943+
3-element Array{Int64,1}:
944+
101
945+
103
946+
106
947+
948+
julia> accumulate(min, 0, [1,2,-1])
949+
3-element Array{Int64,1}:
950+
0
951+
0
952+
-1
953+
```
954+
"""
955+
function accumulate(op, v0, x::AbstractVector)
956+
T = rcum_promote_type(op, typeof(v0), eltype(x))
957+
out = similar(x, T)
958+
accumulate!(op, out, v0, x)
959+
end
960+
961+
function accumulate!(op, y, v0, x::AbstractVector)
962+
isempty(x) && return y
963+
v1 = op(v0, first(x))
964+
_accumulate1!(op, y, v1, x, 1)
965+
end
966+
967+
function _accumulate1!(op, B, v1, A::AbstractVector, axis::Integer)
968+
axis > 0 || throw(ArgumentError("axis must be a positive integer"))
969+
inds = linearindices(A)
970+
inds == linearindices(B) || throw(DimensionMismatch("linearindices of A and B don't match"))
971+
axis > 1 && return copy!(B, A)
972+
i1 = inds[1]
973+
cur_val = v1
974+
B[i1] = cur_val
975+
@inbounds for i in inds[2:end]
976+
cur_val = op(cur_val, A[i])
977+
B[i] = cur_val
978+
end
979+
return B
980+
end
981+
906982
### from abstractarray.jl
907983

908984
"""

0 commit comments

Comments
 (0)