数据乱码(之.net加密)
环境:.net开发程序用 base64对某些数据进行加密。然后oracle进行解密,并明文存入数据库。
数据库:oracle 10.2.0.1
系统:centos 5.6
1)
select * from resultedit t where answer is not null and aclass=101
and serialno=2310
1 101 2310 1 11100
2 101 2310 20 050201
3 101 2310 30 专升本
2)
select d.answer||t.answer from resultedit t ,resultedit d
where t.answer is not null and t.aclass=101
and t.serialno=2310 and t.aclass=d.aclass
截取一部分查询结果
1 专?
2 050201
3 11100
4 专?
5 050201
6 11100
7 11100专?
8 11100050201
9 1110011100
10 专?
11 050201
12 11100
13 专?
14 050201
解决方法:
1、利用convert 转变answer 字符串从ZHS16GBK到UTF8字符集(UTF8为服务器数据库字符集)
update resultedit r set r.answer=convert(r.answer, 'UTF8', 'ZHS16GBK')
where r.aclass='101' and r.answer is not null
2、执行查询:
select d.answer||t.answer from resultedit t ,resultedit d
where t.answer is not null and t.aclass=101
and t.serialno=2310 and t.aclass=d.aclass
1 专升本
2 050201
3 11100
4 专升本
5 050201
6 11100
7 11100专升本
8 11100050201
9 1110011100
3、分析根本原因:
开发人员在密码的过程中没有指定数据库的字符集。正确的配置是把字符集修改成服务器数据库的字符集(之前有乱码是因为加密指定的字符集为default)。红色字体为服务器数据库正确字符集。
#region 加密
public string[] AddPass(string[] args)
{
string[] myanswer = new string[args.Length];
for (int i = 0; i < args.Length; i++)
{
myanswer[i] = Convert.ToBase64String(Encoding.UTF8.GetBytes(args[i].Trim()));
}
return myanswer;
}
#endregion
#region 解析字符串
public string[,] EnCode(string str, int ccount)
{
string[] strs = str.Substring(0, str.Length).Split('#');
string[,] answer = new string[strs.Length, ccount];
char[] spl = { '-', '>' };
for (int i = 0; i < strs.Length; i++)
{
string[] str2 = strs[i].Split('.');
if (str2[2].Contains("->"))
{
string[] str3 = str2[2].Split('@');
for (int j = 0; j < str3.Length; j++)
{
string[] gst = str3[j].Split(spl);
byte[] mk = Convert.FromBase64String(gst[2].Trim());
string strDecryptParameter = Encoding.UTF8.GetString(mk);
answer[i, j] = strDecryptParameter;
}
}
else
{
byte[] chd = Convert.FromBase64String(str2[2].Trim());
string mystr = Encoding.UTF8.GetString(chd);
answer[i, 0] = mystr;
}
}
return answer;
}
#endregion
总结:.net程序加密,输入数据库的时候一定要指定服务器数据库的字符集相同,不让转码会出现问题。
本人仅仅记录了解决问题的方法与思路。如有错误请您指正,因为我不是开发所以研究的不够深入见谅!
附表:下文介绍来源于(http://www.supesoft.com/ArticleDisp.asp?ID=4256&Cmd=Print)
/**//// <summary>
/// 将字符串使用base64算法加密
/// </summary>
/// <param name="sourceString">待加密的字符串</param>
/// <param name="ens">System.Text.Encoding 对象,如创建中文编码集对象:System.Text.Encoding.GetEncoding(54936)</param>
/// <returns>加码后的文本字符串</returns>
public static string EncodingForString(string sourceString, System.Text.Encoding ens)
{
return Convert.ToBase64String(ens.GetBytes(sourceString));
}
/**//// <summary>
/// 将字符串使用base64算法加密
/// </summary>
/// <param name="sourceString">待加密的字符串</param>
/// <returns>加码后的文本字符串</returns>
public static string EncodingForString(string sourceString)
{
return EncodingForString(sourceString, System.Text.Encoding.GetEncoding(65001));
}
/**//// <summary>
/// 从base64编码的字符串中还原字符串,支持中文
/// </summary>
/// <param name="base64String">base64加密后的字符串</param>
/// <param name="ens">System.Text.Encoding 对象,如创建中文编码集对象:System.Text.Encoding.GetEncoding(54936)</param>
/// <returns>还原后的文本字符串</returns>
public static string DecodingForString(string base64String, System.Text.Encoding ens)
{
/**//**
* ***********************************************************
*
* 从base64String中取得的字节值为字符的机内码(ansi字符编码)
* 一般的,用机内码转换为汉字是公式:
* (char)(第一字节的二进制值*256+第二字节值)
* 而在c#中的char或string由于采用了unicode编码,就不能按照上面的公式计算了
* ansi的字节编和unicode编码不兼容
* 故利用.net类库提供的编码类实现从ansi编码到unicode代码的转换
*
* GetEncoding 方法依赖于基础平台支持大部分代码页。但是,对于下列情况提供系统支持:默认编码,即在执行此方法的计算机的区域设置中指定的编码;Little-Endian Unicode (UTF-16LE);Big-Endian Unicode (UTF-16BE);Windows 操作系统 (windows-1252);UTF-7;UTF-8;ASCII 以及 GB18030(简体中文)。
*
*指定下表中列出的其中一个名称以获取具有对应代码页的系统支持的编码。
*
* 代码页 名称
* 1200 “UTF-16LE”、“utf-16”、“ucs-2”、“unicode”或“ISO-10646-UCS-2”
* 1201 “UTF-16BE”或“unicodeFFFE”
* 1252 “windows-1252”
* 65000 “utf-7”、“csUnicode11UTF7”、“unicode-1-1-utf-7”、“unicode-2-0-utf-7”、“x-unicode-1-1-utf-7”或“x-unicode-2-0-utf-7”
* 65001 “utf-8”、“unicode-1-1-utf-8”、“unicode-2-0-utf-8”、“x-unicode-1-1-utf-8”或“x-unicode-2-0-utf-8”
* 20127 “us-ascii”、“us”、“ascii”、“ANSI_X3.4-1968”、“ANSI_X3.4-1986”、“cp367”、“csASCII”、“IBM367”、“iso-ir-6”、“ISO646-US”或“ISO_646.irv:1991”
* 54936 “GB18030”
*
* 某些平台可能不支持特定的代码页。例如,Windows 98 的美国版本可能不支持日语 Shift-jis 代码页(代码页 932)。这种情况下,GetEncoding 方法将在执行下面的 C# 代码时引发 NotSupportedException:
*
* Encoding enc = Encoding.GetEncoding("shift-jis");
*
* **************************************************************/
//从base64String中得到原始字符
return ens.GetString(Convert.FromBase64String(base64String));
}
/**//// <summary>
/// 从base64编码的字符串中还原字符串,支持中文
/// </summary>
/// <param name="base64String">base64加密后的字符串</param>
/// <returns>还原后的文本字符串</returns>
public static string DecodingForString(string base64String)
{
return DecodingForString(base64String, System.Text.Encoding.GetEncoding(65001));
}
//--------------------------------------------------------------------------------------
/**//// <summary>
/// 对任意类型的文件进行base64加码
/// </summary>
/// <param name="fileName">文件的路径和文件名</param>
/// <returns>对文件进行base64编码后的字符串</returns>
public static string EncodingForFile(string fileName)
{
System.IO.FileStream fs = System.IO.File.OpenRead(fileName);
System.IO.BinaryReader br = new System.IO.BinaryReader(fs);
/**//*System.Byte[] b=new System.Byte[fs.Length];
fs.Read(b,0,Convert.ToInt32(fs.Length));*/
string base64String = Convert.ToBase64String(br.ReadBytes((int)fs.Length));
br.Close();
fs.Close();
return base64String;
}
/**//// <summary>
/// 把经过base64编码的字符串保存为文件
/// </summary>
/// <param name="base64String">经base64加码后的字符串</param>
/// <param name="fileName">保存文件的路径和文件名</param>
/// <returns>保存文件是否成功</returns>
public static bool SaveDecodingToFile(string base64String, string fileName)
{
System.IO.FileStream fs = new System.IO.FileStream(fileName, System.IO.FileMode.Create);
System.IO.BinaryWriter bw = new System.IO.BinaryWriter(fs);
bw.Write(Convert.FromBase64String(base64String));
bw.Close();
fs.Close();
return true;
}