1

我想创建一个 postgresql c-Language 函数,其参数和返回值是 varchar 并且在处理过程中使用 Char []。

但是,它没有按预期工作。

我做了一个这样的c程序。

#include <postgres.h>
#include <port.h>
#include <fmgr.h>
#include <stdlib.h>

#ifdef PG_MODULE_MAGIC
PG_MODULE_MAGIC;
#endif

// Required for windows.
extern PGDLLEXPORT Datum VARCHAR_CHAR_ARRAY_VARCHAR(PG_FUNCTION_ARGS);

PG_FUNCTION_INFO_V1(VARCHAR_CHAR_ARRAY_VARCHAR);

Datum VARCHAR_CHAR_ARRAY_VARCHAR(PG_FUNCTION_ARGS) {
    // Get arg.
    VarChar  *arg1 = (VarChar *)PG_GETARG_VARCHAR_P(0);

    // VarChar to Char[].
    char *c = (char *)VARDATA(arg1);

    elog(NOTICE, "VarChar to Char[]."); 
    elog(NOTICE, c);//Log1

    // Do something.(e.g. replace)
    // Since it uses another system, it must be Char [].

    //Char[] to VarChar.
    VarChar *rtn = (VarChar *)VARDATA(c);
    elog(NOTICE, "Char[] to VarChar.");
    elog(NOTICE, rtn);//Log2

    // Return VarChar.
    PG_RETURN_VARCHAR_P(rtn);
}

并且,创建了这样一个脚本。

CREATE OR REPLACE FUNCTION public.VARCHAR_CHAR_ARRAY_VARCHAR(character varying)
  RETURNS character varying AS
'$libdir/test/VARCHAR_CHAR_ARRAY_VARCHAR.dll', 'VARCHAR_CHAR_ARRAY_VARCHAR'
  LANGUAGE c VOLATILE STRICT;

执行结果是这样的。

SELECT VARCHAR_CHAR_ARRAY_VARCHAR('a');
-- Expected value: 'a'
-- Postgresql terminated abnormally... :(

SELECT VARCHAR_CHAR_ARRAY_VARCHAR('Action');
-- Expected return value: 'Action'
-- Actual return value  :'n'
-- Log1  :'NOTICE:  n'
-- Log2  :''

SELECT VARCHAR_CHAR_ARRAY_VARCHAR('ActionAction');
-- Expected return value: 'ActionAction'
-- Actual return value  : ''  <-Why?
-- Log1  :''
-- Log2  :''

SELECT VARCHAR_CHAR_ARRAY_VARCHAR('Action Action');
-- Expected return value: 'Action Action'
-- Actual return value  : 'n Action'
-- Log1  :'NOTICE:  Action Action'
-- Log2  :'NOTICE:  ction'

SELECT VARCHAR_CHAR_ARRAY_VARCHAR('1234567890');
-- Expected return value: '1234567890'
-- Actual return value  : '6789'
-- Log1  :'NOTICE:  1234567890'
-- Log2  :'NOTICE:  90'

执行环境:

  • Windows 服务器 2012 R2 x64
  • PostgreSQL 9.5.9,由 Visual C++ build 1800 编译,64 位

我是第一次使用 C 语言,所以很高兴提供示例代码。

对不起,我的英语不好。

4

1 回答 1

4

手动构建一个textVarChar类似的东西是行不通的。你不能只是

VarChar *rtn = (VarChar *)VARDATA(c);

因为你在那里做的是试图将char*值解释c为 a VarChar*,将前几个字节作为VARLENA标题读取,然后在其中获取一些子字符串作为值。不去上班。

容易使用cstring_to_texttext_to_cstring转换。text*与 完全兼容VarChar*。(他们在utils/builtins.h)。

例如

 PG_RETURN_VARCHAR_P(cstring_to_text(c));

要手动执行此操作,必须palloc使用带有strlen bytes + VARHDRSIZE,的字符串SET_VARSIZE,然后是偏移量strcpy的数据。VARDATAsrc/include/utils/varlena.hsrc/backend/utils/adt/varlena.c。当 PostgreSQL 有用于此目的的辅助函数时,这是繁琐且不必要的。

我个人建议完全忽略varcharC 级编程。将您的函数定义为接受和返回text

于 2017-10-27T04:47:35.143 回答