簡體   English   中英

Neo4j在“ WHERE”中進行類型比較的問題

[英]Neo4j problems with type comparisons in “WHERE”

通過java執行Neo4j查詢時出現以下錯誤:

org.neo4j.graphdb.QueryExecutionException: Don't know how to compare that. Left: "0" (String); Right: 0 (Long)
at org.neo4j.kernel.impl.query.QueryExecutionKernelException.asUserException(QueryExecutionKernelException.java:35)
at org.neo4j.cypher.internal.javacompat.ExecutionResult.converted(ExecutionResult.java:399)
at org.neo4j.cypher.internal.javacompat.ExecutionResult.hasNext(ExecutionResult.java:232)
at main.java.com.bag.server.database.Neo4jDatabaseAccess.readObject(Neo4jDatabaseAccess.java:172)
at main.java.com.bag.server.TestServer.handleNodeRead(TestServer.java:259)
at main.java.com.bag.server.TestServer.appExecuteUnordered(TestServer.java:153)
at bftsmart.tom.server.defaultservices.DefaultRecoverable.executeUnordered(DefaultRecoverable.java:417)
at bftsmart.tom.ServiceReplica.receiveReadonlyMessage(ServiceReplica.java:214)
at bftsmart.tom.core.DeliveryThread.deliverUnordered(DeliveryThread.java:289)
at bftsmart.tom.core.TOMLayer.requestReceived(TOMLayer.java:290)
at bftsmart.communication.client.netty.NettyClientServerCommunicationSystemServerSide.channelRead0(NettyClientServerCommunicationSystemServerSide.java:184)
at bftsmart.communication.client.netty.NettyClientServerCommunicationSystemServerSide.channelRead0(NettyClientServerCommunicationSystemServerSide.java:61)
at io.netty.channel.SimpleChannelInboundHandler.channelRead(SimpleChannelInboundHandler.java:105)
at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:292)
at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:278)
at io.netty.handler.codec.ByteToMessageDecoder.fireChannelRead(ByteToMessageDecoder.java:277)
at io.netty.handler.codec.ByteToMessageDecoder.channelRead(ByteToMessageDecoder.java:264)
at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:292)
at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:278)
at io.netty.channel.DefaultChannelPipeline.fireChannelRead(DefaultChannelPipeline.java:962)
at io.netty.channel.nio.AbstractNioByteChannel$NioByteUnsafe.read(AbstractNioByteChannel.java:131)
at io.netty.channel.nio.NioEventLoop.processSelectedKey(NioEventLoop.java:528)
at io.netty.channel.nio.NioEventLoop.processSelectedKeysOptimized(NioEventLoop.java:485)
at io.netty.channel.nio.NioEventLoop.processSelectedKeys(NioEventLoop.java:399)
at io.netty.channel.nio.NioEventLoop.run(NioEventLoop.java:371)
at io.netty.util.concurrent.SingleThreadEventExecutor$2.run(SingleThreadEventExecutor.java:112)
at io.netty.util.concurrent.DefaultThreadFactory$DefaultRunnableDecorator.run(DefaultThreadFactory.java:137)
at java.lang.Thread.run(Thread.java:745)

但是我可以確定數據庫中的字符串為“ 0”。 我在查詢中將它作為字符串輸入:

String.format(" WHERE r.%s <= %s OR n.%s IS NULL", "snapshotId", Long.toString(0), "snapshotId")

您使用的是字符串修飾而不是參數,因此您丟失了類型信息(因為它只是將所有內容都放在了字符串中)。 如果您在Cypher中鍵入不帶引號的字符'0',它將被解釋為Int 因此,除了您當前的查詢,

WHERE r.snapshotId <= 0 or n.snapshotId IS NULL

你實際上想說

WHERE r.snapshotId <= '0' or n.snapshotId IS NULL

引號括在“ 0”周圍,因此將其視為字符串。

但是,真正的解決方案是代替字符串改寫,而寫一個查詢說:

WHERE r.snapshotId <= {zero_string} or n.snapshotId IS NULL

然后傳遞參數zero_string ,該參數設置為Long.toString(0) 這樣,驅動程序將在打包,解包和解釋數據時為您處理類型。

編輯:或如果您真的需要屬性名稱也是動態的,也將其作為參數傳遞:

WHERE r[{zero_param}] <= {zero_string} or n[{zero_param}] IS NULL

更新:您可以通過傳入Map並進行一些迭代的工作來修改它以適用於多個鍵/值對。 天真的方法是這樣的:

WHERE ALL(k IN KEYS({map_param}) WHERE r[k] <= {map_param}[k] OR n[k] IS NULL)

但這在任何規模上都可能非常慢,因為我認為查詢計划程序無法對其進行優化。 在應用此過濾器之前,請嘗試縮小其他條件下的r匹配范圍。

您的format語句生成的字符串是:

" WHERE r.snapshotId <= 0 OR n.snapshotId IS NULL"

就Cypher而言, Long.toString(0)值仍會導致輸出數字0 ,因為文字值沒有用引號引起來。

要解決此問題,必須確保要比較的值具有兼容的類型。

1.比較字符串

一種方法是確保數字值表示為字符串(用引號引起來):

String.format(" WHERE r.%s <= '%s' OR n.%s IS NULL", "snapshotId", 0, "snapshotId")

2.比較為數值

但是,如果您希望比較的結果在算術上有意義,則通常不使用字符串比較。 例如, (".1" <= "0")true ,而(.1 <= 0)false 因此,對於在算術上有意義的比較,您可能應該將r.snapshotId轉換為浮點數或整數。

例如:

String.format(" WHERE TOFLOAT(r.%s) <= %s OR n.%s IS NULL", "snapshotId", 0, "snapshotId")

使用參數

最后,正如@ToreEschliman所建議的那樣,您應該傳遞參數而不是使用字符串修飾。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM