3 OCI Programming Basics
This chapter introduces concepts and procedures involved in programming with OCI.
After reading this chapter, you should have most of the tools necessary to understand and create a basic OCI application.
This chapter includes the following major sections:
New users should pay particular attention to the information presented in this chapter, because it forms the basis for the rest of the material presented in this guide. The information in this chapter is supplemented by information in later chapters.
See Also:
-
Oracle Database Globalization Support Guide for a discussion of the OCI functions that apply to a multilingual environment
-
Oracle Database Data Cartridge Developer's Guide for a discussion of the OCI functions that apply to cartridge services
Overview of OCI Program Programming
The general goal of an OCI application is to operate on behalf of multiple users.
In an n-tiered configuration, multiple users are sending HTTP requests to the client application. The client application may need to perform some data operations that include exchanging data and performing data processing.
OCI uses the following basic program flow:
-
Create the environment by initializing the OCI programming environment and threads.
-
Allocate necessary handles, and establish server connections and user sessions.
-
Exchange data with the database server by executing SQL statements on the server, and perform necessary application data processing.
-
Execute prepared statements, or prepare a new statement for execution.
-
Terminate user sessions and disconnect from server connections.
-
Free handles and data structures.
Figure 3-1 illustrates the flow of steps in an OCI application. OCI Programming Steps describes each step in more detail.
The diagram and the list of steps present a simple generalization of OCI programming steps. Variations are possible, depending on the functionality of the program. OCI applications that include more sophisticated functionality, such as managing multiple sessions and transactions and using objects, require additional steps.
All OCI function calls are executed in the context of an environment. There can be multiple environments within an OCI process. If an environment requires any process-level initialization, then it is performed automatically.
Note:
It is possible to have multiple active connections and statements in an OCI application.
See Also:
OCI Object-Relational Programming through Using the Object Type Translator with OCI for information about accessing and manipulating objects
OCI Data Structures
Handles and descriptors are opaque data structures that are defined in OCI applications.
Handles and descriptors can be allocated directly, through specific allocate calls, or they can be implicitly allocated by OCI functions.
Note:
Programmers who have previously written 7.x OCI applications must become familiar with these data structures that are used by most OCI calls.
Handles and descriptors store information pertaining to data, connections, or application behavior. Handles are defined in more detail in the next section.
This section includes the following topics: Handles
Related Topics
Handles
Almost every OCI call includes in its parameter list one or more handles.
A handle is an opaque pointer to a storage area allocated by the OCI library. You use a handle to store context or connection information, (for example, an environment or service context handle), or it may store information about OCI functions or data (for example, an error or describe handle). Handles can make programming easier, because the library, rather than the application, maintains this data.
Most OCI applications must access the information stored in handles. The get and set attribute OCI calls, OCIAttrGet()
and OCIAttrSet()
, access and set this information.
Table 3-1 lists the handles defined for OCI. For each handle type, the C data type and handle type constant used to identify the handle type in OCI calls are listed.
Table 3-1 OCI Handle Types
Description | C Data Type | Handle Type Constant |
---|---|---|
OCI environment handle |
|
|
OCI error handle |
|
|
OCI service context handle |
|
|
OCI statement handle |
|
|
OCI bind handle |
|
|
OCI define handle |
|
|
OCI describe handle |
|
|
OCI server handle |
|
|
OCI user session handle |
|
|
OCI authentication information handle |
|
|
OCI connection pool handle |
|
|
OCI session pool handle |
|
|
OCI transaction handle |
|
|
OCI complex object retrieval (COR) handle |
|
|
OCI thread handle |
|
Not applicable |
OCI subscription handle |
|
|
OCI direct path context handle |
|
|
OCI direct path function context handle |
|
|
OCI direct path column array handle |
|
|
OCI direct path stream handle |
|
|
OCI process handle |
|
|
OCI administration handle |
|
|
OCI HA event handle |
|
Not applicable |
OCI SODA collection handle |
|
|
OCI SODA collection cursor handle |
|
|
OCI SODA document cursor handle |
|
|
OCI SODA document handle |
|
|
OCI SODA output options handle |
|
|
OCI SODA operation options handle |
|
|
Related Topics
About Allocating and Freeing Handles
Your application allocates all handles (except the bind, define, and thread handles) for a particular environment handle.
You pass the environment handle as one of the parameters to the handle allocation call. The allocated handle is then specific to that particular environment.
The bind and define handles are allocated for a statement handle, and contain information about the statement represented by that handle.
Note:
The bind and define handles are implicitly allocated by the OCI library, and do not require user allocation.
The environment handle is allocated and initialized with a call to OCIEnvCreate()
or to OCIEnvNlsCreate()
, one of which is required by all OCI applications.
All user-allocated handles are initialized using the OCI handle allocation call, OCIHandleAlloc()
.
The types of handles include: session pool handle, direct path context handle, thread handle, COR handle, subscription handle, describe handle, statement handle, service context handle, error handle, server handle, connection pool handle, event handle, and administration handle.
The thread handle is allocated with the OCIThreadHndInit()
call.
An application must free all handles when they are no longer needed. The OCIHandleFree()
function frees all handles.
Note:
When a parent handle is freed, all child handles associated with it are also freed and can no longer be used. For example, when a statement handle is freed, any bind and define handles associated with it are also freed.
Handles lessen the need for global variables. Handles also make error reporting easier. An error handle is used to return errors and diagnostic information.
Environment Handle
The environment handle defines a context in which all OCI functions are invoked. Each environment handle contains a memory cache that enables fast memory access. All memory allocation under the environment handle is done from this cache. Access to the cache is serialized if multiple threads try to allocate memory under the same environment handle. When multiple threads share a single environment handle, they may block on access to the cache.
The environment handle is passed as the parent parameter to the OCIHandleAlloc() call to allocate all other handle types. Bind and define handles are allocated implicitly.
Error Handle
The error handle is passed as a parameter to most OCI calls.
The error handle maintains information about errors that occur during an OCI operation. If an error occurs in a call, the error handle can be passed to OCIErrorGet()
to obtain additional information about the error that occurred.
Allocating the error handle is one of the first steps in an OCI application because most OCI calls require an error handle as a parameter.
Related Topics
Service Context Handle and Associated Handles
A service context handle defines attributes that determine the operational context for OCI calls to a server.
The service context handle contains three handles as its attributes, that represent a server connection, a user session, and a transaction. These attributes are illustrated in Figure 3-2.
Figure 3-2 Components of a Service Context
Description of "Figure 3-2 Components of a Service Context"
-
A server handle identifies a connection to a database. It translates into a physical connection in a connection-oriented transport mechanism.
-
A user session handle defines a user's roles and privileges (also known as the user's security domain), and the operational context in which the calls execute.
-
A transaction handle defines the transaction in which the SQL operations are performed. The transaction context includes user session state information, including any fetch state and package instantiation.
Breaking the service context handle down in this way provides scalability and enables programmers to create sophisticated multitiered applications and transaction processing (TP) monitors to execute requests on behalf of multiple users on multiple application servers and different transaction contexts.
You must allocate and initialize the service context handle with OCIHandleAlloc()
, OCILogon()
, or OCILogon2()
before you can use it. The service context handle is allocated explicitly by OCIHandleAlloc()
. It can be initialized using OCIAttrSet()
with the server, user session, and transaction handle. If the service context handle is allocated implicitly using OCILogon()
, it is already initialized.
Applications maintaining only a single user session for each database connection at any time can call OCILogon()
to get an initialized service context handle.
In applications requiring more complex session management, the service context handle must be explicitly allocated, and the server and user session handles must be explicitly set into the service context handle. OCIServerAttach()
and OCISessionBegin()
calls initialize the server and user session handle respectively.
An application only defines a transaction explicitly if it is a global transaction or there are multiple transactions active for sessions. It works correctly with the implicit transaction created automatically by OCI when the application makes changes to the database.
See Also:
-
OCI Environment Initialization, and Password and Session Management for more information about establishing a server connection and user session
Statement, Bind, and Define Handles
A statement handle is the context that identifies a SQL or PL/SQL statement and its associated attributes
A statement handle is shown in Figure 3-3.
Information about input and output bind variables is stored in bind handles. The OCI library allocates a bind handle for each placeholder bound with the OCIBindByName()
or OCIBindByName2()
or OCIBindByPos()
or OCIBindByPos2()
function. The user must not allocate bind handles. They are implicitly allocated by the bind call.
Fetched data returned by a query (select statement) is converted and retrieved according to the specifications of the define handles. The OCI library allocates a define handle for each output variable defined with OCIDefineByPos()
or OCIDefineByPos2()
. The user must not allocate define handles. They are implicitly allocated by the define call.
Bind and define handles are implicitly allocated by the OCI library, and are transparently reused if the bind or define operation is repeated. The actual value of the bind or define handle is needed by the application for the advanced bind or define operations described in Binding and Defining in OCI. The handles are freed when the statement handle is freed or when a new statement is prepared on the statement handle. Explicitly allocating bind or define handles may lead to memory leaks. Explicitly freeing bind or define handles may cause abnormal program termination.
Describe Handle
The describe handle is used by the OCI describe call, OCIDescribeAny()
.
The OCIDescribeAny()
call obtains information about schema objects in a database (for example, functions or procedures). The call takes a describe handle as one of its parameters, along with information about the object being described. When the call completes, the describe handle is populated with information about the object. The OCI application can then obtain describe information through the attributes of the parameter descriptors.
See Also:
-
Describing Schema Metadata for more information about using the
OCIDescribeAny()
function
Complex Object Retrieval Handle
The complex object retrieval (COR) handle is used by some OCI applications that work with objects in an Oracle database.
The complex object retrieval (COR) handle contains COR descriptors, provides instructions for retrieving objects referenced by another object.
Related Topics
Thread Handle
The thread handle is used in multithreaded applications.
For information about the thread handle, which is used in multithreaded applications, see the following OCIThread Package.
Related Topics
Subscription Handle
The subscription handle is used by an OCI client application that registers and subscribes to receive notifications of database events or events in the AQ namespace.
The subscription handle encapsulates all information related to a registration from a client.
Related Topics
Direct Path Handles
The direct path handles are necessary for an OCI application that uses the direct path load engine in the Oracle database.
The direct path load interface enables the application to access the direct block formatter of the Oracle database. Figure 3-4 shows the different kinds of direct path handles.
Connection Pool Handle
The connection pool handle is used for applications that pool physical connections into virtual connections.
The connection pool handle is used for applications that pool physical connections into virtual connections by calling specific OCI functions.
Related Topics
Handle Attributes
All OCI handles have attributes that represent data stored in that handle.
You can read handle attributes by using the attribute get call, OCIAttrGet()
, and you can change them with the attribute set call, OCIAttrSet()
.
For example, the statements in Example 3-1 set the user name in the session handle by writing to the OCI_ATTR_USERNAME
attribute:
Some OCI functions require that particular handle attributes be set before the function is called. For example, when OCISessionBegin()
is called to establish a user's login session, the user name and password must be set in the user session handle before the call is made.
Other OCI functions provide useful return data in handle attributes after the function completes. For example, when OCIStmtExecute()
is called to execute a SQL query, describe information relating to the select-list items is returned in the statement handle, as shown in Example 3-2.
See Also:
-
The description of OCIArrayDescriptorAlloc() for an example showing how to allocate a large number of descriptors
Example 3-1 Using the OCI_ATTR_USERNAME Attribute to Set the User Name in the Session Handle
text username[] = "hr"; err = OCIAttrSet ((void *) mysessp, OCI_HTYPE_SESSION, (void *)username, (ub4) strlen((char *)username), OCI_ATTR_USERNAME, (OCIError *) myerrhp);
Example 3-2 Returning Describe Information in the Statement Handle Relating to Select-List Items
ub4 parmcnt; /* get the number of columns in the select list */ err = OCIAttrGet ((void *)stmhp, (ub4)OCI_HTYPE_STMT, (void *) &parmcnt, (ub4 *) 0, (ub4)OCI_ATTR_PARAM_COUNT, errhp);
OCI Descriptors
OCI descriptors and locators are opaque data structures that maintain data-specific information.
Table 3-2 lists OCI descriptors, along with their C data type, and the OCI type constant that allocates a descriptor of that type in a call to OCIDescriptorAlloc()
. The OCIDescriptorFree()
function frees descriptors and locators.
Table 3-2 Descriptor Types
Note:
Although there is a single C type for OCILobLocator
, this locator is allocated with a different OCI type constant for internal and external LOBs. LOB and BFILE Locators discusses this difference.
The following list describes the main purpose of each descriptor type. The sections that follow describe each descriptor type in more detail:
-
OCISnapshot
- Used in statement execution -
OCILobLocator
- Used for LOB (OCI_DTYPE_LOB
) orBFILE
(OCI_DTYPE_FILE
) calls OCIJson
- to represent a JSON document in OCI. It is identified by the descriptor type OCI_DTYPE_JSON-
OCIParam
- Used in describe calls -
OCIRowid
- Used for binding or definingROWID
values -
OCIDateTime
andOCIInterval
- Used for datetime and interval data types -
OCIComplexObjectComp
- Used for complex object retrieval -
OCIAQEnqOptions
,OCIAQDeqOptions
,OCIAQMsgProperties
,OCIAQAgent
- Used for Advanced Queuing -
OCIAQNotify
- Used for publish-subscribe notification -
OCIServerDNs
- Used for LDAP-based publish-subscribe notification
Snapshot Descriptor
The snapshot descriptor is an optional parameter to the execute call, OCIStmtExecute()
.
The snapshot descriptor indicates that a query is being executed against a database snapshot that represents the state of a database at a particular time.
Allocate a snapshot descriptor with a call to OCIDescriptorAlloc()
by passing OCI_DTYPE_SNAP
as the type
parameter.
Related Topics
LOB and BFILE Locators
A large object (LOB) is an Oracle data type that can hold binary large object (BLOB
) or character large object (CLOB
) data.
In the database, an opaque data structure called a LOB locator is stored in a LOB column of a database row, or in the place of a LOB attribute of an object. The locator serves as a pointer to the actual LOB value, which is stored in a separate location.
Note:
Depending on your application, you may or may not want to use LOB locators. You can use the data interface for LOBs, which does not require LOB locators. In this interface, you can bind or define character data for CLOB
columns or RAW
data for BLOB
columns.
The OCI LOB locator is used to perform OCI operations against a LOB (BLOB
or CLOB
) or FILE (BFILE
). OCILobXXX
functions take a LOB locator parameter instead of the LOB value. OCI LOB functions do not use actual LOB data as parameters. They use the LOB locators as parameters and operate on the LOB data referenced by them.
The LOB locator is allocated with a call to OCIDescriptorAlloc()
by passing OCI_DTYPE_LOB
as the type
parameter for BLOB
s or CLOB
s, and OCI_DTYPE_FILE
for BFILE
s.
Note:
The two LOB locator types are not interchangeable. When binding or defining a BLOB
or CLOB
, the application must take care that the locator is properly allocated by using OCI_DTYPE_LOB
. Similarly, when binding or defining a BFILE
, the application must be sure to allocate the locator using OCI_DTYPE_FILE
.
An OCI application can retrieve a LOB locator from the Oracle database by issuing a SQL statement containing a LOB column or attribute as an element in the select list. In this case, the application would first allocate the LOB locator and then use it to define an output variable. Similarly, a LOB locator can be used as part of a bind operation to create an association between a LOB and a placeholder in a SQL statement.
Parameter Descriptor
OCI applications use parameter descriptors to obtain information about select-list columns or schema objects.
This information is obtained through a describe operation.
The parameter descriptor is the only descriptor type that is not allocated using OCIDescriptorAlloc()
. You can obtain it only as an attribute of a describe handle, statement handle, or through a complex object retrieval handle by specifying the position of the parameter using an OCIParamGet()
call.
ROWID Descriptor
The ROWID
descriptor, OCIRowid
, is used by applications that must retrieve and use Oracle ROWIDs.
To work with a ROWID
an application can define a ROWID
descriptor for a rowid position in a SQL select list, and retrieve a ROWID
into the descriptor. This same descriptor can later be bound to an input variable in an INSERT
statement or WHERE
clause.
ROWID
s are also redirected into descriptors using OCIAttrGet()
on the statement handle following an execute operation.
Related Topics
Date, Datetime, and Interval Descriptors
The date, datetime, and interval descriptors are used by applications that use the date, datetime, or interval data types (OCIDate
, OCIDateTime
, and OCIInterval
).
These descriptors can be used for binding and defining, and are passed as parameters to the functions OCIDescriptorAlloc()
and OCIDescriptorFree()
to allocate and free memory.
Complex Object Descriptor
Complex object retrieval (COR) may improve application performance when dealing with objects.
Application performance when dealing with objects may be increased using complex object retrieval (COR).
Related Topics
Advanced Queuing Descriptors
There are a number of Oracle Database Advanced Queuing descriptors for use to maintain data-specific information.
Oracle Database Advanced Queuing provides message queuing as an integrated part of Oracle Database.
User Memory Allocation
The OCIDescriptorAlloc()
call has an xtramem_sz
parameter in its parameter list.
The xtramem_sz
parameter is used to specify the amount of user memory that should be allocated along with a descriptor or locator.
Typically, an application uses this parameter to allocate an application-defined structure that has the same lifetime as the descriptor or locator. This structure can be used for application bookkeeping or storing context information.
Using the xtramem_sz
parameter means that the application does not need to explicitly allocate and deallocate memory as each descriptor or locator is allocated and deallocated. The memory is allocated along with the descriptor or locator, and freeing the descriptor or locator (with OCIDescriptorFree()
) frees the user's data structures as well.
The OCIHandleAlloc()
call has a similar parameter for allocating user memory that has the same lifetime as the handle.
The OCIEnvCreate()
and (OCIEnvInit()
deprecated) calls have a similar parameter for allocating user memory that has the same lifetime as the environment handle.
OCI Programming Steps
The following sections describe in detail each of the steps in developing an OCI application.
Some of the steps are optional. For example, you do not need to describe or define select-list items if the statement is not a query. Application-specific processing also occurs in between any and all of the OCI function steps.
The following sections describe the steps that are required of an OCI application:
See Also:
-
The first sample program in OCI Demonstration Programs for an example showing the use of OCI calls for processing SQL statements.
-
Runtime Data Allocation and Piecewise Operations in OCI for a detailed description of the special case of dynamically providing data at run time
-
About Binding and Defining Arrays of Structures in OCI for a description of the special considerations for operations involving arrays of structures
-
Error Handling in OCI for an outline of the steps involved in processing a SQL statement within an OCI program
-
Overview of OCI Multithreaded Development for information about using the OCI to write multithreaded applications
-
SQL Statements for more information about types of SQL statements
OCI Environment Initialization
This section describes how to initialize the OCI environment, establish a connection to a server, and authorize a user to perform actions against the database.
First, the three main steps in initializing the OCI environment are described in the following sections:
About Creating the OCI Environment
Each OCI function call is executed in the context of an environment that is created with the OCIEnvCreate()
call.
The OCIEnvCreate()
call must be invoked before any other OCI call is executed. The only exception is the setting of a process-level attribute for the OCI shared mode.
The mode
parameter of OCIEnvCreate()
specifies whether the application calling the OCI library functions can:
-
Run in a threaded environment (
mode
=OCI_THREADED
). -
Use objects (
mode
=OCI_OBJECT
). Use with AQ subscription registration. -
Use subscriptions (
mode
=OCI_EVENTS
).
The mode can be set independently in each environment.
It is necessary to initialize in object mode if the application binds and defines objects, or if it uses the OCI's object navigation calls. The program may also choose to use none of these features (mode
= OCI_DEFAULT
) or some combination of them, separating the options with a vertical bar. For example if mode
= (OCI_THREADED
| OCI_OBJECT
), then the application runs in a threaded environment and uses objects.
You can specify user-defined memory management functions for each OCI environment.
See Also:
-
OCIEnvCreate(), OCIEnvNlsCreate(), and OCIInitialize() (deprecated) for more information about the initialization calls
-
OCI Object-Relational Programming, Object-Relational Data Types in OCI, Direct Path Load Interface, Object Advanced Topics in OCI, and Using the Object Type Translator with OCI
About Allocating Handles and Descriptors
Oracle Database provides OCI functions to allocate and deallocate handles and descriptors.
You must allocate handles using OCIHandleAlloc()
before passing them into an OCI call, unless the OCI call, such as OCIBindByPos()
or OCIBindByPos2()
, allocates the handles for you.
You can allocate the types of handles listed in Table 3-1 with OCIHandleAlloc()
Depending on the functionality of your application, it must allocate some or all of these handles.
Related Topics
Application Initialization, Connection, and Session Creation
An application must call OCIEnvNlsCreate()
to initialize the OCI environment handle. Existing applications may have used OCIEnvCreate()
.
Following this step, the application has several options for establishing an Oracle database connection and beginning a user session.
These methods include:
Note:
OCIEnvCreate()
or OCIEnvNlsCreate()
should be used instead of the OCIInitialize()
and OCIEnvInit()
calls. OCIInitialize()
and OCIEnvInit()
calls are supported for backward compatibility.
Single User, Single Connection
The single user, single connection option is the simplified logon method, which can be used if an application maintains only a single user session for each database connection at any time.
When an application calls OCILogon2()
or OCILogon()
, the OCI library initializes the service context handle that is passed to it, and creates a connection to the specified Oracle database for the user making the request.
Example 3-3 shows what a call to OCILogon2()
looks like for a single user session with user name hr
, password hr
, and database oracledb
.
The parameters to this call include the service context handle (which has been initialized), the user name, the user's password, and the name of the database that are used to establish the connection. With the last parameter, mode
, set to OCI_DEFAULT
, this call has the same effect as calling the older OCILogon()
. Use OCILogon2()
for any new applications. The server and user session handles are implicitly allocated by this function.
If an application uses this logon method, the service context, server, and user session handles are all read-only; the application cannot switch session or transaction by changing the appropriate attributes of the service context handle using an OCIAttrSet()
call.
An application that initializes its session and authorization using OCILogon2()
must terminate them using OCILogoff()
.
Note:
For simplicity in demonstrating this feature, this example does not perform the password management techniques that a deployed system normally uses. In a production environment, follow the Oracle Database password management guidelines, and disable any sample accounts. See Oracle Database Security Guide for password management guidelines and other security recommendations.
Example 3-3 Using the OCILogon2 Call for a Single User Session
OCILogon2(envhp, errhp, &svchp, (text *)"hr", (ub4)strlen("hr"), (text *)"hr", (ub4)strlen("hr"), (text *)"oracledb", (ub4)strlen("oracledb"), OCI_DEFAULT);
See Also:
Operating System Considerations for information regarding operating systems providing facilities for spawning processes that allow child processes to reuse state created by their parent process. This section explains why the child process must not use the same database connection as created by the parent.
Client Access Through a Proxy
Proxy authentication is a process typically employed in an environment with a middle tier such as a firewall, in which the end user authenticates to the middle tier, which then authenticates to the database on the user's behalf—as its proxy.
The middle tier logs in to the database as a proxy user. A proxy user can switch identities and, after logging in to the database, switch to the end user's identity. It can perform operations on the end user's behalf, using the authorization appropriate to that particular end user.
Note:
In release 1 of Oracle 11g, standards for acceptable passwords were greatly raised to increase security. Examples of passwords in this section are incorrect. A password must contain no fewer than eight characters. See the guidelines for securing passwords Oracle Database Security Guide for additional information.
Proxy to database users is supported by using OCI and the ALTER
USER
statement, whose BNF syntax is:
ALTER USER <targetuser> GRANT CONNECT THROUGH <proxy> [AUTHENTICATION REQUIRED];
The ALTER
USER
statement is used once in an application. Connections can be made multiple times afterward. In OCI, you can either use connect strings or the function OCIAttrSet()
with the parameter OCI_ATTR_PROXY_CLIENT
.
Even though beginning with Oracle Database 12c
Release 2 (12.2) the maximum length of each identifier is increased to 128 bytes, the user name and proxy combination can not exceed 250 bytes.
After a proxy switch is made, the current and connected user is the target user of the proxy. The identity of the original user is not used for any privilege calculations. The original user can be a local or external user.
Example 3-4 through Example 3-11 show connect strings that you can use in functions such as OCILogon2()
(set mode
= OCI_DEFAULT
), OCILogon()
, OCISessionBegin()
with OCIAttrSet()
(pass the attribute OCI_ATTR_USERNAME
of the session handle), and so on.
In Example 3-4, Dilbert and Joe are two local database users. To enable Dilbert to serve as a proxy for Joe, use the SQL statement shown in Example 3-4.
When user name dilbert
is acting on behalf of joe
, use the connection string shown in Example 3-5. (The user name dilbert
has the password tiger123
).
The left and right brackets "[" and "]" are entered in the connection string.
In Example 3-6, "Dilbert" and "Joe" are two local database users. The names are case-sensitive and must be enclosed in double quotation marks. To enable "Dilbert" to serve as a proxy for "Joe", use the SQL statement shown in Example 3-6.
When "Dilbert" is acting on behalf of "Joe", use the connection string shown in Example 3-7. Be sure to include the double quotation marks (") characters.
When the proxy user is created as "dilbert[mybert]", use the connection string shown in Example 3-8 to connect to the database. (The left and right brackets "[" and "]" are entered in the connection string.)
In Example 3-9, dilbert[mybert] and joe[myjoe] are two database users that contain the left and right bracket characters "[" and "]". If dilbert[mybert] wants to act on behalf of joe[myjoe], Example 3-9 shows the connect statement to use.
In Example 3-10, you can set the target user name by using the ALTER
USER
statement.
Note:
There are compatibility issues of client access through a proxy. Because this feature was introduced in Oracle Database release 10.2, pre-10.2 clients do not have it. If newer clients use the feature with pre-10.2 Oracle databases, the connect fails and the client returns an error after checking the database release level.
Example 3-4 Enabling a Local User to Serve as a Proxy for Another User
ALTER USER joe GRANT CONNECT THROUGH dilbert;
Example 3-5 Connection String to Use for the Proxy User
dilbert[joe]/tiger123@db1
Example 3-6 Preserving Case Sensitivity When Enabling a Local User to Serve as a Proxy for Another User
ALTER USER "Joe" GRANT CONNECT THROUGH "Dilbert";
Example 3-7 Preserving Case Sensitivity in the Connection String
"Dilbert"["Joe"]/tiger123@db1
Example 3-8 Using "dilbert[mybert]" in the Connection String
"dilbert[mybert]"/tiger123 rem the user was already created this way: rem CREATE USER "dilbert[mybert]" IDENTIFIED BY tiger123;
Example 3-9 Using "dilbert[mybert]"["joe[myjoe]"] in the Connection String
"dilbert[mybert]"["joe[myjoe]"]/tiger123
Example 3-10 Setting the Target User Name
ALTER USER joe GRANT CONNECT THROUGH dilbert;
Then, as shown in Example 3-11, in an OCI program, use the OCIAttrSet()
call to set the attribute OCI_ATTR_PROXY_CLIENT
and the proxy dilbert
. In your program, use these statements to connect multiple times.
Example 3-11 Using OCI to Set the OCI_ATTR_PROXY_CLIENT Attribute and the Proxy dilbert
OCIAttrSet(session, OCI_HTYPE_SESSION, (void *)"dilbert", (ub4)strlen("dilbert"), OCI_ATTR_USERNAME, error_handle); OCIAttrSet(session, OCI_HTYPE_SESSION, (void *)"tiger123", (ub4)strlen("tiger123"), OCI_ATTR_PASSWORD, error_handle); OCIAttrSet(session, OCI_HTYPE_SESSION, (void *)"joe", (ub4)strlen("joe"), OCI_ATTR_PROXY_CLIENT, error_handle);
Nonproxy Multiple Sessions or Connections
The nonproxy multiple sessions or connections option uses explicit attach and begin-session calls to maintain multiple user sessions and connections on a database connection.
Specific calls to attach to the Oracle database and begin sessions are:
-
OCIServerAttach()
- Creates an access path to the Oracle database for OCI operations. -
OCISessionBegin()
- Establishes a session for a user against a particular Oracle database. This call is required for the user to execute operations on the Oracle database.
A subsequent call to OCISessionBegin()
using different service context and session context handles logs off the previous user and causes an error. To run two simultaneous nonmigratable sessions, a second OCISessionBegin()
call must be made with the same service context handle and a new session context handle.
These calls set up an operational environment that enables you to execute SQL and PL/SQL statements against a database.
See Also:
-
Session Pooling and Connection Pooling in OCI for more information about maintaining multiple sessions, transactions, and connections
-
Client Character Set Control from OCI for the use of
OCIEnvNlsCreate()
Example 3-12 demonstrates the creation and initialization of an OCI environment.
-
A server context is created and set in the service handle.
-
Then a user session handle is created and initialized using a database user name and password.
-
For simplicity, error checking is not included.
The demonstration program cdemo81.c
in the demo
directory illustrates this process, with error checking.
Example 3-12 Creating and Initializing an OCI Environment
#include <oci.h> ... main() { ... OCIEnv *myenvhp; /* the environment handle */ OCIServer *mysrvhp; /* the server handle */ OCIError *myerrhp; /* the error handle */ OCISession *myusrhp; /* user session handle */ OCISvcCtx *mysvchp; /* the service handle */ ... /* initialize the mode to be the threaded and object environment */ (void) OCIEnvCreate(&myenvhp, OCI_THREADED|OCI_OBJECT, (void *)0, 0, 0, 0, (size_t) 0, (void **)0); /* allocate a server handle */ (void) OCIHandleAlloc ((void *)myenvhp, (void **)&mysrvhp, OCI_HTYPE_SERVER, 0, (void **) 0); /* allocate an error handle */ (void) OCIHandleAlloc ((void *)myenvhp, (void **)&myerrhp, OCI_HTYPE_ERROR, 0, (void **) 0); /* create a server context */ (void) OCIServerAttach (mysrvhp, myerrhp, (text *)"inst1_alias", strlen ("inst1_alias"), OCI_DEFAULT); /* allocate a service handle */ (void) OCIHandleAlloc ((void *)myenvhp, (void **)&mysvchp, OCI_HTYPE_SVCCTX, 0, (void **) 0); /* set the server attribute in the service context handle*/ (void) OCIAttrSet ((void *)mysvchp, OCI_HTYPE_SVCCTX, (void *)mysrvhp, (ub4) 0, OCI_ATTR_SERVER, myerrhp); /* allocate a user session handle */ (void) OCIHandleAlloc ((void *)myenvhp, (void **)&myusrhp, OCI_HTYPE_SESSION, 0, (void **) 0); /* set user name attribute in user session handle */ (void) OCIAttrSet ((void *)myusrhp, OCI_HTYPE_SESSION, (void *)"hr", (ub4)strlen("hr"), OCI_ATTR_USERNAME, myerrhp); /* set password attribute in user session handle */ (void) OCIAttrSet ((void *)myusrhp, OCI_HTYPE_SESSION, (void *)"hr", (ub4)strlen("hr"), OCI_ATTR_PASSWORD, myerrhp); (void) OCISessionBegin ((void *) mysvchp, myerrhp, myusrhp, OCI_CRED_RDBMS, OCI_DEFAULT); /* set the user session attribute in the service context handle*/ (void) OCIAttrSet ((void *)mysvchp, OCI_HTYPE_SVCCTX, (void *)myusrhp, (ub4) 0, OCI_ATTR_SESSION, myerrhp); ... }
About Processing SQL Statements in OCI
What are the specific steps involved in processing SQL statements in OCI.
Using SQL Statements in OCI outlines the specific steps involved in processing SQL statements in OCI.
Commit or Roll Back Operations
An application commits changes to the database by calling OCITransCommit()
.
The OCITransCommit()
call uses a service context as one of its parameters. The transaction is associated with the service context whose changes are committed. This transaction can be explicitly created by the application or implicitly created when the application modifies the database.
Note:
By using the OCI_COMMIT_ON_SUCCESS
mode of the OCIStmtExecute()
call, the application can selectively commit transactions after each statement execution, saving an extra round-trip.
To roll back a transaction, use the OCITransRollback()
call.
If an application disconnects from Oracle Database in a way other than a normal logoff, such as losing a network connection, and OCITransCommit()
has not been called, all active transactions are rolled back automatically.
About Terminating the Application
What should an application do before it terminates.
An OCI application should perform the following steps before it terminates:
-
Delete the user session by calling
OCISessionEnd()
for each session. -
Delete access to the data sources by calling
OCIServerDetach()
for each source. -
Explicitly deallocate all handles by calling
OCIHandleFree()
for each handle. -
Delete the environment handle, which deallocates all other handles associated with it.
Note:
When a parent OCI handle is freed, any child handles associated with it are freed automatically
The calls to OCIServerDetach()
and OCISessionEnd()
are not mandatory but are recommended. If the application terminates, and OCITransCommit()
(transaction commit) has not been called, any pending transactions are automatically rolled back.
See Also:
The first sample program in OCI Demonstration Programs for an example showing handles being freed at the end of an application
Note:
If the application uses the simplified logon method of OCILogon2()
, then a call to OCILogoff()
terminates the session, disconnects from the Oracle database, and frees the service context and associated handles. The application is still responsible for freeing other handles it allocated.
Error Handling in OCI
OCI function calls have a set of return codes.
These OCI function call return codes are listed in Table 3-3, which indicate the success or failure of the call, such as OCI_SUCCESS
or OCI_ERROR
, or provide other information that may be required by the application, such as OCI_NEED_DATA
or OCI_STILL_EXECUTING
. Most OCI calls return one of these codes.
To verify that the connection to the server is not terminated by the OCI_ERROR
, an application can check the value of the attribute OCI_ATTR_SERVER_STATUS
in the server handle. If the value of the attribute is OCI_SERVER_NOT_CONNECTED
, then the connection to the server and the user session must be reestablished.
See Also:
-
"Functions Returning Other Values" for exceptions
-
"OCIErrorGet()" for complete details and an example of usage
Table 3-3 OCI Return Codes
OCI Return Code | Value | Description |
---|---|---|
0 |
The function completed successfully. |
|
1 |
The function completed successfully; a call to |
|
100 |
The function completed, and there is no further data. |
|
-1 |
The function failed; a call to |
|
-2 |
An invalid handle was passed as a parameter or a user callback was passed an invalid handle or invalid context. No further diagnostics are available. |
|
99 |
The application must provide runtime data. |
|
-3123 |
The service context was established in nonblocking mode, and the current operation could not be completed immediately. The operation must be called again to complete. |
|
-24200 |
This code is returned only from a callback function. It indicates that the callback function wants the OCI library to resume its normal processing. |
|
-24201 |
This code is returned only from a callback function. It indicates that the callback function is done with the user row callback. |
If the return code indicates that an error has occurred, the application can retrieve error codes and messages specific to Oracle Database by calling OCIErrorGet()
. One of the parameters to OCIErrorGet()
is the error handle passed to the call that caused the error.
Note:
Multiple diagnostic records can be retrieved by calling OCIErrorGet()
repeatedly until there are no more records (OCI_NO_DATA
is returned). OCIErrorGet()
returns at most a single diagnostic record.
Return and Error Codes for Data
The OCI return code, error number, indicator variable, and column return code are specified when the data fetched is normal, null, or truncated.
In Table 3-4, the OCI return code, error number, indicator variable, and column return code are specified when the data fetched is normal, null, or truncated.
See Also:
Table 3-4 Return and Error Codes
State of Data | Return Code | Indicator - Not provided | Indicator - Provided |
---|---|---|---|
Not null or truncated |
Not provided |
Error = 0 |
Error = 0 Indicator = 0 |
Not null or truncated |
Provided |
Error = 0 Return code = 0 |
Error = 0 Indicator = 0 Return code = 0 |
Null data |
Not provided |
Error = 1405 |
Error = 0 Indicator = -1 |
Null data |
Provided |
Error = 1405 Return code = 1405 |
Error = 0 Indicator = -1 Return code = 1405 |
Truncated data |
Not provided |
Error = 1406 |
Error = 1406 Indicator = data_len |
Truncated data |
Provided |
Error = 24345 Return code = 1405 |
Error = 24345 Indicator = data_len Return code = 1406 |
For truncated data, data_len
is the actual length of the data that has been truncated if this length is less than or equal to SB2MAXVAL
. Otherwise, the indicator is set to -2.
Functions Returning Other Values
Some functions return values other than the OCI error codes.
These other OCI error codes are listed in Table 3-3. When you use these functions, be aware that they return values directly from the function call, rather than through an OUT
parameter. More detailed information about each function and its return values is listed in the reference chapters.
Additional Coding Guidelines
This section explains some additional issues when coding OCI applications.
Operating System Considerations
Operating systems may provide facilities for spawning processes that allow child processes to reuse the state created by their parent process.
After spawning a child process, the child process must not use the same database connection as created by the parent. Any attempt on behalf of the child process to use the same database connection as the parent may cause undesired connection interference and result in intermittent ORA-03137
errors, because Oracle Net expects only one user process to be using a connection to the database.
Where multiple, concurrent connections are required, consider using threads if your platform supports a threads package. Concurrent connections are supported in either single-threaded or multithreaded applications. For better performance with many concurrently opened connections, consider pooling them.
Parameter Types
OCI functions take a variety of different types of parameters, including integers, handles, and character strings.
This section includes the following topics:
See Also:
Connect, Authorize, and Initialize Functions for more information about parameter data types and parameter passing conventions
Address Parameters
Address parameters are used to pass the address of the variable to Oracle Database.
You should be careful when developing in C, because it normally passes scalar parameters by value.
Integer Parameters
Binary integer and short binary integer parameters are numbers whose size is system-dependent.
See Oracle Database documentation that is specific to your operating system for the size of these integers on your system.
Character String Parameters
Character strings are a special type of address parameter.
Each OCI routine that enables a character string to be passed as a parameter also has a string length parameter. The length parameter should be set to the length of the string.
Note:
Unlike earlier versions of OCI, you do not pass -1 for the string length parameter of a null-terminated string.
Inserting Nulls into a Column
How to insert a null into a database column.
You can insert a null into a database column in several ways.
-
One method is to use a literal
NULL
in the text of anINSERT
orUPDATE
statement. For example, the SQL statement makes theENAME
columnNULL
.INSERT INTO emp1 (ename, empno, deptno) VALUES (NULL, 8010, 20)
-
Use indicator variables in the OCI bind call.
-
Insert a
NULL
to set both the buffer length and maximum length parameters to zero on a bind call.Note:
Following the SQL standard requirements, Oracle Database returns an error if an attempt is made to fetch a null select-list item into a variable that does not have an associated indicator variable specified in the define call.
Related Topics
Indicator Variables
Each bind and define OCI call has a parameter that associates an indicator variable, or an array of indicator variables, with a DML statement, a PL/SQL statement, or a query.
The C language does not have the concept of null values; therefore, you associate indicator variables with input variables to specify whether the associated placeholder is a NULL
. When data is passed to an Oracle database, the values of these indicator variables determine whether a NULL
is assigned to a database field.
For output variables, indicator variables determine whether the value returned from Oracle is a NULL
or a truncated value. For a NULL
fetch in an OCIStmtFetch2()
call or a truncation in an OCIStmtExecute()
call, the OCI call returns OCI_SUCCESS_WITH_INFO
. The output indicator variable is set.
The data type of indicator variables is sb2
. For arrays of indicator variables, the individual array elements should be of type sb2
.
Input
What values can be assigned to an indicator variable.
For input host variables, the OCI application can assign the following values to an indicator variable:
Input Indicator Value | Action Taken by Oracle Database |
---|---|
-1 |
Oracle Database assigns a |
>=0 |
Oracle Database assigns the value of the input variable to the column. |
Output
What values can be assigned to an indicator variable.
On output, Oracle Database can assign the following values to an indicator variable:
Output Indicator Value | Meaning |
---|---|
-2 |
The length of the item is greater than the length of the output variable; the item has been truncated. Additionally, the original length is longer than the maximum data length that can be returned in the |
-1 |
The selected value is null, and the value of the output variable is unchanged. |
0 |
Oracle Database assigned an intact value to the host variable. |
>0 |
The length of the item is greater than the length of the output variable; the item has been truncated. The positive value returned in the indicator variable is the actual length before truncation. |
Indicator Variables for Named Data Types and REFs
Indicator variables for most data types introduced after release 8.0 behave as described earlier.
The only exception is SQLT_NTY (a named data type). For data of type SQLT_NTY, the indicator variable must be a pointer to an indicator structure. Data of type SQLT_REF uses a standard scalar indicator, just like other variable types.
When database types are translated into C struct representations using the Object Type Translator (OTT), a null indicator structure is generated for each object type. This structure includes an atomic null indicator, plus indicators for each object attribute.
See Also:
-
Documentation for the OTT in Using the Object Type Translator with OCI, and NULL Indicator Structure for information about NULL indicator structures
-
Descriptions of
OCIBindByName()
andOCIBindByPos()
in Bind, Define, and Describe Functions, and the sections Information for Named Data Type and REF Binds, and Information for Named Data Type and REF Defines, and PL/SQL OUT Binds for more information about setting indicator parameters for named data types andREF
s
About Canceling Calls
How do you cancel long-running or repeated OCI calls.
On most operating systems, you can cancel long-running or repeated OCI calls by entering the operating system's interrupt character (usually Control+C) from the keyboard.
Note:
This is not to be confused with canceling a cursor, which is accomplished by calling OCIStmtFetch2()
with the nrows
parameter set to zero.
When you cancel the long-running or repeated call using the operating system interrupt, the error code ORA-01013
("user requested cancel of current operation") is returned.
When given a particular service context pointer or server context pointer, the OCIBreak()
function performs an immediate (asynchronous) stop of any currently executing OCI function associated with the server. It is normally used to stop a long-running OCI call being processed on the server. The OCIReset()
function is necessary to perform a protocol synchronization on a nonblocking connection after an OCI application stops a function with OCIBreak()
.
Note:
OCIBreak()
works on Windows systems.
The status of potentially long-running calls can be monitored using nonblocking calls. Use multithreading for new applications.
Positioned Updates and Deletes
You can use the ROWID
associated with a SELECT
...FOR
UPDATE
OF
... statement in a later UPDATE
or DELETE
statement.
The ROWID
is retrieved by calling OCIAttrGet()
on the statement handle to retrieve the handle's OCI_ATTR_ROWID
attribute.
For example, consider a SQL statement such as the following:
SELECT ename FROM emp1 WHERE empno = 7499 FOR UPDATE OF sal
When the fetch is performed, the ROWID
attribute in the handle contains the row identifier of the selected row. You can retrieve the ROWID
into a buffer in your program by calling OCIAttrGet()
as follows:
OCIRowid *rowid; /* the rowid in opaque format */ /* allocate descriptor with OCIDescriptorAlloc() */ status = OCIDescriptorAlloc ((void *) envhp, (void **) &rowid, (ub4) OCI_DTYPE_ROWID, (size_t) 0, (void **) 0); status = OCIAttrGet ((void *) mystmtp, OCI_HTYPE_STMT, (void *) rowid, (ub4 *) 0, OCI_ATTR_ROWID, (OCIError *) myerrhp);
You can then use the saved ROWID
in a DELETE
or UPDATE
statement. For example, if rowid
is the buffer in which the row identifier has been saved, you can later process a SQL statement such as the following by binding the new salary to the :1
placeholder and rowid
to the :2
placeholder.
UPDATE emp1 SET sal = :1 WHERE rowid = :2
Be sure to use data type code 104 (ROWID
descriptor, see Table 4-2) when binding rowid
to :2
.
By using prefetching, you can select an array of ROWID
s for use in subsequent batch updates.
See Also:
-
External Data Types for a table of external data types and codes
Reserved Words
Some words are reserved by Oracle.
That is, some reserved words have a special meaning to Oracle and cannot be redefined. For this reason, you cannot use them to name database objects such as columns, tables, or indexes.
This section includes the following topic: Oracle Reserved Namespaces
See Also:
Oracle Database SQL Language Reference and Oracle Database PL/SQL Language Reference to view the lists of the Oracle keywords or reserved words for SQL and PL/SQL
Oracle Reserved Namespaces
What namespaces are reserved by Oracle. For a complete list of functions within a particular namespace, refer to the document that corresponds to the appropriate Oracle library.
Table 3-5 contains a list of namespaces that are reserved by Oracle. The initial characters of function names in Oracle libraries are restricted to the character strings in this list. Because of potential name conflicts, do not use function names that begin with these characters.
Table 3-5 Oracle Reserved Namespaces
Namespace | Library |
---|---|
|
External functions for XA applications only |
|
External SQLLIB functions used by Oracle Precompiler and SQL*Module applications |
|
External OCI functions internal OCI functions |
|
Function names from the Oracle UPI layer |
|
Oracle Net Native Services Product Oracle Net RPC Project Oracle Net Directory Oracle Net Network Library Layer Oracle Net Management Project Oracle Net Interchange Oracle Net Transparent Network Service Oracle Net Drivers Oracle Net Security Service Oracle Net V1 Oracle Net Two Task |
|
Core library functions |
|
Function names from the Oracle Globalization Support layer |
|
Function names from system-dependent libraries |
|
Kernel Objects |
Polling Mode Operations in OCI
OCI has calls that poll for completion.
Examples of such polling mode calls are:
-
OCI calls in nonblocking mode
-
OCI calls that operate on LOB data in pieces such as
OCILobRead2()
andOCILobWrite2()
-
OCIStmtExecute()
andOCIStmtFetch2()
when used withOCIStmtSetPieceInfo()
andOCIStmtGetPieceInfo()
In such cases, OCI requires that the application ensure that the same OCI call is repeated on the connection and nothing else is done on the connection in the interim. Performing any other OCI call on such a connection (when OCI has handed control back to the caller) can result in unexpected behavior.
Hence, with such polling mode OCI calls, the caller must ensure that the same call is repeated on the connection and that nothing else is done until the call completes.
OCIBreak()
and OCIReset()
are exceptions to the rule. These calls are allowed so that the caller can stop an OCI call that has been started.
Nonblocking Mode in OCI
OCI provides the ability to establish a server connection in blocking mode or nonblocking mode.
Note:
Because nonblocking mode requires the caller to repeat the same call until it completes, it increases CPU usage. Instead, use multithreaded mode.
When a connection is made in blocking mode, an OCI call returns control to an OCI client application only when the call completes, either successfully or in error. With the nonblocking mode, control is immediately returned to the OCI program if the call could not complete, and the call returns a value of OCI_STILL_EXECUTING
.
In nonblocking mode, an application must test the return code of each OCI function to see if it returns OCI_STILL_EXECUTING
. If it does, the OCI client can continue to process program logic while waiting to retry the OCI call to the server. This mode is particularly useful in graphical user interface (GUI) applications, real-time applications, and in distributed environments.
The nonblocking mode is not interrupt-driven. Rather, it is based on a polling paradigm, which means that the client application must check whether the pending call is finished at the server by executing the call again with the exact same parameters.
The following features and functions are not supported in nonblocking mode:
-
Direct Path Load
-
LOB buffering
-
Objects
-
Query cache
-
Scrollable cursors
-
Transparent application failover (TAF)
-
OCIAQEnqArray
() -
OCIAQDeqArray()
-
OCIDescribeAny()
-
OCILobArrayRead()
-
OCILobArrayWrite()
-
OCITransStart()
-
OCITransDetach()
Setting Blocking Modes
You can modify or check an application's blocking status by calling OCIAttrSet()
to set the status, or OCIAttrGet()
to read the status on the server context handle with the attrtype
parameter set to OCI_ATTR_NONBLOCKING_MODE
.
You must set this OCI_ATTR_NONBLOCKING_MODE
attribute only after OCISessionBegin()
or OCILogon2()
has been called. Otherwise, an error is returned.
Note:
Only functions that have a server context or a service context handle as a parameter can return OCI_STILL_EXECUTING
.
See Also:
About Using PL/SQL in an OCI Program
PL/SQL is Oracle's procedural extension to the SQL language.
PL/SQL supports tasks that are more complicated than simple queries and SQL data manipulation language (DML) statements. PL/SQL enables you to group some constructs into a single block and execute it as a unit. These constructs include:
-
One or more SQL statements
-
Variable declarations
-
Assignment statements
-
Procedural control statements such as
IF...THEN...ELSE
statements and loops -
Exception handling
You can use PL/SQL blocks in your OCI program to perform the following operations:
-
Call Oracle stored procedures and stored functions
-
Combine procedural control statements with several SQL statements, to be executed as a unit
-
Access special PL/SQL features such as tables,
CURSOR FOR
loops, and exception handling -
Use cursor variables
-
Operate on objects in a server
Note:
-
Although OCI can only directly process anonymous blocks, and not named packages or procedures, you can always put the package or procedure call within an anonymous block and process that block.
-
Note that all OUT variables must be initialized to
NULL
(through an indicator of -1, or an actual length of 0) before a PL/SQL begin-end block can be executed in OCI. -
OCI does not support the PL/SQL
RECORD
data type. -
When binding a PL/SQL VARCHAR2 variable in OCI, the maximum size of the bind variable is 32767 bytes, because of the overhead of control structures.
Note:
When you write PL/SQL code, it is important to remember that the parser treats everything between a pair of hyphens"--" and a carriage return character as a comment. So if comments are indicated on each line by "--", the C compiler can concatenate all lines in a PL/SQL block into a single line without putting a carriage return "\n" for each line. In this particular case, the parser fails to extract the PL/SQL code of a line if the previous line ends with a comment. To avoid the problem, the programmer should put "\n" after each "--" comment to ensure that the comment ends there.
See Also:
Oracle Database PL/SQL Language Reference for information about coding PL/SQL blocks
-
OCI Globalization Support
The following sections introduce OCI functions that can be used for globalization purposes, such as deriving locale information, manipulating strings, character set conversion, and OCI messaging.
These functions are also described in detail in other chapters of this guide because they have multiple purposes and functionality.
Client Character Set Control from OCI
The function OCIEnvNlsCreate()
enables you to set character set information in applications independently from NLS_LANG
and NLS_NCHAR
settings.
OCI_UTF16ID
cannot be set from NLS_LANG
or NLS_NCHAR
and must be set using OCIEnvNlsCreate()
. One application can have several environment handles initialized within the same system environment using different client-side character set IDs and national character set IDs. For example:
OCIEnvNlsCreate(OCIEnv **envhpp, ..., csid, ncsid);
In this example, csid
is the value for the character set ID for the parameter charset
, and ncsid
is the value for the national character set ID for the parameter ncharset
. Either can be 0 or OCI_UTF16ID
. If both are 0, this is equivalent to using OCIEnvCreate()
instead. The other arguments are the same as for the OCIEnvCreate()
call.
Any Oracle character set ID, except AL16UTF16, can be specified through the OCIEnvNlsCtrate()
function to specify the encoding of metadata, SQL CHAR
data, and SQL NCHAR
data.
You can retrieve character sets in NLS_LANG
and NLS_NCHAR
through another function, OCINlsEnvironmentVariableGet()
.
See Also:
-
About Setting Client Character Sets in OCI for a pseudocode fragment that illustrates a sample usage of these calls
Character Control and OCI Interfaces
How is character control performed by OCI interfaces.
The OCINlsGetInfo()
function returns information about any character set, including OCI_UTF16ID
if this value has been used in OCIEnvNlsCreate()
.
The OCIAttrGet()
function returns the character set ID and national character set ID that were passed into OCIEnvNlsCreate()
. This is used to get OCI_ATTR_ENV_CHARSET_ID
and OCI_ATTR_ENV_NCHARSET_ID
. This includes the value OCI_UTF16ID
.
If both charset
and ncharset
parameters were set to 0 by OCIEnvNlsCreate()
, the character set IDs in NLS_LANG
and NLS_NCHAR
are returned.
The OCIAttrSet()
function sets character IDs as the defaults if OCI_ATTR_CHARSET_FORM
is reset through this function. The eligible character set IDs include OCI_UTF16ID
if OCIEnvNlsCreate()
is passed as charset
or ncharset
.
The OCIBindByName()
or OCIBindByName2()
and OCIBindByPos()
or OCIBindByPos2()
functions bind variables with the default character set in the OCIEnvNlsCreate()
call, including OCI_UTF16ID
. The actual length and the returned length are always in bytes if OCIEnvNlsCreate()
is used.
The OCIDefineByPos()
or OCIDefineByPos2()
function defines variables with the value of charset
in OCIEnvNlsCreate()
, including OCI_UTF16ID
, as the default. The actual length and returned length are always in bytes if OCIEnvNlsCreate()
is used. This behavior for bind and define handles is different from that when OCIEnvCreate()
is used and OCI_UTF16ID
is the character set ID for the bind and define handles.
Character-Length Semantics in OCI
OCI works as a translator between server and client, and passes around character information for constraint checking.
There are two kinds of character sets: variable-width and fixed-width. (A single-byte character set is a special case of a fixed-width character set where each byte stands for one character.)
For fixed-width character sets, constraint checking is easier, as the number of bytes is equal to a multiple of the number of characters. Therefore, scanning of the entire string is not needed to determine the number of characters for fixed-width character sets. However, for variable-width character sets, complete scanning is needed to determine the number of characters in a string.
Character Set Support in OCI
How does OCI support character sets.
See Character-Length Semantics Support in Describe Operations and Character Conversion in OCI Binding and Defining for a complete discussion of character set support in OCI.
Controlling Language and Territory in OCI
The NLS language and territory can also be set programmatically using the attributes OCI_ATTR_ENV_NLS_LANGUAGE
and OCI_ATTR_ENV_NLS_TERRITORY
on OCI environment handle.
Other OCI Globalization Support Functions
Many globalization support functions accept either the environment handle or the user session handle.
The OCI environment handle is associated with the client NLS environment variables. This environment does not change when ALTER
SESSION
statements are issued to the server. The character set associated with the environment handle is the client character set. The OCI session handle (returned by OCISessionBegin()
) is associated with the server session environment. The NLS settings change when the session environment is modified with an ALTER
SESSION
statement. The character set associated with the session handle is the database character set.
Note that the OCI session handle does not have NLS settings associated with it until the first transaction begins in the session. SELECT
statements do not begin a transaction.
See Also:
-
Oracle Database Globalization Support Guide for information about OCI programming with Unicode
About Getting Locale Information in OCI
An Oracle Database locale consists of language, territory, and character set definitions.
The locale determines conventions such as day and month names, as well as date, time, number, and currency formats. A globalized application follows a user's locale setting and cultural conventions. For example, when the locale is set to German, users expect to see day and month names in German.
You can retrieve the following information with the OCINlsGetInfo()
function:
-
Days of the week (translated)
-
Abbreviated days of the week (translated)
-
Month names (translated)
-
Abbreviated month names (translated)
-
Yes/no (translated)
-
AM/PM (translated)
-
AD/BC (translated)
-
Numeric format
-
Debit/credit
-
Date format
-
Currency formats
-
Default language
-
Default territory
-
Default character set
-
Default linguistic sort
-
Default calendar
The code in Example 3-13 retrieves locale information and checks for errors.
Example 3-13 Getting Locale Information in OCI
sword MyPrintLinguisticName(envhp, errhp) OCIEnv *envhp; OCIError *errhp; { OraText infoBuf[OCI_NLS_MAXBUFSZ]; sword ret; ret = OCINlsGetInfo(envhp, /* environment handle */ errhp, /* error handle */ infoBuf, /* destination buffer */ (size_t) OCI_NLS_MAXBUFSZ, /* buffer size */ (ub2) OCI_NLS_LINGUISTIC_NAME); /* item */ if (ret != OCI_SUCCESS) { checkerr(errhp, ret, OCI_HTYPE_ERROR); ret = OCI_ERROR; } else { printf("NLS linguistic: %s\n", infoBuf); } return(ret); }
About OCI and the BOM (Byte Order Mark)
OCI does not support nor handle the BOM (byte order mark) and assumes that the byte order is native to the machine on which your application is executing.
Your OCI application must not pass a string containing a BOM expecting that OCI can detect the encoding of the string. Your OCI application must remove the BOM if it exists in the string being passed in and ensure that it is in the encoding the OCI function expects.
About Manipulating Strings in OCI
Multibyte strings and wide-character strings are supported for string manipulation.
Multibyte strings are encoded in native Oracle character sets. Functions that operate on multibyte strings take the string as a whole unit with the length of the string calculated in bytes. Wide-character string (wchar
) functions provide more flexibility in string manipulation. They support character-based and string-based operations where the length the string calculated in characters.
The wide-character data type, OCIWchar
, is Oracle-specific and should not be confused with the wchar_t
data type defined by the ANSI/ISO C standard. The Oracle wide-character data type is always 4 bytes in all operating systems, whereas the size of wchar_t
depends on the implementation and the operating system. The Oracle wide-character data type normalizes multibyte characters so that they have a uniform fixed width for easy processing. This guarantees no data loss for round-trip conversion between the Oracle wide-character set and the native character set.
String manipulation can be classified into the following categories:
-
Conversion of strings between multibyte and wide character
-
Character classifications
-
Case conversion
-
Calculations of display length
-
General string manipulation, such as comparison, concatenation, and searching
See Also:
Example 3-14 shows a simple case of manipulating strings.
The OCI character classification functions are described in detail in OCI Character Classification Functions.
Example 3-15 shows how to classify characters in OCI.
Example 3-14 Basic String Manipulation in OCI
size_t MyConvertMultiByteToWideChar(envhp, dstBuf, dstSize, srcStr) OCIEnv *envhp; OCIWchar *dstBuf; size_t dstSize; OraText *srcStr; /* null terminated source string */ { sword ret; size_t dstLen = 0; size_t srcLen; /* get length of source string */ srcLen = OCIMultiByteStrlen(envhp, srcStr); ret = OCIMultiByteInSizeToWideChar(envhp, /* environment handle */ dstBuf, /* destination buffer */ dstSize, /* destination buffer size */ srcStr, /* source string */ srcLen, /* length of source string */ &dstLen); /* pointer to destination length */ if (ret != OCI_SUCCESS) { checkerr(envhp, ret, OCI_HTYPE_ENV); } return(dstLen); }
Example 3-15 Classifying Characters in OCI
boolean MyIsNumberWideCharString(envhp, srcStr) OCIEnv *envhp; OCIWchar *srcStr; /* wide char source string */ { OCIWchar *pstr = srcStr; /* define and init pointer */ boolean status = TRUE; /* define and initialize status variable */ /* Check input */ if (pstr == (OCIWchar*) NULL) return(FALSE); if (*pstr == (OCIWchar) NULL) return(FALSE); /* check each character for digit */ do { if (OCIWideCharIsDigit(envhp, *pstr) != TRUE) { status = FALSE; break; /* non-decimal digit character */ } } while ( *++pstr != (OCIWchar) NULL); return(status); }
About Converting Character Sets in OCI
Conversion between Oracle character sets and Unicode (16-bit, fixed-width Unicode encoding) is supported.
Replacement characters are used if a character has no mapping from Unicode to the Oracle character set. Therefore, conversion back to the original character set is not always possible without data loss.
Character set conversion functions involving Unicode character sets require data bind and define buffers to be aligned at a ub2
address or an error is raised.
Example 3-16 shows a simple conversion into Unicode.
See Also:
Example 3-16 Converting Character Sets in OCI
/* Example of Converting Character Sets in OCI --------------------------------------------*/ size_t MyConvertMultiByteToUnicode(envhp, errhp, dstBuf, dstSize, srcStr) OCIEnv *envhp; OCIError *errhp; ub2 *dstBuf; size_t dstSize; OraText *srcStr; { size_t dstLen = 0; size_t srcLen = 0; OraText tb[OCI_NLS_MAXBUFSZ]; /* NLS info buffer */ ub2 cid; /* OCIEnv character set ID */ /* get OCIEnv character set */ checkerr(errhp, OCINlsGetInfo(envhp, errhp, tb, sizeof(tb), OCI_NLS_CHARACTER_SET)); cid = OCINlsCharSetNameToId(envhp, tb); if (cid == OCI_UTF16ID) { ub2 *srcStrUb2 = (ub2*)srcStr; while (*srcStrUb2++) ++srcLen; srcLen *= sizeof(ub2); } else srcLen = OCIMultiByteStrlen(envhp, srcStr); checkerr(errhp, OCINlsCharSetConvert( envhp, /* environment handle */ errhp, /* error handle */ OCI_UTF16ID, /* Unicode character set ID */ dstBuf, /* destination buffer */ dstSize, /* size of destination buffer */ cid, /* OCIEnv character set ID */ srcStr, /* source string */ srcLen, /* length of source string */ &dstLen)); /* pointer to destination length */ return dstLen/sizeof(ub2); }
OCI Messaging Functions
The user message API provides a simple interface for cartridge developers to retrieve their own messages and Oracle Database messages.
Example 3-17 creates a message handle, initializes it to retrieve messages from impus.msg
, retrieves message number 128, and closes the message handle. It assumes that OCI environment handles, OCI session handles, and the product, facility, and cache size have been initialized properly.
Example 3-17 Retrieving a Message from a Text Message File
OCIMsg msghnd; /* message handle */ /* initialize a message handle for retrieving messages from impus.msg*/ err = OCIMessageOpen(hndl,errhp, &msghnd, prod,fac,OCI_DURATION_SESSION); if (err != OCI_SUCCESS) /* error handling */ ... /* retrieve the message with message number = 128 */ msgptr = OCIMessageGet(msghnd, 128, msgbuf, sizeof(msgbuf)); /* do something with the message, such as display it */ ... /* close the message handle when there are no more messages to retrieve */ OCIMessageClose(hndl, errhp, msghnd);
lmsgen Utility
The lmsgen
utility converts text-based message files (.msg
) into binary format (.msb
) so that Oracle Database messages and OCI messages provided by the user can be returned to OCI functions in the desired language.
The BNF syntax of the Imsgen utility is as follows:
lmsgen
text_file
product
facility
[language
]
In the preceding syntax:
-
text_file
is a message text file. -
product
is the name of the product. -
facility
is the name of the facility. -
language
is the optional message language corresponding to the language specified in theNLS_LANG
parameter. The language parameter is required if the message file is not tagged properly with language.
Guidelines for Text Message Files
What are the guidelines that text message files must follow.
Text message files must follow these guidelines:
-
Lines that start with "
/"
and "//
" are treated as internal comments and are ignored. -
To tag the message file with a specific language, include a line similar to the following:
# CHARACTER_SET_NAME= Japanese_Japan.JA16EUC
-
Each message contains three fields:
message_number
,warning_level
,message_text
-
The message number must be unique within a message file.
-
The warning level is not currently used. Set to 0.
-
The message text cannot be longer than 76 bytes.
-
The following is an example of an Oracle Database message text file:
/ Copyright (c) 2001 by the Oracle Corporation. All rights reserved. / This is a test us7ascii message file # CHARACTER_SET_NAME= american_america.us7ascii / 00000, 00000, "Export terminated unsuccessfully\n" 00003, 00000, "no storage definition found for segment(%lu, %lu)"
An Example of Creating a Binary Message File from a Text Message File
How do you create a binary message file from a text message file.
The following table contains sample values for the lmsgen
parameters:
lmsgen Parameter | Value |
---|---|
|
|
|
|
|
|
|
|
The text message file is found in the following location:
$HOME/myApp/mesg/impus.msg
One of the lines in the text message file is:
00128,2, "Duplicate entry %s found in %s"
The lmsgen
utility converts the text message file (impus.msg
) into binary format, resulting in a file called impus.msb
:
% lmsgen impus.msg $HOME/myApplication imp AMERICAN
The following output results:
Generating message file impus.msg --> /home/scott/myApplication/mesg/impus.msb NLS Binary Message File Generation Utility: Version 9.2.0.0.0 -Production Copyright (c) Oracle Corporation 1979, 2001. All rights reserved. CORE 9.2.0.0.0 Production