使用gSOAP开发实例(3) iconv解决中文乱码问题

电信provisioning系统中,常常需要与远程服务器实时交换一些数据,以完成用户的请求。由于简单对象访问协议(Simple Object Access Protocol, SOAP)的流行,许多涉及到第三方的应用,我们一般都比较乐意使用SOAP来开发。不过,由于可能涉及到公司的机密,本系列教程的开发实例尽量采用在网上已经公开的Web Service资源。

 

我开发SOAP应用程序已经有一定的经验,在C/C++环境下一般使用gSOAP,而在Java环境下一般采用axis2。比较两者的话,除了开发语言之外,还是有不少差别,处理中文字符就是其中之一。网上分别搜索一下“axis2 乱码”和“gSOAP乱码”,匹配的结果是相差很远的。Axis2好像比较智能,能够识别服务端的字符编码,这方面的问题也少,而最新版本的gSOAP,很可能还是需要程序员做多很多功夫。

 

在第一节客户端的教程中,输出的中文股票名称,其实就是乱码,不过为了主次之分,当时做了特别处理,忽略过去。

 

网上解决gSOAP乱码的主流方案是,初始化soap对象之后对其设置SOAP_C_UTFSTRING参数,例如:

        struct soap soap;

        soap_init(&soap);

        soap_set_mode(&soap, SOAP_C_UTFSTRING);

 

但是,单纯这样修改,在某些特定设置的机器上可能有效,反正我试过,仍然是乱码,如下图。怎么办呢?

 

 

Linux下有一个字符编码转换的工具iconv,同时也提供了一套可编程的接口。利用它,就可以测试出来自于服务端中文字符编码的类型,从而进一步实现在程序中自动转换编码。

 

Iconv常用用法是:iconv -t=to_charset -f=from_charset filename

因此,把需要转换编码的内容保存为一个文件,然后执行iconv试出需要转换的编码类型。from­_charset几乎百分百肯定就是utf8,那么to_charset来来去去就那么几个,一个个试也很快试出来了。最终得出的结果是gbk编码,从而修改客户端程序以解决乱码问题。

 

[cpp]  view plain copy
  1. #include <iconv.h>  
  2.   
  3. #include "soapH.h"  
  4. #include "ChinaStockWebServiceSoap12.nsmap"  
  5.   
  6. #define OUTPUT_LEN 32  
  7.   
  8. int conv_charset(const char *dest, const char *src, char *input, size_t ilen, char *output, size_t olen) {  
  9.     iconv_t conv = iconv_open(dest, src);  
  10.     if ( conv == (iconv_t) -1 )  
  11.         return -1;  
  12.     memset(output, 0, olen);  
  13.     if ( iconv(conv, &input, &ilen, &output, &olen) )  
  14.         return -1;  
  15.     iconv_close(conv);  
  16.     return 0;  
  17. }  
  18.   
  19. int main(int argc, char **argv) {  
  20.     if ( argc != 2 && argc != 3 ) {  
  21.         printf("Usage: %s stock_code [end_point]/n", argv[0]);  
  22.         exit(-1);  
  23.     }  
  24.   
  25.     struct soap soap;  
  26.     soap_init(&soap);  
  27.     soap_set_mode(&soap, SOAP_C_UTFSTRING);  
  28.   
  29.     struct _ns1__getStockInfoByCode request;  
  30.     struct _ns1__getStockInfoByCodeResponse response;  
  31.   
  32.     request.theStockCode = argv[1];  
  33.     char *endpoint = NULL;  
  34.     if ( argc == 3 )  
  35.         endpoint = argv[2];  
  36.     if ( soap_call___ns3__getStockInfoByCode(&soap, endpoint, NULL, &request, &response) == SOAP_OK ) {  
  37.         int element_counter = response.getStockInfoByCodeResult->__sizestring;  
  38.         int i = 0;  
  39.         for ( i = 0; i < element_counter; i++ ) {  
  40.             switch ( i ) {  
  41.                 case 0  : printf("Stock code        : "); break;  
  42.                 case 1  : printf("Stock name        : "); break;  
  43.                 case 2  : printf("Timestamp         : "); break;  
  44.                 case 3  : printf("Latest price      : "); break;  
  45.                 case 4  : printf("Closing price T-1 : "); break;  
  46.                 case 5  : printf("Opening price     : "); break;  
  47.                 case 6  : printf("Ups and downs     : "); break;  
  48.                 case 7  : printf("Mininum price     : "); break;  
  49.                 case 8  : printf("Maxinum price     : "); break;  
  50.                 case 9  : printf("Amount of up/down : "); break;  
  51.                 case 10 : printf("Trading volume    : "); break;  
  52.                 case 11 : printf("Trading amount    : "); break;  
  53.                 case 12 : printf("Buy price         : "); break;  
  54.                 case 13 : printf("Sell price        : "); break;  
  55.                 case 14 : printf("Agency trans      : "); break;  
  56.                 case 15 : printf("Buy  1            : "); break;  
  57.                 case 16 : printf("Buy  2            : "); break;  
  58.                 case 17 : printf("Buy  3            : "); break;  
  59.                 case 18 : printf("Buy  4            : "); break;  
  60.                 case 19 : printf("Buy  5            : "); break;  
  61.                 case 20 : printf("Sell 1            : "); break;  
  62.                 case 21 : printf("Sell 2            : "); break;  
  63.                 case 22 : printf("Sell 3            : "); break;  
  64.                 case 23 : printf("Sell 4            : "); break;  
  65.                 case 24 : printf("Sell 5            : "); break;  
  66.                 default : break;  
  67.             }  
  68.             //printf("%s/n", response.getStockInfoByCodeResult->string[i]);  
  69.             size_t ilen = strlen(response.getStockInfoByCodeResult->string[i]);  
  70.             char output[OUTPUT_LEN];  
  71.             if ( conv_charset("GBK""UTF-8", response.getStockInfoByCodeResult->string[i], ilen, output, OUTPUT_LEN) )  
  72.                 printf("%s/n", response.getStockInfoByCodeResult->string[i]);  
  73.             else  
  74.                 printf("%s/n", output);  
  75.         }  
  76.     }  
  77.     else {  
  78.         soap_print_fault(&soap, stderr);  
  79.     }  
  80.   
  81.     soap_destroy(&soap);  
  82.     soap_end(&soap);  
  83.     soap_done(&soap);  
  84.     return 0;  
  85. }  

 

测试成功,如下图:

 

你可能感兴趣的:(使用gSOAP开发实例(3) iconv解决中文乱码问题)