libiconv字符集转换库在C#中的使用

时间:2022-11-05 14:08:25
libiconv字符集转换库使用方法》一文中说到了libiconv可以实现不同字符集的转换。比如GBK转BIG5等。在项目中因为需要,找到这个库。可是这个库在C#中没有很好的支持。不过,想着既然是C++的库,那只要动态加载DLL的接口就好了。可是调用并不顺利,传进去的IntPtr或者byte数组总是拿不到数据。后面回到了C++的方式去调用,几经调试,总算找到了原因。
是iconv接口在转换完成后,指针的位置往后移了。而在C#中调用DLL后回来的指针,已经是移动后的,所以拿不到所要的数据。
经过多种尝试,没有办法将指针移回到原位。

后来,通过C++的二次封装,在C++中将指针的位置移到了原来的位置,再用C#来调用,总算达到了目的。

#include <fstream>
//包函 libiconv库头文件 
#include "iconv.h"

//导入 libiconv库 
#pragma comment(lib,"libiconv.lib")
using namespace std;

#define DLL_EXPORT extern "C" __declspec(dllexport)

DLL_EXPORT int ChangeCode( const char* pFromCode,
						  const char* pToCode,
						  const char* pInBuf,
						  size_t* iInLen,
						  char* pOutBuf,
						  size_t* iOutLen )
{   

	size_t outLenTemp=*iOutLen;

	iconv_t hIconv = iconv_open( pToCode, pFromCode );
	if ( -1 == (int)hIconv )
	{
		return 	-100;//打开失败,可能不支持的字符集 
	}

	//开始转换 
	int iRet = iconv( hIconv, (const char**)(&pInBuf), iInLen, (char**)(&pOutBuf), iOutLen );
	if (iRet>=0)
	{
		pOutBuf=pOutBuf-(outLenTemp-*iOutLen);//转换后pOutBuf的指针被移动,必须移回到起始位置
	}
	else
	{
		iRet=-200;
	}

	//关闭字符集转换 
	iconv_close( hIconv );
	return iRet;
}

C#调用的部分

  /// <summary>
        /// 字符器转换.
        /// 每次转换都需要打开转换器、字符集转换、关闭转换器。
        /// </summary>
        /// <param name="pFromCode">源字符集编码</param>
        /// <param name="pToCode">目标字符集编码</param>
        /// <param name="pInBuf">待转换的内容</param>
        /// <param name="iInLen">待转换的长度。转换成功后,将变成0.</param>
        /// <param name="pOutBuf">转换后的内容</param>
        /// <param name="iOutLen">转换长度。转换成功后,将变成原值减去转换后的内容所占空间的长度</param>
        /// <returns></returns>
        [DllImport("CharsetConvert.dll", CallingConvention = CallingConvention.Cdecl)]
        public static extern int ChangeCode(string pFromCode,
                                            string pToCode,
                                            byte[] pInBuf,
                                            ref int iInLen,
                                            byte[] pOutBuf,
                                            ref int iOutLen);

   private void buttonOneConvert_Click(object sender, EventArgs e)
        {

            string toCode = "BIG5";
            string fromCode = "GBK";

            string inStr = "國k";
            byte[] inBuf = Encoding.Default.GetBytes(inStr);
            byte[] outBuf = new byte[100];

            int inLen = inBuf.Length;
            int outLen = outBuf.Length;

            int result = CharsetConvter.ChangeCode(fromCode, toCode, inBuf, ref inLen, outBuf, ref outLen);
            if (result < 0)
            {
                MessageBox.Show("转换失败");
            }
            else
            {
                String outStr = Encoding.GetEncoding("BIG5").GetString(outBuf);
                MessageBox.Show(outStr);
            }

        }