根据另一个列表

时间:2016-03-27 11:54:49

标签: java-8 java-stream

我在此论坛中搜索后发布了我的查询。谷歌,但无法解决相同的问题。 例如:Link1 Link2 Link3

我正在尝试根据List 1中的值过滤List 2(多列)。

List1:
 - [Datsun]
 - [Volvo]
 - [BMW]
 - [Mercedes]

List2: 
 - [1-Jun-1995, Audi, 25.3, 500.4, 300]
 - [7-Apr-1996, BMW, 35.3, 250.2, 500]
 - [3-May-1996, Porsche, 45.3, 750.8, 200]
 - [2-Nov-1998, Volvo, 75.3, 150.2, 100]
 - [7-Dec-1999, BMW, 95.3, 850.2, 900]

expected o/p:
 - [7-Apr-1996, BMW, 35.3, 250.2, 500]
 - [2-Nov-1998, Volvo, 75.3, 150.2, 100]
 - [7-Dec-1999, BMW, 95.3, 850.2, 900]

代码

// List 1 in above eg
List<dataCarName> listCarName = new ArrayList<>(); 
// List 2 in above eg
List<dataCar> listCar = new ArrayList<>(); 

// Values to the 2 lists are populated from excel

List<dataCar> listOutput = listCar.stream().filter(e -> e.getName().contains("BMW")).collect(Collectors.toList());

在上面的代码中,如果我提供一个特定值,我可以过滤,但不知道如何检查列表2中的车名是否在列表1中退出。

希望我面临的问题很明确,等待指导(对于Java来说,我还是比较新的,因此如果上述查询非常基本,请原谅)。

修改 我相信上面提供的link-3应该解决,但在我的情况下它不起作用。也许是因为list-1中的值被填充为 org.gradle04.Main.Cars.dataCarName@4148db48 ..等 只有当我通过调用getName方法在列表1上执行forEach时,才能以人类可读的格式获取值。

2 个答案:

答案 0 :(得分:45)

不清楚为什么首先有List<DataCarName>而不是List/Set<String>

您必须提供的谓词必须检查相应的数据汽车实例是否在列表中有其名称。

e -> e.getName().contains("BMW")只会检查数据车的名称是否包含不是您想要的宝马。那么你的第一次尝试可能是

e -> listCarName.contains(e.getName())

但由于listCarNameList<DataCarName>e.getName()是一个字符串(我推测),因此您将获得一个空列表。

您拥有的第一个选项是更改谓词,以便从数据汽车名称列表中获取流,将它们映射到其字符串表示形式,并检查这些名称中的任何一个是否与当前数据汽车实例相对应。您当前正在过滤的名称:

List<DataCar> listOutput =
    listCar.stream()
           .filter(e -> listCarName.stream().map(DataCarName::getName).anyMatch(name -> name.equals(e.getName())))
           .collect(Collectors.toList());

现在这非常昂贵,因为您为数据汽车流管道中的每个实例创建了一个流。更好的方法是用汽车制造Set<String>&#39;提前命名,然后只使用contains作为此集合的谓词:

Set<String> carNames = 
    listCarName.stream()
               .map(DataCarName::getName)
               .collect(Collectors.toSet());

List<DataCar> listOutput =
     listCar.stream()
            .filter(e -> carNames.contains(e.getName()))
            .collect(Collectors.toList());

答案 1 :(得分:6)

在您的DataCar类型中,getName()是否返回String或DataCarName枚举类型?如果它是枚举,您可以遵循Alexis C的方法,但不是使用Collectors.toSet()构建HashSet,而是构建一个EnumSet,它可以提供O(1)性能。修改Alexis的建议,结果如下:

Set<DataCarName> carNames = 
    listCarName.stream()
               .collect(Collectors.toCollection(
                   ()-> EnumSet.noneOf(DataCarName.class)));

List<DataCar> listOutput =
    listCar.stream()
               .filter(car -> carNames.contains(car.getName()))  
               .collect(Collectors.toList());