Pro*C tutorial

Pro*C Tutorial

In this tutorial, we are going to cover the following topics:


What is Pro*C?

Pro*C means creating a program that is embedded with SQL statements. We can access the Oracle database with the help of Pro*C. The C programming language provides the flexibility for data processing by manipulating or retrieving the data from the Oracle database. Therefore, we can say that Pro*C allows C programming language to connect the C program with the Oracle database and do the manipulations according to our requirements.


How can we compile the Pro*C program?

Oracle software provides the PROC compiler (Oracle Precompiler), and this compiler takes the C program having embedded SQL statements.

What is Oracle precompiler?

An oracle precompiler is a programming tool that allows the user to embed the SQL statements in a high-level source program. This compiler takes the source program as input, replacing the embedded SQL statements with the oracle runtime library calls, and this modified program can now compile, link, and execute.

The whole process of compilation of Pro*C program is given below:

Pro*C Tutorial
  • First, we create a C program, which contains the SQL statements, and then we save this file with an extension '.PC'. Here, '.PC' means that the program is a Pro*C program.
  • After creating a program, we compile the program by using the PROC compiler, which is provided by the Oracle. The PROC compiler generates .c file with all the SQL statements replaced by the functions which are already pre-defined in the Oracle runtime library.
  • The file created by the PROC compiler will be compiled again by the C compiler, which is supported by Pro*C. In Windows, the PROC compiler supports Microsoft Visual C++ compiler.
  • The C compiler will create a .exe file.
  • Now, finally, we run the .exe file.

Why do we need the Pro*C compiler?

  • The Pro*C compiler allows you to embed the SQL statements in the C program, and it also provides the required user interface.
  • Unlike many other development tools, the Pro*C allows you to customize applications. It creates user interfaces that incorporate the latest windowing and mouse technology. Sometimes it is possible that we cannot generate the reports from other development tools, but we can achieve this by retrieving the data from the oracle database.

Requirements

The main requirement of Pro*C is to install the Pro*C software. When we are installing the Oracle database, then we have to make sure that we have selected the Pro*C components. If we have to check whether the installation of oracle contains the Pro*C components is by checking the PRECOMP directory of Oracle.


Directory Structure

When we install the Oracle, a directory structure is created on our hard drive for the Oracle products. The main oracle directory will contain the subdirectories and files which are required for the Pro*C.

When we install the Pro*C in Oracle, then Oracle Universal installer creates a directory known as precomp in the ORACLE_BASE\ORACLE_HOME directory. This subdirectory, i.e., precomp contains the executable files, library files, and some sample programs which is given below:

precomp Directory Structure

DirectoryDescription
\adminIt contains the configuration files.
\demo\procIt contains the sample programs for Pro*C.
\demo\sqlIt contains sql scripts for sample programs.
\doc\procIt contains the documentation files for pro*c.
\help\procIt contains help files for Pro*C.
\lib\msvcIt contains library files for Pro*C
\mesgIt contains message files.
\misc\procIt contains miscellaneous files for Pro*C.
\publicIt contains the header files having .h extension.

Restrictions

All the Windows operating systems can contain the spaces in the files and directory names, but Oracle Pro*C precompiler does not precompile the files, which include space in the file name.

For example, if the file name is the first program.pc, then this file name will be invalid.


Embedded SQL statements

Here, embedded SQL means placing the SQL statements inside the source program. As we house the statements inside the C program, so C program is also known as the host program, and the language we use is known as the host language. The Pro*C provides the ability to embed the SQL statements inside the program.

Embedded SQL statements are of two types:

  • Executable statements
  • Directives

Executable statements

Executable statements are the SQL statements that allow you to manipulate the data in the Oracle database. These statements call the Oracle runtime library. It also allows your program to connect to the Oracle database, to define the query, to manipulate the data, and process the transactions. These statements are written where C executable statements can be placed.

Directives

Directives or declarative statements are the SQL statements that neither call the Oracle runtime libraries nor operate on the Oracle data. It is used to declare the Oracle objects, SQL objects. These statements can be written where the C variables can be declared.


Pro*C Syntax

In the C program, all the SQL statements must start with EXEC SQL, and should end with the semicolon ;.We can write the SQL statement anywhere in the program but with one restriction that declarative statements should not come after the executable statements.

Suppose we want to retrieve the student marks from the database based on their id, then its program would be written as:

Preprocessor directives

The preprocessor directives that we can use when we are dealing with Pro*C in C are #include and #if. But, Pro*C does not know about the #define directive. Let's understand this through a simple scenario which is given below:

The above code is invalid, as Pro*C does not work with a #define directive.


Host Variables

Host variables are the variables of the host language that are used with the SQL embedded statements. Host variables are a key of communication between the Oracle and C program. These variables are declared similarly as we make the declaration in the C program, and it can be referenced by both our program and Oracle.

The host variables can be placed where the SQL expressions are used, and these variables are declared between the BEGIN DECLARE SECTION and END DECLARE SECTION. When we write the SQL statements, then the host variables are prefixed with a ':' colon.

The following is the list of the C data types that are supported by Oracle:

  • char
  • char[n]
  • int
  • short
  • long
  • float
  • double
  • VARCHAR[n]

Pointers

A pointer variable can also be used as a host variable in SQL statements. Let's understand through a simple example.

In the above code, we declare a pointer variable of integer type, i.e., *age. After declaration, we are executing a SELECT statement in which we are retrieving the value of age from the student table, and then we are storing the value of age in a host variable, i.e., age. The result will be stored in the *age, not in age.

Structures

C structures are also used in Pro*C. The member variables of the structure can be treated as the host variables in the host program. When we provide the name of the structure in the SQL statement, then each host variable must be prefixed with a: colon.

In the above code, we have created a structure named as a student, which contains two variables, i.e., student_id and name. After creating structure, we declare the variable, i.e., s1 of type student. Then, we insert the value of these two variables in a database by using the insert command.

Arrays

Arrays can be used as a host variable in SQL embedded statements. Let's understand this through a simple example:

In the above code, we have created a single-dimensional array of integer type. We implement the SQL INSERT command, which will insert all the 10 tuples in one go.

Let's look at the other example in which we use the two-dimensional array.

In Pro*C, arrays can only be single-dimensional. But, Pro*C precompiles the above code successfully as it considers the two-dimensional as a single-dimensional array of characters instead of a two-dimensional array of characters.

As we have shown in the above example that we can use the indicator variable in a SELECT statement to determine whether the output host variable contains the NULLs or truncated value. The following table shows the possible values of the indicator variable that can be given by the Oracle with its description:

Value returned by OracleDescription
-1It means that the value of column is NULL, then the value of the host variable will be indeterminate.
0An intact column value is assigned to the host variable.
>0Truncated value is assigned to the host variable.
-2Truncated value is assigned to the host variable.

If we want to create the indicator variable of a host variable in a struct, we can simply do that by creating an indicator variable of each host variable in a struct. To add the name of an indicator variable in a SQL statement, we need to write the name of the indicator variable of a structure that must be prefixed with a ':' colon, and must immediately follow the host variable.


Datatype Equivalencing

It is a very important feature as it adds flexibility to your application. It means that you can customize according to the requirements based on how Oracle interprets the input data and formats the output data.

Oracle contains two types of data types:

  • Internal datatypes
  • External datatypes

internal datatype: These data types are used by Oracle to define how data is going to be stored in a column.

External datatype: These data types are used by Oracle to format the output data, and then this output data will be stored in a host variable.

Let's see how we can equivalence the data type.

On a variable-by-variable basis, we use the var statement for equivalencing. The syntax of the var statement is given below:

For example, we want to retrieve the student names from the student table; then, we need to pass these student names to the function that accepts the C style strings (the last character must be a termination character '\0'). To equivalence the host variable with a String external datatype, we use the following code:

The column, i.e., student name, contains 11 characters. As the column student name contains the 11 characters, so we need to allot the 12 characters (11 characters of student name plus terminator character (\0)). We use the STRING data type that provides the interface with the C style strings. Oracle will automatically add the '\0' character.

Till now, we equivalence the built-in data types, i.e., in the above example, we equivalence the char array to the Oracle external datatype (String). We can also equivalence the user-defined data types by using the TYPE command. The syntax of the type statement is given below:


Dynamic vs Static SQL statements

Mainly, static SQL statements are used for fixed applications, but sometimes it is required for a program to create the SQL statements dynamically. To create the dynamic SQL statement, first, we need to store the SQL statement in a string variable. After storing the statement, we use the PREPARE statement to convert the character string into a SQL statement. Finally, we execute the statement by using the EXECUTE statement. Let's understand this scenario through an example.


Transactions

Oracle Pro*C also follows the concept of the transaction as defined by the SQL standard. A transaction is a series of statements that Oracle uses to either made all the changes permanent or undo all the changes done since the transaction began. We use two statements, i.e., EXEC SQL COMMIT and EXEC SQL ROLLBACK. The EXEC SQL COMMIT statement is used to make all the changes permanent since the transaction began. The EXEC SQL ROLLBACK is used to undo all the changes since the transaction began. If we start the next transaction without writing the statement EXEC SQL COMMIT, then all the changes made during this transaction will be discarded.


Error Handling

C programming provides the in-built error handling mechanism which we use in our source program. Error handling is a mechanism that provides the status of our source program. We need some mechanism that handles the error, so Pro*C contains the two error handling concepts which are given below:

  • SQLCA (SQL Communication Area)
  • Whenever statement

SQLCA

SQLCA (SQL Communication Area) is a data structure or an area used by our program to check for errors. This data structure contains some pre-defined variables used by Oracle. These variables contain the status information of the program, which is passed at the runtime.

The structure of sqlca is given below:

SQLCA Components

The following are the components of SQLCA:

  • sqlcaid: It is an array of char character which is initialized to "SQLCA", and used to determine the SQL Communication area.
  • sqlcabc: It is declared as an integer type to hold the length of the SQLCA structure in bytes.
  • sqlcode: It is declared as an integer type that stores the status code of the most recently executed SQL statement. The status determines the outcome of the SQL statement, and the outcome can come in the following ways:
OutcomeDescription
0The statement has been executed successfully with no error.
>0Some error occurs while executing the statement. For example, when we are using SELECT command with Where clause, then no such row found mentioned in the Where clause condition.
<0In this case, statement is not executed due to the database, system, application or network error.
  • sqlerrm: It is defined as a structure inside the sqlca.

This field contains two components:

sqlerrml: It is declared as an integer type that holds the length of the text message stored in sqlerrmc.

sqlerrmc: It is declared as a string that holds the text message with respect to the error code stored in a sqlcode.

  • sqlerrp: It is declared as a string but reserved for future use.
  • sqlerrd: It is declared as an array of integers that holds six elements.
FieldsDescription
sqlerrp[0]It is reserved for future use.
sqlerrp[1]It is reserved for future use.
sqlerrp[2]It contains the no. of rows which are processed by the SQL statement.
sqlerrp[3]It is reserved for the future use.
sqlerrp[4]It contains the character position at which the error begins in the most recently executed statement.
sqlerrp[5]It is reserved for the future use.
  • sqlwarn: It is declared as an array of characters having eight elements. These elements are used as a warning flag in sqlca. If the flag is set, then Oracle assigns 'W' to the element.
FieldsDescription
sqlwarn[0]It is set only when the value of another flag is set.
sqlwarn[1]It is set when the Oracle assigns a truncated value to the output host variable.
sqlwarn[2]It is set when the NULL column value is not considered while computing the SQL aggregate such MAX, MIN, AVG or SUM.
sqlwarn[3]It is set when the no of columns retrieved by using the SELECT statement is not equal to the no. of host variables specified in a INTO clause.
  • sqlext: It is declared as a string which is reserved for future use.

Whenever statement

The whenever statement is used for error handling. It performs implicit error checking and handling. The syntax of the whenever statement is given below:

When the Whenever statement is executed, then the Oracle will automatically check the SQLCA for the condition mentioned in the whenever statement. If such condition is found in the sqlca, then the action given in the whenever statement will be performed.

The condition can be NOT FOUND, SQLERROR, SQLWARNING, and the action can be CONTINUE, GOTO label, STOP, DO routine.

The condition can be of following types:

  • SQLWARNING: If the oracle returns a warning, then the sqlwarn[0] will be set.
  • SQLERROR: If the oracle returns an error, then the value of sqlcode will be negative.
  • NOT FOUND: If the Oracle is not able to find the row based on the condition mentioned in the WHERE clause of the SELECT statement, then the value of the sqlcode will be positive.

If the Oracle finds any of the above condition, then the following actions can be taken:

  • CONTINUE: The program will continue execution with its next statement if possible.
  • DO: In this action, the control will be transferred to the error handling function in the program. When the control is reached at the end of the error handling function, then the control will be transferred to the statement that comes after the failed SQL statement.
  • DO BREAK: This break statement is mainly used in our program, and can be used as an action in loops. When the condition present inside a WHENEVER statement is satisfied, then our program exits from the loop.
  • DO CONTINUE: This continue statement is mainly used in our program, and can be used as action in loops. When the condition inside a WHENEVER statement is satisfied, then our program continue with the next iteration inside the loop.
  • GOTO: The program control transfers to the labeled statement.
  • STOP: Our program stops running. When the WHENEVER condition is satisfied, then exit() function is called to stop the execution of the program.

Steps to create a program

The following are the steps required to create and run the program:

  • First, we need to install the Oracle database, and Microsoft Visual C++. In Visual C++, we will create our C program that contains the embedded SQL statements. This program will connect with a oracle database.
  • The oracle database contains the in-built PROC components. To check whether the PROC has been installed successfully, open the cmd and type the 'proc' command.
Pro*C Tutorial

After entering the proc command in command prompt, we press enter:

Pro*C Tutorial

The above screen shows that the proc has been installed successfully.

  • Now, we will create a C program in Visual C++. Open Visual C++. When we open the Visual C++, the image shown below will appear:
Pro*C Tutorial
  • Click on the File appearing at the top menu. Move the mouse over the New, and then click on the Project.
    Pro*C Tutorial
  • When we click on the project, the screen appears shown below:
Pro*C Tutorial

In the above screen, click on the General appearing at the left-side under the Visual C++ category, and then click on the Empty Project. Now, we need to provide the name to this empty project.

Pro*C Tutorial

In the above screen, we can observe that we have given the name of our project as Demo.

  • After providing the name to our project, we click on the OK button.
    Pro*C Tutorial

In the above screen, we can see that the workspace of our project has been created. At the leftmost side, we can see the hierarchy of our project which contains External Dependencies, Header Files, Resource files, and Source files.

  • Now, we have to add the library files of Oracle, so right-click on the header files, then keep the mouse cursor over the Add and then click on the Existing-item.
Pro*C Tutorial
  • Move to the oracle folder which contains precomp as a subdirectory. Click on the precomp.
Pro*C Tutorial

When we click on the precomp folder, the screen shown below will appear:

Pro*C Tutorial

Click on the lib folder, which is shown in the above screen.

Now add the two object library files, i.e., orasql19, and another one is orasqx19, which is present in the msvc folder.

Pro*C Tutorial
Pro*C Tutorial

When we add the above two library files, our Solution Explorer window would look like:

Pro*C Tutorial

In the above screen, the highlighted area shows that both the library files have been added successfully in our project.

  • Now, we add source code in our project. To do so, right-click on the source files, keep the mouse cursor over the Add and then click on the New item.
    Pro*C Tutorial
  • After clicking on the New item, the screen appears as shown below:
Pro*C Tutorial

In the above screen, click on the Code that appears at the leftmost side, and then click on the C++ File. We also have provided the file name as first with a .pc extension.

  • After providing the name to the source file, we click on the Add button.
Pro*C Tutorial

In the above screen, we can observe that the workspace for the file that we have added has been created.

  • The following is the code of pc file:
  • Open the cmd, and move to that folder, i.e., E:\Demo\Demo where first.pc file is stored.
Pro*C Tutorial
  • Now, we use the proc compiler to precompile the first.pc program as shown in the below screenshot:
Pro*C Tutorial
Pro*C Tutorial

In the above screen, we can observe that first.c has been created after compilation of first.pc by Oracle precompiler.

  • Now, we add the first.c file in Visual C++. To add this file, right-click on the Source files, then keep the mouse cursor on Add and then click on the Existing item. The below screen shows that first.c file has been added in Visual C++.
Pro*C Tutorial
  • After the compilation of first.pc, the code of first.c would be:

Run

We can run our program by simply clicking on the Debug that appears at the top menu, and then click on the Start Debugging.

Pro*C Tutorial