Мы переносим классическое приложение ASP с использованием SQL Server на мультиязычность, и для этого в качестве первого шага мы должны перейти с кодировки ansi / windows-1252 на Unicode.
Нам удалось переместитьASP-программы (сохранение в UTF-8 с использованием спецификации), но теперь мы сталкиваемся с проблемами SQL Server.
Мы преобразовали все столбцы из CHAR
в NCHAR
и из VARCHAR
вNVARCHAR
.
Наша проблема появляется в динамических операторах SQL (я знаю, я знаю, мы удалим ее в будущем).
Когда мы используем строковый литерал Unicode, в руководстве к SQL Server необходимо использовать формат N'MyUnicode '(префикс с заглавной буквой N - литералом).Но для нас это кошмар (нам нужно искать код, искать литералы, но только в инструкциях SQL.
Есть ли способ запросить SQL, чтобы рассматривать любой литерал как Unicode?
Наш тестовый код: 1 и 3 работает, 2 не пройден
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Language" content="en" />
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
</head>
<body>
<%
' Respose charset UTF-8 is equivalent to "Content-Type" content="text/html; charset=UTF-8"
Response.Charset="UTF-8"
Response.LCID=1033
Session.LCID=1033
Response.Write "Testing charset UTF-8 with BOM: " & "áéΔ" & "<br/>"
Set Conn = Server.CreateObject("ADODB.Connection")
Conn.Open "Provider=SQLOLEDB; Database=testUnicode;User Id=test;Password=xxxxxxxx"
SQLStmt = "select * from test"
Set rs = Conn.execute(SQLStmt)
If (rs.EOF) then
Response.Write "Select (all records) failed. No record return.<br/>"
Else
Response.Write "Select (all records) succeed. The table contains:<br/>"
Do while not rs.EOF
Response.Write rs.fields("HTML") & " = " & Server.HTMLEncode(rs.fields("UNICODE")) & " Value=" & AscW(rs.fields("UNICODE")) & "<br/>"
valueToSearch=rs.fields("UNICODE")
rs.MoveNext
Loop
rs.Close
Response.Write "<br/><br/>"
Response.Write "Checking for specific record<br/>"
' This works!!
valueToSearch="Δ"
Response.Write "Checking for value = " & AscW(valueToSearch) & "<br/>"
SQLStmt = "select * from test where UNICODE='" & valueToSearch & "'"
Set rs = Conn.execute(SQLStmt)
If (rs.EOF) then
Response.Write "Select (one record " & Server.HTMLEncode(valueToSearch) & ") failed. No record return. Select=" & SQLStmt & "<br/>"
Else
Response.Write "Select (one record " & Server.HTMLEncode(valueToSearch) & ") succeed.<br/>"
rs.MoveFirst
Response.Write rs.fields("HTML") +" = "+Server.HTMLEncode(rs.fields("UNICODE"))+"<br/>"
Response.Write "TESTBIT = " + Server.HTMLEncode(rs.fields("TESTBIT"))+"<br/>"
Response.Write "TEXTBIT TEXTO = " + trim(rs.fields("TESTBIT")) +"<br/>"
End if
Response.Write "<br/>"
' This fail!
valueToSearch="é"
Response.Write "Checking for value = " & AscW(valueToSearch) & "<br/>"
SQLStmt = "select * from test where UNICODE='" & valueToSearch & "'"
Set rs = Conn.execute(SQLStmt)
If (rs.EOF) then
Response.Write "Select (one record " & Server.HTMLEncode(valueToSearch) & ") failed. No record return. Select=" & SQLStmt & "<br/>"
Else
Response.Write "Select (one record " & Server.HTMLEncode(valueToSearch) & ") succeed.<br/>"
rs.MoveFirst
Response.Write rs.fields("HTML") +" = "+Server.HTMLEncode(rs.fields("UNICODE"))+"<br/>"
Response.Write "TESTBIT = " + Server.HTMLEncode(rs.fields("TESTBIT"))+"<br/>"
Response.Write "TEXTBIT TEXTO = " + trim(rs.fields("TESTBIT")) +"<br/>"
End if
Response.Write "<br/>"
' This works!
valueToSearch="é"
Response.Write "Checking for value = " & AscW(valueToSearch) & "<br/>"
SQLStmt = "select * from test where UNICODE=N'" & valueToSearch & "'"
Set rs = Conn.execute(SQLStmt)
If (rs.EOF) then
Response.Write "Select (one record " & Server.HTMLEncode(valueToSearch) & ") failed. No record return. Select=" & SQLStmt & "<br/>"
Else
Response.Write "Select (one record " & Server.HTMLEncode(valueToSearch) & ") succeed.<br/>"
rs.MoveFirst
Response.Write rs.fields("HTML") +" = "+Server.HTMLEncode(rs.fields("UNICODE"))+"<br/>"
Response.Write "TESTBIT = " + Server.HTMLEncode(rs.fields("TESTBIT"))+"<br/>"
Response.Write "TEXTBIT TEXTO = " + trim(rs.fields("TESTBIT")) +"<br/>"
End if
End if
Conn.Close
%>
</body>
Наша тестовая таблица выглядит следующим образом
CREATE TABLE [dbo].[test]
(
[HTML] [NVARCHAR](50) NULL,
[UNICODE] [NVARCHAR](50) NULL,
[TESTBIT] BIT
) ON [PRIMARY]
GO
INSERT [dbo].[test] ([HTML], [UNICODE], TESTBIT)
VALUES (N'á', N'á', 1), (N'é', N'é', 1),
/* Greek Delta Δ */
(N'Δ', N'Δ',0);
GO