基于另一个列表的 1 个列表的流筛选器

2022-08-31 16:23:47

我在此论坛和Google中搜索后发布了我的查询,但无法解决相同的问题。例如: 链接1 链接2 链接3

我正在尝试根据列表 1 中的值筛选列表 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时,我才能以人类可读的格式获得值。


答案 1

目前尚不清楚为什么您有一个第一名而不是.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>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());

答案 2

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

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());

推荐