Обсуждение: Libpq binary mode SELECT ... WHERE ID IN ($1) Question
Hello,
Suppose I have a table:
create table foo ( id int4, name varchar(50) );
I want to prepare and execute in binary mode:
select name from foo where id in ($1);
Execute works when I have a single value for $1, and treat it as a
normal INT4. However, when I attempt to send multiple values for the
parameter, I get no meaningful results.
My code snippets are below.
When setting up an array of numbers to pass as a parameter, is there
something special that needs to be done? The documentation is quite
vague; I did poke around the source and found in contrib some int_array
code, but don't fully understand it.
I suspect I need to do something like ...
(a) when I prepare, do something to tell postgres that I will have an
array of values, instead of a single value, and/or
(b) perhaps encapsulate the array of integers in some manner.
If I could be pointed to an example or documentation, it would be much
appreciated.
Thanks,
Matt Sanchez
The prepare code snippet:
Oid oids[1] = { 23 }; //INT4OID
result = PQprepare( pgconn, "getname",
"select name from foo where id in ($1)"
1, oids );
The execute code snippet:
int ids[4] = { 3, 5, 6, 8 }; // param values we want to setup
/* convert numbers to network byte order ... */
char * paramvalues[1] = (char *) ids;
int paramlengths[1];
int paramformats[1] = { 1 };
paramlengths[0] = = 4 * sizeof( int );
result = PQexecPrepared( pgconn,
"getname", // statement name
1, // number of params
paramvalues,
paramlenths,
paramformats,
1 );
You could try to prepare a query like this:
select name from foo where id=any($1);
and then pass the array of integers as $1 (although, I don't know how
you can do that as I've never used the C interface of libpq).
On Fri, Nov 13, 2009 at 1:34 AM, Matt Sanchez <matt-sanchez@comcast.net> wrote:
> Hello,
>
> Suppose I have a table:
> create table foo ( id int4, name varchar(50) );
>
> I want to prepare and execute in binary mode:
> select name from foo where id in ($1);
>
> Execute works when I have a single value for $1, and treat it as a
> normal INT4. However, when I attempt to send multiple values for the
> parameter, I get no meaningful results.
>
> My code snippets are below.
>
> When setting up an array of numbers to pass as a parameter, is there
> something special that needs to be done? The documentation is quite
> vague; I did poke around the source and found in contrib some int_array
> code, but don't fully understand it.
>
> I suspect I need to do something like ...
> (a) when I prepare, do something to tell postgres that I will have an
> array of values, instead of a single value, and/or
> (b) perhaps encapsulate the array of integers in some manner.
>
> If I could be pointed to an example or documentation, it would be much
> appreciated.
>
> Thanks,
> Matt Sanchez
>
> The prepare code snippet:
>
> Oid oids[1] = { 23 }; //INT4OID
>
> result = PQprepare( pgconn, "getname",
> "select name from foo where id in ($1)"
> 1, oids );
>
> The execute code snippet:
>
> int ids[4] = { 3, 5, 6, 8 }; // param values we want to setup
>
> /* convert numbers to network byte order ... */
>
> char * paramvalues[1] = (char *) ids;
> int paramlengths[1];
> int paramformats[1] = { 1 };
>
> paramlengths[0] = = 4 * sizeof( int );
>
> result = PQexecPrepared( pgconn,
> "getname", // statement name
> 1, // number of params
> paramvalues,
> paramlenths,
> paramformats,
> 1 );
>
>
> --
> Sent via pgsql-general mailing list (pgsql-general@postgresql.org)
> To make changes to your subscription:
> http://www.postgresql.org/mailpref/pgsql-general
>
On Fri, Nov 13, 2009 at 1:34 AM, Matt Sanchez <matt-sanchez@comcast.net> wrote:
> Hello,
>
> Suppose I have a table:
> create table foo ( id int4, name varchar(50) );
>
> I want to prepare and execute in binary mode:
> select name from foo where id in ($1);
>
> Execute works when I have a single value for $1, and treat it as a
> normal INT4. However, when I attempt to send multiple values for the
> parameter, I get no meaningful results.
>
> My code snippets are below.
>
> When setting up an array of numbers to pass as a parameter, is there
> something special that needs to be done? The documentation is quite
> vague; I did poke around the source and found in contrib some int_array
> code, but don't fully understand it.
>
> I suspect I need to do something like ...
> (a) when I prepare, do something to tell postgres that I will have an
> array of values, instead of a single value, and/or
> (b) perhaps encapsulate the array of integers in some manner.
>
> If I could be pointed to an example or documentation, it would be much
> appreciated.
>
> Thanks,
> Matt Sanchez
>
> The prepare code snippet:
>
> Oid oids[1] = { 23 }; //INT4OID
>
> result = PQprepare( pgconn, "getname",
> "select name from foo where id in ($1)"
> 1, oids );
>
> The execute code snippet:
>
> int ids[4] = { 3, 5, 6, 8 }; // param values we want to setup
>
> /* convert numbers to network byte order ... */
>
> char * paramvalues[1] = (char *) ids;
> int paramlengths[1];
> int paramformats[1] = { 1 };
>
> paramlengths[0] = = 4 * sizeof( int );
>
> result = PQexecPrepared( pgconn,
> "getname", // statement name
> 1, // number of params
> paramvalues,
> paramlenths,
> paramformats,
> 1 );
Here is how to do it with libpqtypes (http://libpqtypes.esilo.com/).
libpqtypes will reinvent the way you use libpq.
PGint4 i;
PGarray arr;
PGparam *param;
/* One dimesional arrays do not require setting dimesion info. For
* convience, you can zero the structure or set ndims to zero.
*/
arr.ndims = 0;
/* create the param object that will contain the elements */
arr.param = PQparamCreate(conn);
/* Use PQputf(3) to put the array elements */
for(i=0; i < 1000; i++)
PQputf(arr.param, "%int4", i);
/* The PGarray must be put into a PGparam struture. So far, only
* the array elements have been put. 'param' can continue to be
* used to pack more parameters. The array is now a single parameter
* within 'param'.
*/
param = PQparamCreate(conn);
PQputf(param, "%int[]", &arr);
/* no longer needed */
PQparamClear(arr.param);
/* send it up :-) */
PQparamExec(conn, param, "select unnest($1)", 1);
merlin
Merlin Moncure wrote:
> On Fri, Nov 13, 2009 at 1:34 AM, Matt Sanchez <matt-sanchez@comcast.net> wrote:
>> Hello,
>>
>> Suppose I have a table:
>> create table foo ( id int4, name varchar(50) );
>>
>> I want to prepare and execute in binary mode:
>> select name from foo where id in ($1);
>>
>> Execute works when I have a single value for $1, and treat it as a
>> normal INT4. However, when I attempt to send multiple values for the
>> parameter, I get no meaningful results.
>>
>> My code snippets are below.
>>
>> When setting up an array of numbers to pass as a parameter, is there
>> something special that needs to be done? The documentation is quite
>> vague; I did poke around the source and found in contrib some int_array
>> code, but don't fully understand it.
>>
>> I suspect I need to do something like ...
>> (a) when I prepare, do something to tell postgres that I will have an
>> array of values, instead of a single value, and/or
>> (b) perhaps encapsulate the array of integers in some manner.
>>
>> If I could be pointed to an example or documentation, it would be much
>> appreciated.
>>
>> Thanks,
>> Matt Sanchez
>>
>> The prepare code snippet:
>>
>> Oid oids[1] = { 23 }; //INT4OID
>>
>> result = PQprepare( pgconn, "getname",
>> "select name from foo where id in ($1)"
>> 1, oids );
>>
>> The execute code snippet:
>>
>> int ids[4] = { 3, 5, 6, 8 }; // param values we want to setup
>>
>> /* convert numbers to network byte order ... */
>>
>> char * paramvalues[1] = (char *) ids;
>> int paramlengths[1];
>> int paramformats[1] = { 1 };
>>
>> paramlengths[0] = = 4 * sizeof( int );
>>
>> result = PQexecPrepared( pgconn,
>> "getname", // statement name
>> 1, // number of params
>> paramvalues,
>> paramlenths,
>> paramformats,
>> 1 );
>
> Here is how to do it with libpqtypes (http://libpqtypes.esilo.com/).
> libpqtypes will reinvent the way you use libpq.
>
> PGint4 i;
> PGarray arr;
> PGparam *param;
>
> /* One dimesional arrays do not require setting dimesion info. For
> * convience, you can zero the structure or set ndims to zero.
> */
> arr.ndims = 0;
>
> /* create the param object that will contain the elements */
> arr.param = PQparamCreate(conn);
>
> /* Use PQputf(3) to put the array elements */
> for(i=0; i < 1000; i++)
> PQputf(arr.param, "%int4", i);
>
> /* The PGarray must be put into a PGparam struture. So far, only
> * the array elements have been put. 'param' can continue to be
> * used to pack more parameters. The array is now a single parameter
> * within 'param'.
> */
> param = PQparamCreate(conn);
> PQputf(param, "%int[]", &arr);
>
> /* no longer needed */
> PQparamClear(arr.param);
>
> /* send it up :-) */
> PQparamExec(conn, param, "select unnest($1)", 1);
>
> merlin
>
>
/* For prepared statements, use this */
PQparamExecPrepared(conn, param, "getname", 1);
/* Here is another way of doing it. You can putf more than
* one array element at a time. Also, PQexecf is handy
* but has no support for prepared statements :(
*/
PGresult *res;
PGarray arr = {0};
arr.param = param = PQparamCreate(conn);
PQputf(arr.param, "%int4 %int4 %int4 %int4", 3, 5, 6, 8);
res = PQexecf(conn,
"select name from foo where id in (%int4[])", &arr);
PQparamClear(arr.param);
--
Andrew Chernow
eSilo, LLC
every bit counts
http://www.esilo.com/