草庐IT

关于 java:JPA 条件谓词查询结合 Integer 和 String 参数

codeneng 2023-03-28 原文

JPA criteria predicate query combining Integer and String arguments

我正在使用 JPA(EclipseLink 风格),框架 JSF 2.0,查询包含 4 个字段的 MySQL 联系人表:

1
contactId(int Pk), firstName (String), surname(String), countryId(int Foreign key)

如果仅使用字符串参数,则以下代码可以正常工作,以便用户可以根据需要输入尽可能多或尽可能少的搜索字符串(即:如果搜索框留空,则相当于选择 * 并且整个数据集是回)。

问题在于组合一个整数参数 (countryId),它是将国家添加到搜索条件的外键。经过一些研究,我仍然无法理解正确的方法来做到这一点。

是否需要在 searchContacts 方法中将整数转换为字符串表示形式?我收集到 JPA Criteria API 提供类型转换但不提供类型转换方法?如果是这样,在下面的方法中包含一个整数并将这个整数传递给 predicateArray 的最佳方法是什么?

提前致谢!

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
public ListDataModel<Contacts> searchContacts(String firstname, String surName, int countryId) {

    CriteriaBuilder cb = em.getCriteriaBuilder();
    CriteriaQuery<Contacts> query = cb.createQuery(Contacts.class);
    Root<Contacts> cont = query.from(Contacts.class);
    query.select(cont);

    List<Predicate> predicateList = new ArrayList<Predicate>();
    Predicate firstnamePredicate, surnamePredicate, countryPredicate;


    if ((firstname != null) && (!(firstname.isEmpty()))) {
        firstnamePredicate = cb.like(cb.upper(cont.<String>get("firstname")),"%" + firstname.toUpperCase() +"%");
        predicateList.add(firstnamePredicate);
    }

    if ((surName != null) && (!(surName.isEmpty()))) {
        surnamePredicate = cb.like(cb.upper(cont.<String>get("surname")),"%" + surName.toUpperCase() +"%");
        predicateList.add(surnamePredicate);
    }
// here is where I am stuck and trying the solution suggested     by     meskobalazs, except I changed null to 0 since countryId is an integer
    if (countryId != 0) {
    countryPredicate = cb.equal(cont.<Integer>get("countryid"), countryId);
    predicateList.add(countryPredicate);
   }
    Predicate[] predicateArray = new Predicate[predicateList.size()];
    predicateList.toArray(predicateArray);
    query.where(predicateArray);
    ListDataModel<Contacts> contactList = new ListDataModel<Contacts>(em.createQuery(query).getResultList());

    return contactList;
}

}

上述导致以下 EJB 异常:

1
2
3
4
5
6
7
8
9
10
Caused by: Exception [EclipseLink-6078] (Eclipse Persistence Services -     2.5.0.v20130507-3faac2b): org.eclipse.persistence.exceptions.QueryException
Exception Description: The class of the argument for the object comparison is incorrect.
Expression: [
Base com.manaar.domains.Contacts]
Mapping: [org.eclipse.persistence.mappings.ManyToOneMapping[countryid]]
Argument: [1]
Query: ReadAllQuery(referenceClass=Contacts )
at    org.eclipse.persistence.exceptions.QueryException.incorrectClassForObjectComparison(QueryException.java:595)
at org.eclipse.persistence.mappings.OneToOneMapping.buildObjectJoinExpression(OneToOneMapping.java:287)
at org.eclipse.persistence.internal.expressions.RelationExpression.normalize(RelationExpression.java:803)

countryid 是一个整数,在联系人实体到国家实体中具有 manyToOne 映射。联系人实体是:

1
2
3
4
5
6
7
8
9
10
11
12
13
@Entity
@Table(name ="contacts")
@XmlRootElement
public class Contacts implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@Basic(optional = false)
@NotNull
@Column(name ="CONTACTSID")
private Integer contactsid;
@JoinColumn(name ="countryid", referencedColumnName ="COUNTRYID")
@ManyToOne
private Country countryid;

国家实体是:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@Entity
@Table(name ="country")
@XmlRootElement

public class Country implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@Basic(optional = false)
@NotNull
@Column(name ="COUNTRYID")
private Integer countryid;
@Size(max = 255)
@Column(name ="COUNTRYNAME")
private String countryname;
.....

  • 嗨,Jay,你能分享你的联系人实体吗?它来查找您的 CountryId 是整数还是 Country 表的实体。另外也共享异常。
  • 嗨,阿杰。感谢您的回复。我在正文中包括了联系人和国家实体,并解释了它们之间的关系。我没有包括任何例外,因为我什至没有尝试任何正确的方法,所以我需要首先了解正确的工作概念。如果您需要任何进一步的信息,请告诉我。


谓词的解决方案

Predicate 类没有类型(通用),因此您不需要以任何不同的方式收集不同的谓词。这应该可以正常工作:

1
2
3
4
if (countryId != null) {
    countryPredicate = cb.equal(cont.<Integer>get("countryid"), countryId);
    predicateList.add(countryPredicate);
}

JPA 规范元模型

这只是一个建议,但您也可以考虑为您的实体构建规范元模型(例如使用 Apache Dali),这样您就可以以类型安全的方式 get 您的字段。

代替

1
cont.<MyType>get("typeName")

你可以写:

1
cont.get(_MyType.typeName)

如果您正在重构实体,这尤其有用,因为您可能会忘记更新字符串,但您不能忘记更新元模型,因为代码甚至无法编译。

更新

我遗漏了一些与您的实体有关的设计问题。有两种可能的解决方案。

创建一个新的实体字段

第一个是在 Contacts 中创建一个实际的 Integer 字段并将其映射到与 countryId 相同的字段。

所以你应该将 Country 字段重命名为 Country,并创建一个新字段。

1
2
3
4
5
6
@Column(name ="countryid", insertable = false, updatable = false)
private Integer countryId;

@JoinColumn(name ="countryid", referencedColumnName ="countryid")
@ManyToOne
private Country country;

完成此操作后,您就可以使用我原来的解决方案了。

使用连接

第二种解决方案是在查询中使用 Join

1
2
3
4
5
if (countryId != null) {
    Join<Contacts, Country> country = cont.join("countryid");
    countryPredicate = cb.equal(country.<Integer>get("countryId"), countryId);
    predicateList.add(countryPredicate);
}

  • 感谢您提供有用的建议mesko。根据您的第一个建议,我尝试了 if(countryId !=0),因为 null 不适用于整数。不幸的是,它产生了一个 EJB 异常,我将在问题的主体中发布。至于规范元模型,我首先想根据您的第一个建议掌握一种添加整数参数的基本方法。
  • 当然,它适用于 Integer。那是一个对象,不要将它与 int 原语混淆。
  • 我也很惊讶如果我尝试添加 if(countryId !=null) 会出现编译时错误,但如果我声明 (countryId != 0) 则没有错误。我之前使用过整数和 null 并且它可以工作,但由于某种原因在这种情况下不是
  • 我现在看到了,方法不对,第三个参数应该是Integer
  • 好的,对不起我的疏忽。我修复了第三个参数并且编译错误消失了,但我仍然有相同的 EjbException
  • 哦,我明白了原因,不幸的是,您的变量名非常糟糕。 countryid 实际上是一个 Country 实体,而不是 ID。
  • 是的。不幸的是,它是 Netbeans IDE 自动生成的代码(应该可以节省我的时间)。因此,国家/地区 ID 是联系人数据库中的外键。所以诀窍是如何在 CountryPredicate 的 get 方法中引用这个来获取 countryid.countryid,其中第二个 countryid 是整数。这可能是使用正确 SQL 的问题。或者我应该把 countryid 改成 country
  • 好的,我不能感谢你!第二种方法有效,Join 成功。我也将尝试第一种方法,但我知道这将涉及更多配置来更改映射。再次感谢!
  • 是的,第一个在编码方面更繁琐,但它节省了连接,这是一项昂贵的操作。
  • 那么它可能是一个更好的选择,因为我有大约 2 个外键,用于公司和部门。所以我对第一种方法的问题是,如果映射被 countryId 整数更改,原始 Country countryid 实体会发生什么情况?仍然需要在 ORM 映射中保留 Country 实体吗?
  • 让我们在聊天中继续这个讨论。

有关关于 java:JPA 条件谓词查询结合 Integer 和 String 参数的更多相关文章

  1. ruby-on-rails - 结合 meta_search 与 acts_as_taggable_on - 2

    我在开发的Rails3网站的一些搜索功能上遇到了一个小问题。我有一个简单的Post模型,如下所示:classPost我正在使用acts_as_taggable_on来更轻松地向我的帖子添加标签。当我有一个标记为“rails”的帖子并执行以下操作时,一切正常:@posts=Post.tagged_with("rails")问题是,我还想搜索帖子的标题。当我有一篇标题为“Helloworld”并标记为“rails”的帖子时,我希望能够通过搜索“hello”或“rails”来找到这篇帖子。因此,我希望标题列的LIKE语句与acts_as_taggable_on提供的tagged_with方法

  2. ruby - ECONNRESET (Whois::ConnectionError) - 尝试在 Ruby 中查询 Whois 时出错 - 2

    我正在用Ruby编写一个简单的程序来检查域列表是否被占用。基本上它循环遍历列表,并使用以下函数进行检查。require'rubygems'require'whois'defcheck_domain(domain)c=Whois::Client.newc.query("google.com").available?end程序不断出错(即使我在google.com中进行硬编码),并打印以下消息。鉴于该程序非常简单,我已经没有什么想法了-有什么建议吗?/Library/Ruby/Gems/1.8/gems/whois-2.0.2/lib/whois/server/adapters/base.

  3. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  4. ruby - RSpec - 使用测试替身作为 block 参数 - 2

    我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

  5. ruby - 如何在 Ruby 中拆分参数字符串 Bash 样式? - 2

    我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"

  6. ruby - 检查方法参数的类型 - 2

    我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)

  7. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  8. ruby-on-rails - 在默认方法参数中使用 .reverse_merge 或 .merge - 2

    两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option

  9. ruby - 如何根据特征实现 FactoryGirl 的条件行为 - 2

    我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden

  10. ruby-on-rails - 在 Rails 和 ActiveRecord 中查询时忽略某些字段 - 2

    我知道我可以指定某些字段来使用pluck查询数据库。ids=Item.where('due_at但是我想知道,是否有一种方法可以指定我想避免从数据库查询的某些字段。某种反拔?posts=Post.where(published:true).do_not_lookup(:enormous_field) 最佳答案 Model#attribute_names应该返回列/属性数组。您可以排除其中一些并传递给pluck或select方法。像这样:posts=Post.where(published:true).select(Post.attr

随机推荐