所有的文檔和網上的文章都說N可以解決問題。但如果使用wstring bind后select...,則會發現得到的 wstring 格式的column很大可能結果是亂碼。
其實可以這樣解釋這個問題:鍵盤是沒法輸入uniocde編碼的,除非用微軟拼音的內碼輸入。因此數據庫的客戶端軟件輸入表的內容并非unicode,即使在表設計的時候用了N。因此select出來的結果并不能用wsting來解析。
那么我們怎么在初始化數據庫的時候使用unicode字符串呢?
SQLRETURN SQLExecDirect(
SQLHSTMT StatementHandle,
SQLCHAR * StatementText,
SQLINTEGER TextLength);
sql 在執行的時候,調用SQLExecDirect,其sql語句是SQLCHAR 類型,其實就是unsigned char 的一段空間,并不一定要求這是一個以'/0'結尾的ascii 字符串。如果StatementText是一個ascii 字符串,TextLength可以設置為SQL_NTS。
StatementText是可以嵌入unicode字符的,比如使用INSERT INTO T (c1,c2) VALUES(N'unicode string',data),這時候TextLength要填入整個串的長度而不是SQL_NTS。
拼寫出這樣一個串既不能用窄字符串的函數集合,也不能用寬字符串的函數集合。這里使用memcpy類的函數來構造這樣的串,然后調用SQLExecDirect,取得了成功。比如,
USE [test]
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE TABLE [dbo].[電話薄](
[姓名] [ntext] NULL,
[電話] [nchar](32) NULL
) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]
/*
* insert a unicode string into the table
*
*/
struct buf_s{
void * buf;
int len;
};
/*
*just be same as memcpy ,except for the return.
*/
int memapend(void * dst, void * src ,int len)
{
memcpy(dst,src,len);
return len;
}
void insert()
{
unsigned char sql_buf[256]={0};
int len_sql=0;
char str1[]="insert into 電話薄 (姓名,電話) values(N'";
int len1=strlen(str1);
wchar_t str2[]=L"張三";
int len2=wcslen(str2)*sizeof(wchar_t);
char str3[]="',N'";
int len3=strlen(str3);
wchar_t str4[]=L"010123456";
int len4=wcslen(str4)*sizeof(wchar_t);
char str5[]="')";
int len5=strlen(str5);
buf_s buf_s_a []={str1,len1,
str2,len2,
str3,len3,
str4,len4,
str5,len5};
for (int i=0;i<sizeof(buf_s_a)/sizeof(buf_s);i++)
{
len_sql+=memapend(sql_buf+len_sql, buf_s_a[i].buf,buf_s_a[i].len);
}
SQLExecDirect(h,sql_buf,len_sql);
}
btw:find a good site for ms_sqlserver:
http://www.functionx.com/sqlserver/