Skip to content

Commit 5dbd45d

Browse files
committed
Fix use of norm and remove vecnorm
The behavior of `norm(x, p)` for matrices changed in Julia PR 27401 to be the same as `norm(vec(x), p)` and `vecnorm` was removed. The previous behavior of `norm` for matrices is now captured by `opnorm`. This updates our extensions to these functions to match, and emits deprecation warnings as appropriate. As a somewhat related change, this also deprecates the positional dimension argument to `sum` in favor of keyword arguments, in keeping with Base.
1 parent 4152844 commit 5dbd45d

File tree

6 files changed

+47
-28
lines changed

6 files changed

+47
-28
lines changed

src/atoms/affine/sum.jl

+8-2
Original file line numberDiff line numberDiff line change
@@ -57,9 +57,13 @@ function conic_form!(x::SumAtom, unique_conic_forms::UniqueConicForms=UniqueConi
5757
return get_conic_form(unique_conic_forms, x)
5858
end
5959

60-
sum(x::AbstractExpr) = SumAtom(x)
60+
# Dispatch to an internal helper function that handles the dimension argument in
61+
# the same manner as Base, with dims=: denoting a regular sum
62+
sum(x::AbstractExpr; dims=:) = _sum(x, dims)
6163

62-
function sum(x::AbstractExpr, dimension::Int)
64+
_sum(x::AbstractExpr, ::Colon) = SumAtom(x)
65+
66+
function _sum(x::AbstractExpr, dimension::Integer)
6367
if dimension == 1
6468
return Constant(ones(1, x.size[1]), Positive()) * x
6569
elseif dimension == 2
@@ -68,3 +72,5 @@ function sum(x::AbstractExpr, dimension::Int)
6872
error("Sum not implemented for dimension $dimension")
6973
end
7074
end
75+
76+
Base.@deprecate sum(x::AbstractExpr, dim::Int) sum(x, dims=dim)

src/atoms/sdp_cone/operatornorm.jl

+16
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
# All expressions and atoms are subtypes of AbstractExpr.
66
# Please read expressions.jl first.
77
#############################################################################
8+
import LinearAlgebra: opnorm
89
export operatornorm, sigmamax
910

1011
### Operator norm
@@ -42,6 +43,21 @@ end
4243
operatornorm(x::AbstractExpr) = OperatorNormAtom(x)
4344
sigmamax(x::AbstractExpr) = OperatorNormAtom(x)
4445

46+
function opnorm(x::AbstractExpr, p::Real=2)
47+
if length(size(x)) <= 1 || minimum(size(x)) == 1
48+
throw(ArgumentError("argument to `opnorm` must be a matrix"))
49+
end
50+
if p == 1
51+
return maximum(sum(abs(x), dims=1))
52+
elseif p == 2
53+
return operatornorm(x)
54+
elseif p == Inf
55+
return maximum(sum(abs(x), dims=2))
56+
else
57+
throw(ArgumentError("matrix p-norms only defined for p = 1, 2, and Inf"))
58+
end
59+
end
60+
4561
# Create the equivalent conic problem:
4662
# minimize t
4763
# subject to

src/atoms/second_order_cone/norm.jl

+10-13
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import LinearAlgebra.norm
2-
export norm_inf, norm, norm_1, vecnorm
2+
export norm_inf, norm, norm_1
33

44
# deprecate these soon
55
norm_inf(x::AbstractExpr) = maximum(abs(x))
@@ -25,19 +25,16 @@ function norm(x::AbstractExpr, p::Real=2)
2525
error("vector p-norms not defined for p < 1")
2626
end
2727
else
28-
# x is a matrix
29-
if p == 1
30-
return maximum(sum(abs(x), 1))
31-
elseif p == 2
32-
return operatornorm(x)
33-
elseif p == Inf
34-
return maximum(sum(abs(x), 2))
35-
else
36-
error("matrix p-norms only defined for p = 1, 2, and Inf")
37-
end
28+
# TODO: After the deprecation period, allow this again but make it consistent with
29+
# LinearAlgebra, i.e. make norm(x, p) for x a matrix the same as norm(vec(x), p).
30+
Base.depwarn("`norm(x, p)` for matrices will in the future be equivalent to " *
31+
"`norm(vec(x), p)`. Use `opnorm(x, p)` for the Julia 0.6 behavior of " *
32+
"computing the operator norm for matrices.", :norm)
33+
return opnorm(x, p)
3834
end
3935
end
4036

41-
function vecnorm(x::AbstractExpr, p::Real=2)
42-
return norm(vec(x), p)
37+
if isdefined(LinearAlgebra, :vecnorm) # deprecated but defined
38+
import LinearAlgebra: vecnorm
4339
end
40+
Base.@deprecate vecnorm(x::AbstractExpr, p::Real=2) norm(vec(x), p)

src/atoms/second_order_cone/norm2.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
# Please read expressions.jl first.
66
#############################################################################
77
import LinearAlgebra.norm2
8-
export EucNormAtom, norm2, vecnorm
8+
export EucNormAtom, norm2
99
export sign, monotonicity, curvature, conic_form!
1010

1111

src/expressions.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
# Each type which subtypes AbstractExpr (Variable and Constant being exceptions)
55
# must have:
66
#
7-
## head::Symbol -- a symbol such as :vecnorm, :+ etc
7+
## head::Symbol -- a symbol such as :norm, :+ etc
88
## children::(AbstractExpr,) -- The expressions on which the current expression
99
## -- is operated
1010
## id_hash::UInt64 -- identifier hash, can be a hash of children

test/test_socp.jl

+11-11
Original file line numberDiff line numberDiff line change
@@ -49,11 +49,11 @@ TOL = 1e-3
4949
@testset "frobenius norm atom" begin
5050
m = Variable(4, 5)
5151
c = [m[3, 3] == 4, m >= 1]
52-
p = minimize(vecnorm(m, 2), c)
52+
p = minimize(norm(vec(m), 2), c)
5353
@test vexity(p) == ConvexVexity()
5454
solve!(p)
5555
@test p.optval sqrt(35) atol=TOL
56-
@test evaluate(vecnorm(m, 2)) sqrt(35) atol=TOL
56+
@test evaluate(norm(vec(m), 2)) sqrt(35) atol=TOL
5757
end
5858

5959
@testset "quad over lin atom" begin
@@ -226,18 +226,18 @@ TOL = 1e-3
226226
@test norm(g, 2) ^ 2 0 atol=TOL
227227
end
228228

229-
@testset "norm consistent with Base" begin
229+
@testset "matrix norm consistent with Base" begin
230230
A = randn(4, 4)
231231
x = Variable(4, 4)
232232
x.value = A
233-
@test evaluate(norm(x)) opnorm(A) atol=TOL
234-
@test evaluate(norm(x, 1)) opnorm(A, 1) atol=TOL
235-
@test evaluate(norm(x, 2)) opnorm(A, 2) atol=TOL
236-
@test evaluate(norm(x, Inf)) opnorm(A, Inf) atol=TOL
237-
@test evaluate(vecnorm(x, 1)) norm(vec(A), 1) atol=TOL
238-
@test evaluate(vecnorm(x, 2)) norm(vec(A), 2) atol=TOL
239-
@test evaluate(vecnorm(x, 7)) norm(vec(A), 7) atol=TOL
240-
@test evaluate(vecnorm(x, Inf)) norm(vec(A), Inf) atol=TOL
233+
@test evaluate(opnorm(x)) opnorm(A) atol=TOL
234+
@test evaluate(opnorm(x, 1)) opnorm(A, 1) atol=TOL
235+
@test evaluate(opnorm(x, 2)) opnorm(A, 2) atol=TOL
236+
@test evaluate(opnorm(x, Inf)) opnorm(A, Inf) atol=TOL
237+
@test evaluate(norm(vec(x), 1)) norm(vec(A), 1) atol=TOL
238+
@test evaluate(norm(vec(x), 2)) norm(vec(A), 2) atol=TOL
239+
@test evaluate(norm(vec(x), 7)) norm(vec(A), 7) atol=TOL
240+
@test evaluate(norm(vec(x), Inf)) norm(vec(A), Inf) atol=TOL
241241
end
242242

243243

0 commit comments

Comments
 (0)