| PostgreSQL 7.4.5 Documentation | ||||
|---|---|---|---|---|
| Prev | Fast Backward | Chapter 27. libpq - C Library | Fast Forward | Next | 
The PQexec function is adequate for submitting commands in
normal, synchronous
applications.  It has a couple of deficiencies, however, that can be of importance to some users:
PQexec waits for the command to be completed.  The application may have other
work to do (such as maintaining a user interface), in which case it won't
want to block waiting for the response.
Since the execution of the client application is suspended while it waits for the result, it is hard for the application to decide that it would like to try to cancel the ongoing command. (It can be done from a signal handler, but not otherwise.)
PQexec can return only one PGresult structure.  If the submitted command
string contains multiple SQL commands, all but the last PGresult are
discarded by PQexec.
Applications that do not like these limitations can instead use the
underlying functions that PQexec is built from:
PQsendQuery and PQgetResult.
There are also PQsendQueryParams and
PQsendQueryPrepared, which can be used with
PQgetResult to duplicate the functionality of
PQexecParams and PQexecPrepared
respectively.
PQsendQuery          Submits a command to the server without
          waiting for the result(s).  1 is returned if the command was
          successfully dispatched and 0 if not (in which case, use
          PQerrorMessage to get more information about the failure).
int PQsendQuery(PGconn *conn, const char *command);
          After successfully calling PQsendQuery, call
          PQgetResult one or more
          times to obtain the results.  PQsendQuery may not be called
          again (on the same connection) until PQgetResult has returned a null pointer,
          indicating that the command is done.
PQsendQueryParamsSubmits a command and separate parameters to the server without waiting for the result(s).
int PQsendQueryParams(PGconn *conn,
                      const char *command,
                      int nParams,
                      const Oid *paramTypes,
                      const char * const *paramValues,
                      const int *paramLengths,
                      const int *paramFormats,
                      int resultFormat);
        This is equivalent to PQsendQuery except that
        query parameters can be specified separately from the query string.
        The function's parameters are handled identically to
        PQexecParams.  Like
        PQexecParams, it will not work on 2.0-protocol
        connections, and it allows only one command in the query string.
PQsendQueryPreparedSends a request to execute a prepared statement with given parameters, without waiting for the result(s).
int PQsendQueryPrepared(PGconn *conn,
                        const char *stmtName,
                        int nParams,
                        const char * const *paramValues,
                        const int *paramLengths,
                        const int *paramFormats,
                        int resultFormat);
        This is similar to PQsendQueryParams, but the
        command to be executed is specified by naming a previously-prepared
        statement, instead of giving a query string.
        The function's parameters are handled identically to
        PQexecPrepared.  Like
        PQexecPrepared, it will not work on 2.0-protocol
        connections.
PQgetResult          Waits for the next result from a prior
          PQsendQuery,
          PQsendQueryParams, or
          PQsendQueryPrepared call,
          and returns it.  A null pointer is returned when the command is complete
          and there will be no more results.
PGresult *PQgetResult(PGconn *conn);
          PQgetResult must be called repeatedly until it returns a null pointer,
          indicating that the command is done.  (If called when no command is
          active, PQgetResult will just return a null pointer at once.)
          Each non-null result from PQgetResult should be processed using
          the same PGresult accessor functions previously described.
          Don't forget to free each result object with PQclear when done with it.
          Note that PQgetResult will block only if a command is active and the
          necessary response data has not yet been read by PQconsumeInput.
Using PQsendQuery and PQgetResult
solves one of PQexec's problems:
If a command string contains multiple SQL commands, the results of those
commands can be obtained individually.  (This allows a simple form of
overlapped processing, by the way: the client can be handling the
results of one command while the server is still working on later
queries in the same command string.)  However, calling PQgetResult will
still cause the client to block until the server completes the
next SQL command.  This can be avoided by proper use of two
more functions:
PQconsumeInputIf input is available from the server, consume it.
int PQconsumeInput(PGconn *conn);
PQconsumeInput normally returns 1 indicating "no error",
but returns 0 if there was some kind of trouble (in which case
PQerrorMessage can be consulted).  Note that the result
does not say 
whether any input data was actually collected. After calling
PQconsumeInput, the application may check
PQisBusy and/or PQnotifies to see if
their state has changed.
PQconsumeInput may be called even if the application is not
prepared to deal with a result or notification just yet.  The
function will read available data and save it in a buffer, thereby
causing a select() read-ready indication to go away.  The
application can thus use PQconsumeInput to clear the
select() condition immediately, and then examine the results at leisure.
PQisBusyReturns 1 if a command is busy, that is, PQgetResult would block
waiting for input.  A 0 return indicates that PQgetResult can
be called with assurance of not blocking.
int PQisBusy(PGconn *conn);
PQisBusy will not itself attempt to read data from the server;
therefore PQconsumeInput must be invoked first, or the busy
state will never end.
A typical application using these functions will have a main loop that uses
select() or poll() to wait for all the
conditions that it must
respond to.  One of the conditions will be input available from the server,
which in terms of select() means readable data on the file
descriptor identified by PQsocket.
When the main loop detects input ready, it should call
PQconsumeInput to read the input.  It can then call
PQisBusy, followed by PQgetResult
if PQisBusy returns false (0).  It can also call
PQnotifies to detect NOTIFY messages (see Section 27.6).
A client that uses
PQsendQuery/PQgetResult can
also attempt to cancel a command that is still being processed by the
server.
PQrequestCancelRequests that the server abandon processing of the current command.
int PQrequestCancel(PGconn *conn);
The return value is 1 if the cancel request was successfully
dispatched and 0 if not.  (If not, PQerrorMessage tells why not.)
Successful dispatch is no guarantee that the request will have any
effect, however.  Regardless of the return value of PQrequestCancel,
the application must continue with the normal result-reading
sequence using PQgetResult.  If the cancellation
is effective, the current command will terminate early and return
an error result.  If the cancellation fails (say, because the
server was already done processing the command), then there will
be no visible result at all.
Note that if the current command is part of a transaction block, cancellation will abort the whole transaction.
PQrequestCancel can safely be invoked from a signal handler.
So, it is also possible to use it in conjunction with plain
PQexec, if the decision to cancel can be made in a signal
handler.  For example, psql invokes
PQrequestCancel from a SIGINT signal handler, thus allowing
interactive cancellation of commands that it issues through PQexec.
By using the functions described above, it is possible to avoid blocking while waiting for input from the database server. However, it is still possible that the application will block waiting to send output to the server. This is relatively uncommon but can happen if very long SQL commands or data values are sent. (It is much more probable if the application sends data via COPY IN, however.) To prevent this possibility and achieve completely nonblocking database operation, the following additional functions may be used.
PQsetnonblockingSets the nonblocking status of the connection.
int PQsetnonblocking(PGconn *conn, int arg);
Sets the state of the connection to nonblocking if arg is 1, or blocking if arg is 0. Returns 0 if OK, -1 if error.
    In the nonblocking state, calls to
    PQsendQuery,
    PQputline, PQputnbytes,
    and PQendcopy
    will not block but instead return an error if they need to be called
    again.
   
    Note that PQexec does not honor nonblocking mode;
    if it is called, it will act in blocking fashion anyway.
   
PQisnonblockingReturns the blocking status of the database connection.
int PQisnonblocking(const PGconn *conn);
Returns 1 if the connection is set to nonblocking mode and 0 if blocking.
PQflushAttempts to flush any queued output data to the server. Returns 0 if successful (or if the send queue is empty), -1 if it failed for some reason, or 1 if it was unable to send all the data in the send queue yet (this case can only occur if the connection is nonblocking).
int PQflush(PGconn *conn);
After sending any command or data on a nonblocking connection, call
PQflush.  If it returns 1, wait for the socket to be
write-ready and call it again; repeat until it returns 0.  Once
PQflush returns 0, wait for the socket to be read-ready
and then read the response as described above.