Almost verbatim from Stuart Marks' answer:
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.Collectors;
class Class {
public static <T> Predicate<T> distinctByKey(Function<? super T, Object> keyExtractor) {
Map<Object, Boolean> seen = new ConcurrentHashMap<>();
return t -> seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE) == null;
}
private static List<Pojo> getList() {
return Arrays.asList(
new Pojo("123", 100),
new Pojo("123", 100),
new Pojo("123", 100),
new Pojo("456", 200)
);
}
public static void main(String[] args) {
System.out.println(getList().stream()
// extract a key for each Pojo in here.
// concatenating name and price together works as an example
.filter(distinctByKey(p -> p.getName() + p.getPrice()))
.collect(Collectors.toList()));
}
}
class Pojo {
private final String name;
private final Integer price;
public Pojo(final String name, final Integer price) {
this.name = name;
this.price = price;
}
public String getName() {
return name;
}
public Integer getPrice() {
return price;
}
@Override
public String toString() {
final StringBuilder sb = new StringBuilder("Pojo{");
sb.append("name='").append(name).append('\'');
sb.append(", price=").append(price);
sb.append('}');
return sb.toString();
}
}
This main method yields:
[Pojo{name='123', price=100}, Pojo{name='456', price=200}]
Edit
Made price an int
per Eugene's prompting.
Note: that you could use something more interesting as a key if you wanted to flesh it out:
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.Collectors;
class Class {
public static <T> Predicate<T> distinctByKey(Function<? super T, Object> keyExtractor) {
Map<Object, Boolean> seen = new ConcurrentHashMap<>();
return t -> seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE) == null;
}
private static List<Pojo> getList() {
return Arrays.asList(
new Pojo("123", 100),
new Pojo("123", 100),
new Pojo("123", 100),
new Pojo("456", 200)
);
}
private static class NameAndPricePojoKey {
final String name;
final int price;
public NameAndPricePojoKey(final Pojo pojo) {
this.name = pojo.getName();
this.price = pojo.getPrice();
}
@Override
public boolean equals(final Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
final NameAndPricePojoKey that = (NameAndPricePojoKey) o;
if (price != that.price) return false;
return name != null ? name.equals(that.name) : that.name == null;
}
@Override
public int hashCode() {
int result = name != null ? name.hashCode() : 0;
result = 31 * result + price;
return result;
}
}
public static void main(String[] args) {
System.out.println(getList().stream()
// extract a key for each Pojo in here.
.filter(distinctByKey(NameAndPricePojoKey::new))
.collect(Collectors.toList()));
}
}
class Pojo {
private String name;
private Integer price;
private Object otherField1;
private Object otherField2;
public Pojo(final String name, final Integer price) {
this.name = name;
this.price = price;
}
public String getName() {
return name;
}
public void setName(final String name) {
this.name = name;
}
public Integer getPrice() {
return price;
}
public void setPrice(final Integer price) {
this.price = price;
}
public Object getOtherField1() {
return otherField1;
}
public void setOtherField1(final Object otherField1) {
this.otherField1 = otherField1;
}
public Object getOtherField2() {
return otherField2;
}
public void setOtherField2(final Object otherField2) {
this.otherField2 = otherField2;
}
@Override
public boolean equals(final Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
final Pojo pojo = (Pojo) o;
if (name != null ? !name.equals(pojo.name) : pojo.name != null) return false;
if (price != null ? !price.equals(pojo.price) : pojo.price != null) return false;
if (otherField1 != null ? !otherField1.equals(pojo.otherField1) : pojo.otherField1 != null) return false;
return otherField2 != null ? otherField2.equals(pojo.otherField2) : pojo.otherField2 == null;
}
@Override
public int hashCode() {
int result = name != null ? name.hashCode() : 0;
result = 31 * result + (price != null ? price.hashCode() : 0);
result = 31 * result + (otherField1 != null ? otherField1.hashCode() : 0);
result = 31 * result + (otherField2 != null ? otherField2.hashCode() : 0);
return result;
}
@Override
public String toString() {
final StringBuilder sb = new StringBuilder("Pojo{");
sb.append("name='").append(name).append('\'');
sb.append(", price=").append(price);
sb.append(", otherField1=").append(otherField1);
sb.append(", otherField2=").append(otherField2);
sb.append('}');
return sb.toString();
}
}
get the distinct by key
, but instead is how to get first the first mapping:n1, p1
and then the secondn2, p3
. at least that is not clear for me – Strumpetby price or name, but not already seen
... I don't even know how to phrase that – Strumpetby price and name, both not already seen
is what they're going for. And in that case, you have to look at both fields simultaneously, not one then the other, like in the OP. – Slapstick