iconv库的移植

本文介绍如何在ARM平台交叉编译libiconv库并应用于字符编码转换,提供了一个具体的UTF-8转GB2312的例子。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. 需求

    如果你在系统中涉及到字符编码集的时候,比如网页端一般使用字符编码是 utf-8,迪文屏一般支持的字符编码是 GB2312...对于字符编码间的转换在一些时候是一个绕不过的坎。那此篇文章对你可能有所有帮助。

    如果你的程序在ubuntu中,在程序中包括 #include <iconv.h>,使用iconv集就能正常实现字符编码的转换,但如果是在arm板中,简单的包含头文件,iconv函数集是不能运行的,需要我们单独的移植 libiconv 库。


2. 正文

    (1)交叉编译libiconv库

        ①下载:我们先从官网下载libiconv库源码文件, 我此次选择的版本是 libiconv-1.10.tar.gz

        ②解压

        ③编译前配置

> ./configure --prefix="你自己想要存放编译出来的文件夹" --host=arm-linux --CC=“交叉编译工具同交叉编译选项/设置全部变量情况下,此CC省略”

 

        ④编译和整理到指定的文件中

> make

> make install


    (2)使用方法

有两种方法:

        ①配置系统文件,支持libiconv库

在 /etc/profile 文件中添加

export LD_PRELOAD="preloadable_libiconv.so文件存放的位置"

相应的对应上边的操作,我们需要把编译出来的文件中找到此文件,放到相应位置上。

此方法在开机启动的程序中使用,不能保证每次重启后都能加载成功的现象,因此,此方法暂时抛弃。

        ②使用动态库

将生成的库文件中的 libiconv.so、libiconv.so.2、libiconv.so.2.3.0 拷贝到arm板的根文件系统的库文件夹下,如果cp的话要加 -a 选项,保留软连接特性。

在要使用的程序中,我们将 iconv.h 头文件添加到工程中,Makefile中添加 -liconv 链接选项,在使用到iconv函数集的文件中包含进头文件 #include "iconv.h" 。即可。

特殊说明,如果不包含iconv.h文件,是不成功的。不能使用 #include <iconv.h>


3. iconv函数集使用例子

/*
* 将UTF8转到GB2312,遇到不能转的,找到邻近的继续转换
* 转换后,结果放到s_outbuffer中,其中存放的字符的长度是函数的返回值
*/
int ConvertUTF8ToASCII( const char* inbuffer, char *s_outbuffer  )
{
    const char* srcStart = inbuffer;
    char* tempOutBuffer = s_outbuffer;
    size_t srcLen = strlen(inbuffer);
    size_t outLen = 218;

    if (srcLen == 0) return 0;
    iconv_t cd = iconv_open("GB2312//TRANSLIT", "UTF-8"); // UTF8 -> GB2312
    if (cd == (iconv_t)(-1)) {
        perror("iconv_open is error");
    }
    int ret = iconv(cd, (char **)&srcStart, &srcLen, &tempOutBuffer, &outLen);
    if (ret < 0) {
        printf("ret is %d\n", ret);
        perror("iconv");
    }
    iconv_close(cd);
    return 218-outLen;
}

注意函数中 outLen = 218 参数,根据你要转的buf的容限来更改,并相应的更改return中的218的值。

libconv在windows系统的移植,编译,测试工程代码 确实有需要的请下载使用。以下是测试代码 #include <stdlib.h> //include <sys/ipc.h> #include <stdio.h> #include <string.h> //#include <fcntl.h> //include <unistd.h> #include <iconv.h> int code_convert(char *from_charset,char *to_charset,char *inbuf,int inlen,char *outbuf,int outlen) { iconv_t cd; char **pin = &inbuf; char **pout = &outbuf; size_t iconv_ret; cd = iconv_open(to_charset,from_charset); if (cd==(iconv_t)-1) { perror("iconv_open():"); exit(0); } memset(outbuf,0,outlen); iconv_ret = iconv(cd,pin,&inlen,pout,&outlen); if (iconv_ret == (size_t)-1) { perror("iconv():"); exit(0); } iconv_close(cd); return 0; } int u2g(char *inbuf,int inlen,char *outbuf,int outlen) //UNICODE码转为GB2312码 { return code_convert("utf-8","gb2312",inbuf,inlen,outbuf,outlen); } int g2u(char *inbuf,size_t inlen,char *outbuf,size_t outlen) //GB2312码转为UNICODE码 { return code_convert("gb2312","utf-8",inbuf,inlen,outbuf,outlen); } /* 这是一个iconv的测试例程: “纵海杯”东南大学第三届嵌入式系统设计邀请赛 */ int main(void) { unsigned char in_utf8[] = {0x20,0xe8,0xbf,0x99,0xe6,0x98,0xaf,0xe4,0xb8,0x80,0xe4,0xb8,0xaa,0x69,0x63,0x6f, 0x6e,0x76,0xe7,0x9a,0x84,0xe6,0xb5,0x8b,0xe8,0xaf,0x95,0xe4,0xbe,0x8b,0xe7,0xa8, 0x8b,0xef,0xbc,0x9a,0x0a,0xe2,0x80,0x9c,0xe7,0xba,0xb5,0xe6,0xb5,0xb7,0xe6,0x9d, 0xaf,0xe2,0x80,0x9d,0xe4,0xb8,0x9c,0xe5,0x8d,0x97,0xe5,0xa4,0xa7,0xe5,0xad,0xa6, 0xe7,0xac,0xac,0xe4,0xb8,0x89,0xe5,0xb1,0x8a,0xe5,0xb5,0x8c,0xe5,0x85,0xa5,0xe5, 0xbc,0x8f,0xe7,0xb3,0xbb,0xe7,0xbb,0x9f,0xe8,0xae,0xbe,0xe8,0xae,0xa1,0xe9,0x82, 0x80,0xe8,0xaf,0xb7,0xe8,0xb5,0x9b,0x00}; //UTF-8编码 char in_gb2312[] =" 这是一个iconv的测试例程:\n\ “纵海杯”东南大学第三届嵌入式系统设计邀请赛"; //GB2312编码 char out[256]; int rc; printf("\r\n------iconv test------\r\n"); printf("\r\n---Start UTF8 to GB2312 convert...\r\n"); rc = u2g(in_utf8,strlen(in_utf8),out,256); printf("%s\n",out); rc = strcmp(in_gb2312,out); if(rc==0) { printf("---End UTF8 to GB2312 convert OK\r\n"); } else { printf("---End UTF8 to GB2312 convert FAIL\r\n"); } printf("\r\n---Start GB2312 to UTF8 convert...\n"); rc = g2u(in_gb2312,strlen(in_gb2312),out,256); printf("%s\n",out); rc = strcmp(in_utf8,out); if(rc==0) { printf("---End GB2312 to UTF8 convert OK\r\n"); } else { printf("---End GB2312 to UTF8 convert FAIL\r\n"); } printf("\r\n------iconv test over------\r\n"); system("pause");; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值