gpt4 book ai didi

c# - 来自 MS-Access 与 .NET 的相同查询的排序顺序不同

转载 作者:太空狗 更新时间:2023-10-30 01:20:29 26 4
gpt4 key购买 nike

我正在将旧版 Access 应用程序移植到 .NET Framework,并且需要保持相同的数据排序顺序。连接到同一个数据库,ACCESS 应用程序使用如下配置的 ODBC 连接:

Microsoft SQL Server Native Client Version 11.00.2100

Data Source Name: Central Host-Set Database
Data Source Description:
Server: localhost\sqlexpress
Use Integrated Security: Yes
Database: CentralSQL15
Language: (Default)
Data Encryption: No
Trust Server Certificate: No
Multiple Active Result Sets(MARS): No
Mirror Server:
Translate Character Data: Yes
Log Long Running Queries: No
Log Driver Statistics: No
Use Regional Settings: No
Use ANSI Quoted Identifiers: Yes
Use ANSI Null, Paddings and Warnings: Yes

在 .NET 应用程序中,我最初使用带有 SQL Server 连接字符串的 SqlClient,但是我也尝试使用 OdbcConnection/OdbcCommand/OdbcDataReader,使用与 Access 完全相同的 ODBC 连接。但是,ACCESS 和 .NET 之间的排序顺序始终不同。

例如查询:

SELECT sessionstn3270.sessionid, 
sessionnames.sessionname,
sessionstn3270.sessiontypeid,
sessionstn3270.usernameid,
sessionnames.host,
sessionnames.port,
lus.lu,
sessionnames.lastmodifiedsn,
templates.template,
templates.templatetype,
sessionnames.sessiontype
FROM (sessionnames
LEFT JOIN templates
ON sessionnames.template = templates.templateid)
INNER JOIN (sessionstn3270
LEFT JOIN lus
ON sessionstn3270.sessionid = lus.sessionid)
ON sessionnames.sessionnameid = sessionstn3270.sessionnameid
WHERE usernameid = 3978
ORDER BY templates.templatetype,
sessionnames.sessiontype;

对于 SQL Management Studio 和 .NET 应用程序,SessionID 按以下顺序返回:

17797
17798
17799
17800
17801
105372

在 MS Access 中,它们按以下顺序返回:

17801
17800
17999
17998
17797
105372

在 95% 的情况下,我可以通过添加 ORDER BY SessionID DESC 来复制相同的搜索结果,但这不是 100% 我尝试过对不同字段使用不同的排序顺序,但没有一个 100% 有效的时间。

有人知道还有什么可能会影响 Access 和 .NET 之间的返回结果顺序吗?

我从 http://expressprofiler.codeplex.com/ 下载了 ExpressProfiler并发现以下内容:

当 Access 向 ACCESS 数据库发送 SQL 查询时,会发生以下事务:

SELECT "dbo"."SessionsTN3270"."SessionID" ,"dbo"."SessionsTN3270"."SessionID","dbo"."SessionNames"."SessionType" ,"dbo"."SessionNames"."SessionNameID","dbo"."Templates"."TemplateType" ,"dbo"."Templates"."TemplateID" FROM "dbo"."SessionsTN3270",\oj "dbo"."SessionNames" LEFT OUTER JOIN "dbo"."Templates" ON ("dbo"."SessionNames"."Template" = "dbo"."Templates"."TemplateID" ) \ WHERE (("dbo"."SessionsTN3270"."UserNameID" = 3978 ) AND ("dbo"."SessionNames"."SessionNameID" = "dbo"."SessionsTN3270"."SessionNameID" ) ) 
go
declare @p1 int
set @p1=1
exec sp_prepexec @p1 output,N'@P1 int',N'SELECT "SessionID" ,"dbo"."LUs"."LUID" FROM "dbo"."LUs" WHERE ("SessionID" = @P1)',17797
select @p1
go
exec sp_execute 1,17798
go
exec sp_execute 1,17799
go
exec sp_execute 1,17800
go
exec sp_execute 1,17801
go
exec sp_execute 1,105372
go
SELECT CASE DATABASEPROPERTYEX( DB_NAME(), 'Updateability') WHEN 'READ_ONLY' THEN 'Y' ELSE 'N' END
go
declare @p1 int
set @p1=1
exec sp_prepexec @p1 output,N'@P1 int,@P2 int,@P3 int,@P4 int,@P5 int,@P6 int,@P7 int,@P8 int,@P9 int,@P10 int',N'SELECT "TemplateID","Template","TemplateType" FROM "dbo"."Templates" WHERE "TemplateID" = @P1 OR "TemplateID" = @P2 OR "TemplateID" = @P3 OR "TemplateID" = @P4 OR "TemplateID" = @P5 OR "TemplateID" = @P6 OR "TemplateID" = @P7 OR "TemplateID" = @P8 OR "TemplateID" = @P9 OR "TemplateID" = @P10',15,15,15,15,15,33,33,33,33,33
select @p1
go
declare @p1 int
set @p1=2
exec sp_prepexec @p1 output,N'@P1 int,@P2 int,@P3 int,@P4 int,@P5 int,@P6 int,@P7 int,@P8 int,@P9 int,@P10 int',N'SELECT "SessionNameID","SessionName","Host","Template","SessionType","Port","LastModifiedSN","SSMA_TimeStamp" FROM "dbo"."SessionNames" WHERE "SessionNameID" = @P1 OR "SessionNameID" = @P2 OR "SessionNameID" = @P3 OR "SessionNameID" = @P4 OR "SessionNameID" = @P5 OR "SessionNameID" = @P6 OR "SessionNameID" = @P7 OR "SessionNameID" = @P8 OR "SessionNameID" = @P9 OR "SessionNameID" = @P10',204,203,202,201,200,272,272,272,272,272
select @p1
go
declare @p1 int
set @p1=3
exec sp_prepexec @p1 output,N'@P1 int,@P2 int,@P3 int,@P4 int,@P5 int,@P6 int,@P7 int,@P8 int,@P9 int,@P10 int',N'SELECT "SessionID","SessionTypeID","UserNameID","SessionNameID" FROM "dbo"."SessionsTN3270" WHERE "SessionID" = @P1 OR "SessionID" = @P2 OR "SessionID" = @P3 OR "SessionID" = @P4 OR "SessionID" = @P5 OR "SessionID" = @P6 OR "SessionID" = @P7 OR "SessionID" = @P8 OR "SessionID" = @P9 OR "SessionID" = @P10',17801,17800,17799,17798,17797,105372,105372,105372,105372,105372
select @p1
go
declare @p1 int
set @p1=4
exec sp_prepexec @p1 output,N'@P1 int,@P2 int,@P3 int,@P4 int,@P5 int,@P6 int,@P7 int,@P8 int,@P9 int,@P10 int',N'SELECT "LUID","SessionID","LU" FROM "dbo"."LUs" WHERE "LUID" = @P1 OR "LUID" = @P2 OR "LUID" = @P3 OR "LUID" = @P4 OR "LUID" = @P5 OR "LUID" = @P6 OR "LUID" = @P7 OR "LUID" = @P8 OR "LUID" = @P9 OR "LUID" = @P10',18539,18539,18539,18539,18539,18539,18539,18539,18539,18539
select @p1
go
SELECT "SessionTypeID" ,"SessionType" FROM "dbo"."SessionTypes" WHERE ("HostTypeID" = 200 )
go
SELECT "UserNameID" ,"WSUserName" ,"RecordType" ,"Role" ,"SecurityLevel" ,"BuildingCode" ,"FloorLevel" ,"ServerID" ,"Airline" ,"Description" ,"LastModified" FROM "dbo"."WSUsers"
go

但是当 SQL Server 运行查询时,就简单多了:

SELECT SessionsTN3270.SessionID, SessionNames.SessionName, SessionsTN3270.SessionTypeID, SessionsTN3270.UserNameID, SessionNames.Host, SessionNames.Port, LUs.LU, SessionNames.LastModifiedSN, Templates.Template, Templates.TemplateType, SessionNames.SessionType

FROM (SessionNames LEFT JOIN Templates ON SessionNames.Template = Templates.TemplateID) INNER JOIN (SessionsTN3270 LEFT JOIN LUs ON SessionsTN3270.SessionID = LUs.SessionID) ON SessionNames.SessionNameID = SessionsTN3270.SessionNameID WHERE UserNameID = 3978 ORDER BY Templates.TemplateType, SessionNames.SessionType;

go

从这个 SQL 跟踪中,我发现我可以使用查询从 SQL 服务器复制相同的订单,像这样开始:

SELECT "dbo"."SessionsTN3270"."SessionID" ,"dbo"."SessionsTN3270"."SessionID","dbo"."SessionNames"."SessionType" ,"dbo"."SessionNames"."SessionNameID","dbo"."Templates"."TemplateType" ,"dbo"."Templates"."TemplateID" FROM "dbo"."SessionsTN3270", "dbo"."SessionNames" LEFT OUTER JOIN "dbo"."Templates" ON ("dbo"."SessionNames"."Template" = "dbo"."Templates"."TemplateID" ) 
WHERE (("dbo"."SessionsTN3270"."UserNameID" = 3978 ) AND ("dbo"."SessionNames"."SessionNameID" = "dbo"."SessionsTN3270"."SessionNameID" ) )

最佳答案

解决方案是使用 ExpressProfiler ( http://expressprofiler.codeplex.com/ ) 如果您有完整的 SQL,请使用 SQL 中包含的 SQL Profile。通过使用 ExpressProfiler 跟踪查询,我能够匹配 ACCESS 使用的查询,因为它 ACCESS 在发送到 SQL 服务器之前修改了 SQL 查询,然后我可以获得相同顺序的结果。我在这篇博文中提供了完整的详细信息:http://chentiangemalc.wordpress.com/2013/09/23/case-of-the-sql-vs-access-sort-order/

关于c# - 来自 MS-Access 与 .NET 的相同查询的排序顺序不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18949232/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com