Java дженерики и класс <P>выпуск - PullRequest
1 голос
/ 20 апреля 2020

У меня есть следующий код:

public class LazyProductDataModel<P extends Product> extends LazyDataModel<P> { 
    private List<P> data = new ArrayList<P>();
    private SearchProductsCriteria<P> criteria;

    public LazyProductDataModel(SearchProductsCriteria<P> criteria) {
        this.criteria = criteria;
    }

    public SearchProductsCriteria<P> getCriteria() {
        return criteria;
    }

    public void setCriteria(SearchProductsCriteria<P> criteria) {
        this.criteria = criteria;
    }

    @Override
    public List<P> load(int first, int pageSize, String sortField, SortOrder sortOrder, Map<String, String> filters) {
        if (criteria == null) {
            return data;
        }

        int currentPage = first==0 ? 1 : first >= pageSize ? (first / pageSize) + 1 : 0;

        criteria.setPageNumber(currentPage);
        criteria.setPageSize(pageSize);

        try {
            if (criteria.getProductId() != null && criteria.getProductId() != 0) {
                P product = ServiceClientFactory.getInstance().
                        getProductRetrieveClient(criteria.getProductClass()).getProduct(criteria.getProductId());
                if (product != null) {
                    data.add(product);
                    this.setRowCount(1);
                }

            } else {
                LazyDataResponse<P> lazyResponse = ServiceClientFactory.getInstance().
                        getProductSearchClient(criteria.getProductClass()).searchProductsLazyResponse(criteria); 
                data = lazyResponse.getList();
                this.setRowCount(lazyResponse.getTotalRecordsCount());
            }

        } catch (Exception e) {
            LOGGER.error("Exception happened at LazyProductDataModel#load.", e);
            data = new ArrayList<P>();
            this.setRowCount(0);
        }

        return data;
    }
}

3-я строка в блоке try, getProductRetrieveClient (crit.getProductClass ()). GetProduct (crit.getProductId ()); Я звоню crit.getProductClass (), это возвращает Class<P>, и, как вы видите в вышеприведенном классе, P расширяет Product, что означает, что я должен быть в состоянии передать любой класс, который расширяет Product, но в моем классе я пытаюсь использовать его, но он идет не так, как ожидалось:

Вот как я пытаюсь его использовать, пожалуйста, помогите, я новичок в дженериках topi c.

@ManagedBean
@ViewScoped
public class ReportController implements Serializable {

    private ProductReportCriteria<? extends Product> criteria = new ProductReportCriteria<>();

    public ReportController() {

    }

    @PostConstruct
    public void init() {
        criteria.setLocale(userToolKit.getLocale());
        criteria.setProductClass(Movie.class);
    }

    public ProductReportCriteria<?> getCriteria() {
        return criteria;
    }

    public void setCriteria(ProductReportCriteria<? extends Product> criteria) {
        this.criteria = criteria;
    }
}

в последнем методе onFilter внутри оператора if я получаю ошибку компиляции,

The method setProductClass(Class<capture#9-of ? extends Product>) in the type SearchProductsCriteria<capture#9-of ? extends Product> is not applicable for the arguments (Class<Movie>)

public class Movie extends Product implements Serializable {
}
public class Product implements Serializable {
}

Наконец,

public class ProductReportCriteria<P extends Product> extends SearchProductsCriteria<P> implements Serializable {
}

public class SearchProductsCriteria<P> implements Serializable {
    private Class<P> productClass;
    public Class<P> getProductClass() {
        return productClass;
    }
    public void setProductClass(Class<P> productClass) {
        this.productClass = productClass;
    }
}

1 Ответ

3 голосов
/ 20 апреля 2020

ProductReportCriteria<? extends Product> criteria не означает ничего, что расширяет Product, это означает a специфицирует c вещь, которая расширяет Product, компилятор просто не не знаю, что .

Более простой пример - List<? extends Number>: это не означает "список, содержащий какой-либо подкласс Number": вы не можете добавить Integer к такому список (list.add(Integer.valueOf(0))), хотя Integer extends Number. Это потому, что List<Double> является List<? extends Number>, и было бы проблематично c, если бы вы могли добавить Integer к List<Double>:

List<Double> doubles = new ArrayList<>();
List<? extends Number> numbers = doubles;  // Fine.
numbers.add(Integer.valueOf(0));           // Doesn't work; pretend it does.
Double d = doubles.get(0);                 // ClassCastException!

Если вы хотите criteria Потребительские методы для принятия любого экземпляра Product, измените его тип на любой из:

ProductReportCriteria<? super Product>
ProductReportCriteria<Product>
ProductReportCriteria<?>       // Same as Product, because of declared bounds

Подробнее о PECS .

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...