如何使 Postgresql 的psql 使用 中文提示信息

磨砺技术珠矶,践行数据之道,追求卓越价值 

回到上一级页面: PostgreSQL基础知识与基本操作索引页     回到顶级页面:PostgreSQL索引页

[作者 高健@博客园  luckyjackgao@gmail.com] 

 

和很多的开源软件一样,Postgresql 中使用  GNU 的 gettext 机制来完成多语言变换。

 它在自己的src各子目录下准备了很多的po文件,比如 src/bin/psql 目录下的:

zh_CN.po , jp.po, fr.po 等。

 按理说,configure 时,加入 –enable-nls=zh_CN 就可以了。但是执行了很多次都没有成功。

由于一个偶然的因素,才发现了其中的奥秘,现在把正确的作法说明如下:

 -------------正确作法开始------------------

步骤一

./configure
gmake
gmake install

 

步骤二

删除postgresql安装文件所在目录(就是包含configure文件的那个),

重新解压 tar文件,然后再执行 

 

./configure --enable-nls=zh_CN
gmake

gmake install

-------------正确作法完了------------------

 

再把错误方法说明一下:

-------------错误作法开始------------------

./configure --enable-nls=zh_CN

gmake

gmake install

-------------错误作法完了------------------

直接用 ./configure –enable-nls=zh_CN的时候,在执行 gmake时候,会碰到如下错误

"undefined reference to  libpq_gettext"

 

其原因是这样的:PostgreSQL中调用 gettext的时候,设计了一个函数libpq_gettext。

而这个函数却是定义在 fe-misc.c 文件中的。

在./configure —enable-nls 的情况下,

对libpq_gettext 的调用,将转化为寻找外部函数 libpq_gettext的过程。

由于PostgreSQL开发者的疏忽,此时 定义 libpq_gettext函数的 fe-misc尚未被编译,libpq_gettext尚未存在,所以gmake会出错。

 

反过来,./configure 没有加 –enable-nls参数的情况下,

根据宏定义,对libpq_gettext(x)调用,直接转换为参数x本身。尚未涉及fe-misc, 所以gmake会成功。

具体来说,请参看如下的函数定义信息。

可以这样地推断:

PostgreSQL的开发者,也是先进行了一般的编译/链接,然后并没有删除目标文件就开始调试加参数的configure/gmake/gmake install。

所以他们没有发现这个纰漏。因此,当我们加各种参数来编译PostgreSQL无法通过的时候,可以参考本文的方法。

 

最后,看一下psql下使用中文信息的效果:

 [postgresql@localhost ~]$ /usr/local/pgsql/bin/psql

psql(9.0.2)
输入 "help" 来获取帮助信息.

postgres=# \password
输入新的密码:
再次输入:
postgres=#\q

而如果不加 ./configure --enable-nls ,则信息为:

Enter new password:

Enter password again:

----------------------------------------------------函数定义信息开始---------------------------------------------

Libpq_int.h中的宏如下定义:

#ifdef ENABLE_NLS
    libpq_gettext(const char *msgid)
   __attribute__(format_arg(1));
#else
    #define libpq_gettext(x)  (x)
#endif  

fe_misc.c中对 libpq_gettext是有明确的定义的:

#ifdef ENABLE_NLS
char *libpq_gettext(const char *msgid)
{
    static bool already_bound = false;
     if (!already_bound)
     {
        /** dgettext() preserves errno, but bindtextdomain() doesn't */
        #ifdef WIN32
        int  save_errno = GetLastError();
        #else
        int  save_errno = errno;
        #endif

        const char *ldir;
        already_bound = true;
        /** No relocatable lookup here because the binary could be anywhere */
        ldir = getenv("PGLOCALEDIR");

        if (!ldir)
            ldir = LOCALEDIR;

        bindtextdomain(PG_TEXTDOMAIN("libpq"), ldir);

        #ifdef WIN32
        SetLastError(save_errno);
        #else
        errno = save_errno;
        #endif
    }
    return dgettext(PG_TEXTDOMAIN("libpq"), msgid);
}

----------------------------------------------------函数定义信息结束---------------------------------------------

 

[作者 高健@博客园  luckyjackgao@gmail.com]  

回到上一级页面: PostgreSQL基础知识与基本操作索引页     回到顶级页面:PostgreSQL索引页

磨砺技术珠矶,践行数据之道,追求卓越价值

 

posted @ 2012-07-10 15:56  健哥的数据花园  阅读(3241)  评论(0编辑  收藏  举报