Skip to content

Commit db38268

Browse files
christophstroblmp911de
authored andcommitted
Enforce non-null contract on vector elements.
Also add shortcuts for empty sources and simplify copy calls that do not transform source values. See #3193 Original pull request: #3194
1 parent 059d09a commit db38268

File tree

7 files changed

+130
-46
lines changed

7 files changed

+130
-46
lines changed

src/main/java/org/springframework/data/domain/DoubleVector.java

+11-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 the original author or authors.
2+
* Copyright 2025 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -30,7 +30,7 @@ class DoubleVector implements Vector {
3030

3131
private final double[] v;
3232

33-
public DoubleVector(double[] v) {
33+
DoubleVector(double[] v) {
3434
this.v = v;
3535
}
3636

@@ -39,17 +39,22 @@ public DoubleVector(double[] v) {
3939
*/
4040
static Vector copy(double[] v) {
4141

42-
double[] copy = new double[v.length];
43-
System.arraycopy(v, 0, copy, 0, copy.length);
42+
if (v.length == 0) {
43+
return new DoubleVector(new double[0]);
44+
}
4445

45-
return new DoubleVector(copy);
46+
return new DoubleVector(Arrays.copyOf(v, v.length));
4647
}
4748

4849
/**
4950
* Copy the given numeric values and wrap within a Vector.
5051
*/
5152
static Vector copy(Collection<? extends Number> v) {
5253

54+
if (v.isEmpty()) {
55+
return new DoubleVector(new double[0]);
56+
}
57+
5358
double[] copy = new double[v.size()];
5459
int i = 0;
5560
for (Number number : v) {
@@ -87,11 +92,7 @@ public float[] toFloatArray() {
8792

8893
@Override
8994
public double[] toDoubleArray() {
90-
91-
double[] copy = new double[this.v.length];
92-
System.arraycopy(this.v, 0, copy, 0, copy.length);
93-
94-
return copy;
95+
return Arrays.copyOf(this.v, this.v.length);
9596
}
9697

9798
@Override

src/main/java/org/springframework/data/domain/FloatVector.java

+11-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 the original author or authors.
2+
* Copyright 2025 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -30,7 +30,7 @@ class FloatVector implements Vector {
3030

3131
private final float[] v;
3232

33-
public FloatVector(float[] v) {
33+
FloatVector(float[] v) {
3434
this.v = v;
3535
}
3636

@@ -39,17 +39,22 @@ public FloatVector(float[] v) {
3939
*/
4040
static Vector copy(float[] v) {
4141

42-
float[] copy = new float[v.length];
43-
System.arraycopy(v, 0, copy, 0, copy.length);
42+
if (v.length == 0) {
43+
return new FloatVector(new float[0]);
44+
}
4445

45-
return new FloatVector(copy);
46+
return new FloatVector(Arrays.copyOf(v, v.length));
4647
}
4748

4849
/**
4950
* Copy the given numeric values and wrap within a Vector.
5051
*/
5152
static Vector copy(Collection<? extends Number> v) {
5253

54+
if (v.isEmpty()) {
55+
return new FloatVector(new float[0]);
56+
}
57+
5358
float[] copy = new float[v.size()];
5459
int i = 0;
5560
for (Number number : v) {
@@ -76,11 +81,7 @@ public int size() {
7681

7782
@Override
7883
public float[] toFloatArray() {
79-
80-
float[] copy = new float[this.v.length];
81-
System.arraycopy(this.v, 0, copy, 0, copy.length);
82-
83-
return copy;
84+
return Arrays.copyOf(this.v, this.v.length);
8485
}
8586

8687
@Override

src/main/java/org/springframework/data/domain/NumberVector.java

+22-21
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 the original author or authors.
2+
* Copyright 2025 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -18,6 +18,7 @@
1818
import java.util.Arrays;
1919
import java.util.Collection;
2020

21+
import org.springframework.util.Assert;
2122
import org.springframework.util.ObjectUtils;
2223

2324
/**
@@ -30,7 +31,9 @@ class NumberVector implements Vector {
3031

3132
private final Number[] v;
3233

33-
public NumberVector(Number[] v) {
34+
NumberVector(Number[] v) {
35+
36+
Assert.noNullElements(v, "Vector [v] must not contain null elements");
3437
this.v = v;
3538
}
3639

@@ -39,41 +42,39 @@ public NumberVector(Number[] v) {
3942
*/
4043
static Vector copy(Number[] v) {
4144

42-
Number[] copy = new Number[v.length];
43-
System.arraycopy(v, 0, copy, 0, copy.length);
45+
if (v.length == 0) {
46+
return new NumberVector(new Number[0]);
47+
}
4448

45-
return new NumberVector(copy);
49+
return new NumberVector(Arrays.copyOf(v, v.length));
4650
}
4751

4852
/**
4953
* Copy the given {@link Number} and wrap it within a Vector.
5054
*/
51-
static Vector copy(Collection<? extends Number> numbers) {
52-
53-
Number[] copy = new Number[numbers.size()];
55+
static Vector copy(Collection<? extends Number> v) {
5456

55-
int i = 0;
56-
for (Number number : numbers) {
57-
copy[i++] = number;
57+
if (v.isEmpty()) {
58+
return new NumberVector(new Number[0]);
5859
}
5960

60-
return new NumberVector(copy);
61+
return new NumberVector(v.toArray(Number[]::new));
6162
}
6263

6364
@Override
6465
public Class<? extends Number> getType() {
6566

66-
Class<?> candidate = null;
67-
for (Object val : v) {
68-
if (val != null) {
69-
if (candidate == null) {
70-
candidate = val.getClass();
71-
} else if (candidate != val.getClass()) {
72-
return Number.class;
73-
}
67+
if (this.v.length == 0) {
68+
return Number.class;
69+
}
70+
71+
Class<? extends Number> candidate = this.v[0].getClass();
72+
for (int i = 1; i < this.v.length; i++) {
73+
if (candidate != this.v[i].getClass()) {
74+
return Number.class;
7475
}
7576
}
76-
return (Class<? extends Number>) candidate;
77+
return candidate;
7778
}
7879

7980
@Override

src/main/java/org/springframework/data/domain/Vector.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 the original author or authors.
2+
* Copyright 2025 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -75,6 +75,9 @@ static Vector of(double... values) {
7575
static Vector of(Collection<? extends Number> values) {
7676

7777
Assert.notNull(values, "Vector values must not be null");
78+
if(values.isEmpty()) {
79+
return NumberVector.copy(new Number[0]);
80+
}
7881

7982
Class<?> cet = CollectionUtils.findCommonElementType(values);
8083

src/test/java/org/springframework/data/domain/FloatVectorUnitTests.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 the original author or authors.
2+
* Copyright 2025 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.

src/test/java/org/springframework/data/domain/NumberVectorUnitTests.java

+45-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 the original author or authors.
2+
* Copyright 2025 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -15,22 +15,53 @@
1515
*/
1616
package org.springframework.data.domain;
1717

18-
import static org.assertj.core.api.Assertions.*;
18+
import static org.assertj.core.api.Assertions.assertThat;
19+
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
1920

21+
import java.util.ArrayList;
2022
import java.util.Arrays;
23+
import java.util.List;
2124

2225
import org.junit.jupiter.api.Test;
2326

2427
/**
2528
* Unit tests for {@link NumberVector}.
2629
*
2730
* @author Mark Paluch
31+
* @author Christoph Strobl
2832
*/
2933
class NumberVectorUnitTests {
3034

3135
Number[] values = new Number[] { 1.1, 2.2, 3.3, 4.4, 5.5, 6.6f };
3236
Number[] floats = new Number[] { (float) 1.1d, (float) 2.2d, (float) 3.3d, (float) 4.4d, (float) 5.5, 6.6 };
3337

38+
@Test // GH-3193
39+
void shouldErrorOnNullElements() {
40+
41+
List<Long> source = new ArrayList<>(3);
42+
source.add(1L);
43+
source.add(null);
44+
source.add(3L);
45+
46+
assertThatExceptionOfType(IllegalArgumentException.class).isThrownBy(() -> NumberVector.copy(source));
47+
assertThatExceptionOfType(IllegalArgumentException.class)
48+
.isThrownBy(() -> NumberVector.copy(new Number[] { 1L, null, 3L }));
49+
}
50+
51+
@Test // GH-3193
52+
void shouldAcceptEmptySource() {
53+
54+
Vector vector = NumberVector.copy(List.of());
55+
56+
assertThat(vector.size()).isEqualTo(0);
57+
assertThat(vector.getType()).isEqualTo(Number.class);
58+
59+
vector = NumberVector.copy(new Number[] {});
60+
61+
assertThat(vector.size()).isEqualTo(0);
62+
assertThat(vector.getType()).isEqualTo(Number.class);
63+
}
64+
3465
@Test // GH-3193
3566
void shouldCreateVector() {
3667

@@ -48,6 +79,17 @@ void shouldCopyVectorValues() {
4879
assertThat(vector.getSource()).isNotSameAs(vector).isEqualTo(values);
4980
}
5081

82+
@Test // GH-3193
83+
void shouldFigureOutCommonType() {
84+
85+
assertThat(NumberVector.copy(List.of()).getType()).isEqualTo(Number.class);
86+
assertThat(NumberVector.copy(List.of(1)).getType()).isEqualTo(Integer.class);
87+
assertThat(NumberVector.copy(List.of(1L, 2L)).getType()).isEqualTo(Long.class);
88+
assertThat(NumberVector.copy(List.of(1F, 2F)).getType()).isEqualTo(Float.class);
89+
assertThat(NumberVector.copy(List.of(1D, 2D)).getType()).isEqualTo(Double.class);
90+
assertThat(NumberVector.copy(List.of(1D, 2F, 3F)).getType()).isEqualTo(Number.class);
91+
}
92+
5193
@Test // GH-3193
5294
void shouldRenderToString() {
5395

@@ -66,7 +108,7 @@ void shouldCompareVector() {
66108
}
67109

68110
@Test // GH-3193
69-
void sourceShouldReturnSource() {
111+
void sourceShouldReturnSource() { // this one is questionable
70112

71113
Vector vector = new NumberVector(values);
72114

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
/*
2+
* Copyright 2025 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.domain;
17+
18+
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
19+
20+
import java.util.Collection;
21+
22+
import org.junit.jupiter.api.Test;
23+
24+
/**
25+
* @author Christoph Strobl
26+
*/
27+
public class VectorUnitTests {
28+
29+
@Test // GH-3193
30+
void staticInitializersErrorOnNull() {
31+
32+
assertThatExceptionOfType(IllegalArgumentException.class).isThrownBy(() -> Vector.of((double[]) null));
33+
assertThatExceptionOfType(IllegalArgumentException.class).isThrownBy(() -> Vector.of((float[]) null));
34+
assertThatExceptionOfType(IllegalArgumentException.class).isThrownBy(() -> Vector.of((Collection<Number>) null));
35+
}
36+
}

0 commit comments

Comments
 (0)