Microsoft Word - C-Programming
Microsoft Word - C-Programming
Microsoft Word - C-Programming
-Easy to learn
-Structured language
-It produces efficient programs.
-It can handle low-level activities.
-It can be compiled on a variety of computers.
Facts about C
-C was invented to write an operating system called UNIX.
-C is a successor of B language which was introduced around 1970
-The language was formalized in 1988 by the American National Standard
Institue (ANSI).
-By 1973 UNIX OS almost totally written in C.
-Today C is the most widely used System Programming Language.
-Most of the state of the art software have been implemented using C
Why to use C ?
C was initially used for system development work, in particular the programs
that
make-up the operating system. C was adoped as a system development
language because it produces code that runs nearly as fast as code written
in assembly language.
C Program File
All the C programs are writen into text files with extension ".c" for example
hello.c. You can use "vi" editor to write your C program into a file.
This tutorial assumes that you know how to edit a text file and how to write
programming insturctions inside a program file.
C Compilers
When you write any program in C language then to run that program you
need to
compile that program using a C Compiler which converts your program into
a
language understandable by a computer. This is called machine language
(ie.
binary format). So before proceeding, make sure you have C Compiler
available
at your computer. It comes alongwith all flavors of Unix and Linux.
If you are working over Unix or Linux then you can type gcc -v or cc -v and
check the result. You can ask your system administrator or you can take
help
from anyone to identify an available C Compiler at your computer.
If you don't have C compiler installed at your computer then you can use
below
given link to download a GNU C Compiler and use it.
To know more about compilation you can go through this small tutorial
C - Program Structure
A C program basically has the following form:
Ø Preprocessor Commands
Ø Functions
Ø Variables
Ø Statements & Expressions
Ø Comments
The following program is written in the C programming language. Open a text file
hello.c using vi editor and put the following lines inside that file.
#include <stdio.h>
int main()
{
/* My first program */
printf("Hello, World! \n");
return 0;
}
Preprocessor Commands: These commands tells the compiler to do
preprocessing before doing actual compilation. Like #include <stdio.h> is a
preprocessor command which tells a C compiler to include stdio.h file before
going to actual compilation. You will learn more about C Preprocessors in C
Preprocessors session.
Functions: are main building blocks of any C Program. Every C Program will
have one or more functions and there is one mandatory function which is called
main() function. This function is prefixed with keyword int which means this
function returns an integer value when it exits. This integer value is retured using
return statement.
The C Programming language provides a set of built-in functions. In the above
example printf() is a C built-in function which is used to print anything on the
screen. Check Builtin function section for more detail.
You will learn how to write your own functions and use them in Using Function
session.
Variables: are used to hold numbers, strings and complex data for
manipulation. You will learn in detail about variables in C Variable Types.
Statements & Expressions : Expressions combine variables and constants to
create new values. Statements are expressions, assignments, function calls, or
control flow statements which make up C programs.
Comments: are used to give additional useful information inside a C Program.
All the comments will be put inside /*...*/ as given in the example above. A
comment can span through multiple lines.
Note the followings
Ø C is a case sensitive programming language. It means in C printf and
Printf will have different meanings.
Ø C has a free-form line structure. End of each C statement must be
marked with a semicolon.
Ø Multiple statements can be one the same line.
Ø White Spaces (ie tab space and space bar ) are ignored.
Ø Statements can continue over multiple lines.
C Program Compilation
To compile a C program you would have to Compiler name and program files
name. Assuming your compiler's name is cc and program file name is hello.c,
give following command at Unix prompt.
$cc hello.c
This will produce a binary file called a.out and an object file hello.o in your
current directory. Here a.out is your first program which you will run at Unix
prompt like any other system program. If you don't like the name a.out then you
can produce a binary file with your own name by using -o option while compiling
C program. See an example below
$cc -o hello hello.c
Now you will get a binary with name hello. Execute this program at Unix prompt
but before executing / running this program make sure that it has execute
permission set. If you don't know what is execute permission then just follow
these two steps
$chmod 755 hello
$./hello
This will produce following result
Hello, World
Congratulations!! you have written your first program in "C". Now believe me its
not difficult to learn "C".
C - Reserved Keywords
The following names are reserved by the C language. Their meaning is already
defined, and they cannot be re-defined to mean anything else.
auto
else
long
switch
break
enum
register
typedef
case
extern
return
union
char
float
short
unsigned
const
for
signed
void
continue
goto
sizeof
volatile
default
if
static
while
do
int
struct
_Packed
Double
While naming your functions and variables, other than these names, you can
choose any names of reasonable length for variables, functions etc.
C - Basic Datatypes
C has a concept of 'data types' which are used to define a variable before its use.
The definition of a variable will assign storage for the variable and define the type of
data that will be held in the location.
The value of a variable can be changed any time.
C has the following basic built-in datatypes.
Ø int
Ø float
Ø double
Ø char
Please note that there is not a boolean data type. C does not have the traditional
view about logical comparison, but thats another story.
int - data type
int is used to define integer numbers.
{
int Count;
Count = 5;
}
float - data type
float is used to define floating point numbers.
{
float Miles;
Miles = 5.6;
}
double - data type
double is used to define BIG floating point numbers. It reserves twice the storage
for the number. On PCs this is likely to be 8 bytes.
{
double Atoms;
Atoms = 2500000;
}
char - data type
char defines characters.
{
char Letter;
Letter = 'x';
}
Modifiers
The data types explained above have the following modifiers.
Ø short
Ø long
Ø signed
Ø unsigned
The modifiers define the amount of storage allocated to the variable. The amount of
storage allocated is not cast in stone. ANSI has the following rules:
short int <= int <= long int
float <= double <= long double
What this means is that a 'short int' should assign less than or the same amount of
storage as an 'int' and the 'int' should be less or the same bytes than a 'long int'.
What this means in the real world is:
Type Bytes Range
---------------------------------------------------------------------
short int 2 -32,768 -> +32,767 (32kb)
unsigned short int 2 0 -> +65,535 (64Kb)
unsigned int 4 0 -> +4,294,967,295 ( 4Gb)
int 4 -2,147,483,648 -> +2,147,483,647 ( 2Gb)
long int 4 -2,147,483,648 -> +2,147,483,647 ( 2Gb)
signed char 1 -128 -> +127
unsigned char 1 0 -> +255
float 4
double 8
long double 12
These figures only apply to todays generation of PCs. Mainframes and midrange
machines could use different figures, but would still comply with the rule above.
You can find out how much storage is allocated to a data type by using the sizeof
operator discussed in Operator Types Session.
Here is an example to check size of memory taken by various datatypes.
int
main()
{
printf("sizeof(char) == %d\n", sizeof(char));
printf("sizeof(short) == %d\n", sizeof(short));
printf("sizeof(int) == %d\n", sizeof(int));
printf("sizeof(long) == %d\n", sizeof(long));
printf("sizeof(float) == %d\n", sizeof(float));
printf("sizeof(double) == %d\n", sizeof(double));
printf("sizeof(long double) == %d\n", sizeof(long double));
printf("sizeof(long long) == %d\n", sizeof(long long));
return 0;
}
Qualifiers
A type qualifier is used to refine the declaration of a variable, a function, and
parameters, by specifying whether:
The value of a variable can be changed.
The value of a variable must always be read from memory rather than from
a register
Standard C language recognizes the following two qualifiers:
const
volatile
The const qualifier is used to tell C that the variable value can not change after
initialisation.
const float pi=3.14159;
Now pi cannot be changed at a later time within the program.
Another way to define constants is with the #define preprocessor which has the
advantage that it does not use any storage
The volatile qualifier declares a data type that can have its value changed in ways
outside the control or detection of the compiler (such as a variable updated by the
system clock or by another program). This prevents the compiler from optimizing
code referring to the object by storing the object's value in a register and re-reading
it from there, rather than from memory, where it may have changed. You will use
this qualifier once you will become expert in "C". So for now just proceed.
int x[10];
The square brackets mean subscripting; parentheses are used only for function
references. Array indexes begin at zero, so the elements of x are:
Thus Array are special type of variables which can be used to store multiple values
of same data type. Those values are stored and accessed using subscript or index.
Arrays occupy consecutive memory slots in the computer's memory.
main( ) {
int n, c;
char line[100];
n = 0;
while( (c=getchar( )) != '\n' ) {
if( n < 100 )
line[n] = c;
n++;
}
printf("length = %d\n", n);
}
Array Initialization
Ø As with other declarations, array declarations can include an optional
initialization
Ø Scalar variables are initialized with a single value
Ø Arrays are initialized with a list of values
Ø The list is enclosed in curly braces
The number of initializers cannot be more than the number of elements in the array
but it can be less in which case, the remaining elements are initialized to 0.if you
like, the array size can be inferred from the number of initializers by leaving the
square brackets empty so these are identical declarations:
C - Variable Types
A variable is just a named area of storage that can hold a single value (numeric
or character). The C language demands that you declare the name of each
variable that you are going to use and its type, or class, before you actually try
to do anything with it.
Local Variables
Ø Local variables scope is confined within the block or function where it is
defined. Local variables must always be defined at the top of a block.
Ø When a local variable is defined - it is not initalised by the system, you
must initalise it yourself.
Ø When execution of the block starts the variable is available, and when the block ends
the variable 'dies'.
main()
{
int i=4;
int j=10;
i++;
if (j > 0)
{
/* i defined in 'main' can be seen */
printf("i is %d\n",i);
}
if (j > 0)
{
/* 'i' is defined and so local to this block */
int i=100;
printf("i is %d\n",i);
}/* 'i' (value 100) dies here */
printf("i is %d\n",i); /* 'i' (value 5) is now visable.*/
}
This will generate following output
i is 5
i is 100
i is 5
Here ++ is called incremental operator and it increase the value of any integer
variable by 1. Thus i++ is equivalent to i = i + 1;
You will see -- operator also which is called decremental operator and it
idecrease the value of any integer variable by 1. Thus i-- is equivalent to i = i -
1;
Global Variables
Global variable is defined at the top of the program file and it can be visible and
modified by any function that may reference it.
Global variables are initalised automatically by the system when you define
them!
int 0
char '\0'
float 0
pointer NULL
If same variable name is being used for global and local variable then local
variable takes preference in its scope. But it is not a good practice to use global
variables and local variables with the same name.
C - Storage Classes
A storage class defines the scope (visibility) and life time of variables and/or
functions within a C Program.
There are following storage classes which can be used in a C Program
Ø auto
Ø register
Ø static
Ø extern
{
int Count;
auto int Month;
}
The example above defines two variables with the same storage class. auto can
only be used within functions, i.e. local variables.
register - Storage Class
register is used to define local variables that should be stored in a register
instead of RAM. This means that the variable has a maximum size equal to the
register size (usually one word) and cant have the unary '&' operator applied to it
(as it does not have a memory location).
{
register int Miles;
}
Register should only be used for variables that require quick access - such as
counters. It should also be noted that defining 'register' goes not mean that the
variable will be stored in a register. It means that it MIGHT be stored in a register
- depending on hardware and implimentation restrictions.
static variables can be 'seen' within all functions in this source file. At link time,
the static variables defined here will not be seen by the object modules that are
brought in.
static can also be defined within a function. If this is done the variable is initalised
at run time but is not reinitalized when the function is called. This inside a
function static variable retains its value during vairous calls.
void func(void);
static count=10; /* Global variable - static is the default */
main()
{
while (count--)
{
func();
}
}
void func( void )
{
static i = 5;
i++;
printf("i is %d and count is %d\n", i, count);
}
This will produce following result
i is 6 and count is 9
i is 7 and count is 8
i is 8 and count is 7
i is 9 and count is 6
i is 10 and count is 5
i is 11 and count is 4
i is 12 and count is 3
i is 13 and count is 2
i is 14 and count is 1
i is 15 and count is 0
NOTE : Here keyword void means function does not return anything and it does
not take any parameter. You can memoriese void as nothing. static variables are
initialized to 0 automatically.
Definition vs Declaration : Before proceeding, let us understand the difference
between defintion and declaration of a variable or function. Definition means
where a variable or function is defined in realityand actual memory is allocated
for variable or function. Declaration means just giving a reference of a variable
and function. Through declaration we assure to the complier that this variable or
function has been defined somewhere else in the program and will be provided at
the time of linking. In the above examples char *func(void) has been put at the
top which is a declaration of this function where as this function has been defined
below to main() function.
There is one more very important use for 'static'. Consider this bit of code.
char *func(void);
main()
{
char *Text1;
Text1 = func();
}
char *func(void)
{
char Text2[10]="martin";
return(Text2);
}
Now, 'func' returns a pointer to the memory location where 'text2' starts BUT
text2 has a storage class of 'auto' and will disappear when we exit the function
and could be overwritten but something else. The answer is to specify
The storage assigned to 'text2' will remain reserved for the duration if the
program.
File 1: main.c
int count=5;
main()
{
write_extern();
}
File 2: write.c
void write_extern(void);
void write_extern(void)
{
printf("count is %i\n", count);
}
This fill produce write program which can be executed to produce result.
Count in 'main.c' will have a value of 5. If main.c changes the value of count -
write.c will see the new value
C - Using Constants
A C constant is usually just the written version of a number. For example 1, 0,
5.73, 12.5e9. We can specify our constants in octal or hexadecimal, or force
them to be treated as long integers.
Character constants are usually just the character enclosed in single quotes; 'a',
'b', 'c'. Some characters can't be represented in this way, so we use a 2
character sequence as follows.
'\n' newline
'\t' tab
'\\' backslash
'\'' single quote
'\0' null ( Usedautomatically to terminate character string )
In addition, a required bit pattern can be specified using its octal equivalent.
Character constants are rarely used, since string constants are more convenient.
A string constant is surrounded by double quotes eg "Brian and Dennis". The
string is actually stored as an array of characters. The null character '\0' is
automatically placed at the end of such a string to act as a string terminator.
A character is a different type to a single character string. This is important poing
to note.
Defining Constants
ANSI C allows you to declare constants. When you declare a constant it is a bit
like a variable declaration except the value cannot be changed.
The const keyword is to declare a constant, as shown below:
int const a = 1;
const int a =2;
Note:
Ø You can declare the const before or after the type. Choose one an stick to
it.
Ø It is usual to initialise a const with a value as it cannot get a value any
other way.
The preprocessor #define is another more flexible (see Preprocessor Chapters)
method to define constants in a program.
#define TRUE 1
#define FALSE 0
#define NAME_SIZE 20
I've made the constant names uppercase, but you can name them which ever
way you want.
Here, colors is the name given to the set of constants - the name is optional.
Now, if you don't assign a value to a constant, the default value for the first one
in the list - RED in our case, has the value of 0. The rest of the undefined
constants have a value 1 more than the one before, so in our case, YELLOW is 1,
GREEN is 2 and BLUE is 3.
But you can assign values if you wanted to:
enum colors {RED=1, YELLOW, GREEN=6, BLUE };
#include <stdio.h>
int main() {
enum {RED=5, YELLOW, GREEN=4, BLUE};
printf("RED = %d\n", RED);
printf("YELLOW = %d\n", YELLOW);
printf("GREEN = %d\n", GREEN);
printf("BLUE = %d\n", BLUE);
return 0;
}
This will produce following results
RED = 5
YELLOW = 6
GREEN = 4
BLUE = 5
C - Operator Types
What is Operator? Simple answer can be given using expression 4 + 5 is equal
to 9. Here 4 and 5 are called operands and + is called operator. C language
supports following type of operators.
Ø Arithmetic Operators
Ø Logical (or Relational) Operators
Ø Bitwise Operators
Ø Assignment Operators
Ø Misc Operators
Bitwise Operators:
Bitwise operator works on bits and perform bit by bit operation.
Assume if A = 60; and B = 13; Now in binary format they will be as follows:
A = 0011 1100
B = 0000 1101
-----------------
A&B = 0000 1100
A|B = 0011 1101
A^B = 0011 0001
~A = 1100 0011
Show Examples
There are following Bitwise operators supported by C language
Operator Description Example
& Binary AND Operator copies a bit
to the result if it exists in both
operands. (A & B) will give 12 which is
0000 1100
| Binary OR Operator copies a bit if
it exists in eather operand. (A | B) will give 61 which is
0011 1101
^ Binary XOR Operator copies the
bit if it is set in one operand but
not both. (A ^ B) will give 49 which is
0011 0001
~ Binary Ones Complement
Operator is unary and has the
efect of 'flipping' bits. (~A ) will give -60 which is
1100 0011
<< Binary Left Shift Operator. The
left operands value is moved left
by the number of bits specified by
the right operand. A << 2 will give 240 which is
1111 0000
>> Binary Right Shift Operator. The
left operands value is moved right
by the number of bits specified by
the right operand. A >> 2 will give 15 which is
0000 1111
Assignment Operators:
There are following assignment operators supported by C language:
Show Examples
Precedence of C Operators:
Operator precedence determines the grouping of terms in an expression. This
affects how an expression is evaluated. Certain operators have higher
precedence than others; for example, the multiplication operator has higher
precedence than the addition operator:
For example x = 7 + 3 * 2; Here x is assigned 13, not 20 because operator * has
higher precedenace than + so it first get multiplied with 3*2 and then adds into 7.
Here operators with the highest precedence appear at the top of the table, those
with the lowest appear at the bottom. Within an expression, higher precedenace
operators will be evaluated first.
Branching is deciding what actions to take and looping is deciding how many
times to take a certain action.
Branching:
Branching is so called because the program chooses to follow one branch or
another.
if statement
This is the most simple form of the branching statements.
It takes an expression in parenthesis and an statement or block of statements. if
the expression is true then the statement or block of statements gets executed
otherwise these statements are skipped.
NOTE: Expression will be assumed to be true if its evaulated values is non-zero.
if statements take the following form:
Show Example
if (expression)
statement;
or
if (expression)
{
Block of statements;
}
or
if (expression)
{
Block of statements;
}
else
{
Block of statements;
}
or
if (expression)
{
Block of statements;
}
else if(expression)
{
Block of statements;
}
else
{
Block of statements;
}
? : Operator
The ? : operator is just like an if ... else statement except that because it is an
operator you can use it within expressions.
? : is a ternary operator in that it takes three values, this is the only ternary
operator C has.
Show Example
switch statement:
The switch statement is much like a nested if .. else statement. Its mostly a
matter of preference which you use, switch statement can be slightly more
efficient and easier to read.
Show Example
switch( expression )
{
case constant-expression1: statements1;
[case constant-expression2: statements2;]
[case constant-expression3: statements3;]
[default : statements4;]
}
Show Example
Looping
Loops provide a way to repeat commands and control how many times they are
repeated. C provides a number of looping way.
while loop
The most basic loop in C is the while loop.A while statement is like a repeating if
statement. Like an If statement, if the test condition is true: the statments get
executed. The difference is that after the statements have been executed, the
test condition is checked again. If it is still true the statements get executed
again.This cycle repeats until the test condition evaluates to false.
Basic syntax of while loop is as follows:
Show Example
while ( expression )
{
Single statement
or
Block of statements;
}
for loop
for loop is similar to while, it's just written differently. for statements are often
used to proccess lists such a range of numbers:
Basic syntax of for loop is as follows:
Show Example
do...while loop
do ... while is just like a while loop except that the test condition is checked
at
the end of the loop rather than the start. This has the effect that the content
of
the loop are always executed at least once.
Basic syntax of do...while loop is as follows:
Show Example
do
{
Single statement
or
Block of statements;
}while(expression);
#include
main()
{
int i;
int j = 10;
for( i = 0; i <= j; i ++ )
{
if( i == 5 )
{
continue;
}
printf("Hello %d\n", i );
}
}
Input : In any programming language input means to feed some data into
program. This can be given in the form of file or from command line. C
programming language provides a set of built-in functions to read given input and
feed it to the program as per requirement.
Here we will discuss only one input function and one putput function just to
understand the meaning of input and output. Rest of the functions are given into
C - Built-in Functions
printf() function
This is one of the most frequently used functions in C for output. ( we will discuss
what is function in subsequent chapter. ).
Try following program to understand printf() function.
#include <stdio.h>
main()
{
int dec = 5;
char str[] = "abc";
char ch = 's';
float pi = 3.14;
printf("%d %s %f %c\n", dec, str, pi, ch);
}
5 abc 3.140000 c
#include <stdio.h>
main()
{
int x;
int args;
printf("Enter an integer: ");
if (( args = scanf("%d", &x)) == 0) {
printf("Error: not an integer\n");
} else {
printf("Read in %d\n", x);
}
}
Here %d is being used to read an integer value and we are passing &x to
store
the vale read input. Here &indicates the address of variavle x.
This program will prompt you to enter a value. Whatever value you will enter
at
command prompt that will be output at the screen using printf() function. If
you
eneter a non-integer value then it will display an error message.
Enter an integer: 20
Read in 20
C - Pointing to Data
A pointer is a special kind of variable. Pointers are designed for storing memory
address i.e. the address of another variable. Declaring a pointer is the same as
declaring a normal variable except you stick an asterisk '*' in front of the variables
identifier.
Ø There are two new operators you will need to know to work with pointers. The
"address of" operator '&' and the "dereferencing" operator '*'. Both are prefix
unary operators.
Ø When you place an ampersand in front of a variable you will get it's address,
this can be stored in a pointer vairable.
Ø When you place an asterisk in front of a pointer you will get the value at the
memory address pointed to.
#include <stdio.h>
int main()
{
int my_variable = 6, other_variable = 10;
int *my_pointer;
printf("the address of my_variable is : %p\n", &my_variable);
printf("the address of other_variable is : %p\n",
&other_variable);
my_pointer = &my_variable;
printf("\nafter \"my_pointer = &my_variable\":\n");
printf("\tthe value of my_pointer is %p\n", my_pointer);
printf("\tthe value at that address is %d\n", *my_pointer);
my_pointer = &other_variable;
printf("\nafter \"my_pointer = &other_variable\":\n");
printf("\tthe value of my_pointer is %p\n", my_pointer);
printf("\tthe value at that address is %d\n", *my_pointer);
return 0;
}
char *y;
char x[100];
y = &x[0];
y = x;
Since x is the address of x[0] this is legal and consistent. Now `*y' gives x[0]. More
importantly notice the following:
Pointer Arithmetic:
C is one of the few languages that allows pointer arithmetic. In other words,
you
actually move the pointer reference by an arithmetic operation. For
example:
On a typical 32-bit machine, *ip would be pointing to 5 after initialization. But ip++;
increments the pointer 32-bits or 4-bytes. So whatever was in the next 4-bytes, *ip
would be pointing at it.
Pointer arithmetic is very useful when dealing with arrays, because arrays and
pointers share a special relationship in C.
#include <stdio.h>
int main() {
int *ptr;
int arrayInts[10] = {1,2,3,4,5,6,7,8,9,10};
ptr = arrayInts; /* ptr = &arrayInts[0]; is also fine */
printf("The pointer is pointing to the first ");
printf("array element, which is %d.\n", *ptr);
printf("Let's increment it.....\n");
ptr++;
printf("Now it should point to the next element,");
printf(" which is %d.\n", *ptr);
printf("But suppose we point to the 3rd and 4th: %d %d.\n",
*(ptr+1),*(ptr+2));
ptr+=2;
printf("Now skip the next 4 to point to the 8th: %d.\n",
*(ptr+=4));
ptr--;
printf("Did I miss out my lucky number %d?!\n", *(ptr++));
printf("Back to the 8th it is then..... %d.\n", *ptr);
return 0;
}
As you can see, you must be careful when specifying the const qualifier when using
pointers.
#include <stdio.h>
int main() {
char *ptr;
char arrayChars[8] = {'F','r','i','e','n','d','s','\0'};
ptr = arrayChars;
printf("The array reads %s.\n", arrayChars);
printf("Let's change it..... ");
*ptr = 'f'; /* ptr points to the first element */
printf(" now it reads %s.\n", arrayChars);
printf("The 3rd character of the array is %c.\n",
*(ptr+=2));
printf("Let's change it again..... ");
*(ptr - 1) = ' ';
printf("Now it reads %s.\n", arrayChars);
return 0;
}
type casting in C language.Type is used to caste a variable from one data type to
another datatype to make it compatible to the lvalue.
NOTE-2 : lvalue is something which is used to left side of a statement and in which
we can assign some value. A constant can't be an lvalue because we can not assign
any value in contact. For example x = y, here x is lvalue and y is rvalue.
However, above example will produce following result:
C - Using Functions
A function is a module or block of program code which deals with a particular
task.
Making functions is a way of isolating one block of code from other
independent blocks
of code.
Functions serve two purposes.
They allow a programmer to say: `this piece of code does a specific job
which
stands by itself and should not be mixed up with anyting else',
Second they make a block of code reusable since a function can be reused in
many different contexts without repeating parts of the program text.
A function can take a number of parameters, do required processing and
then return a
value. There may be a function which does not return any value.
You already have seen couple of built-in functions like printf(); Similar way
you can
define your own functions in C language.
Consider the following chunk of code
To turn it into a function you simply wrap the code in a pair of curly brackets to convert
it into a single compound statement and write the name that you want to give it in front
of the brackets:
Demo()
{
int total = 10;
printf("Hello World");
total = total + l;
}
curved brackets after the function's name are required. You can pass one or more
paramenters to a function as follows:
By default function does not return anything. But you can make a function to return any
value as follows:
A return keyword is used to return a value and datatype of the returned value is
specified before the name of function. In this case function returns total which is int
type. If a function does not return a value then void keyword can be used as return
value.
Once you have defined your function you can use it within a program:
main()
{
Demo();
}
Pass by Value: mechanism is used when you don't want to change the
value of
passed paramters. When parameters are passed by value then functions in C
create copies of the passed in variables and do required processing on these
copied variables.
Here are two programs to understand the difference: First example is for
Pass by value:
#include <stdio.h>
/* function declaration goes here.*/
void swap( int p1, int p2 );
int main()
{
int a = 10;
int b = 20;
printf("Before: Value of a = %d and value of b = %d\n", a, b );
swap( a, b );
printf("After: Value of a = %d and value of b = %d\n", a, b );
}
void swap( int p1, int p2 )
{
int t;
t = p2;
p2 = p1;
p1 = t;
printf("Value of a (p1) = %d and value of b(p2) = %d\n", p1, p2
);
}
Here is the result produced by the above example. Here the values of a and
b remain
unchanged before calling swap function and after calling swap function.
#include <stdio.h>
/* function declaration goes here.*/
void swap( int *p1, int *p2 );
int main()
{
int a = 10;
int b = 20;
printf("Before: Value of a = %d and value of b = %d\n", a, b );
swap( &a, &b );
printf("After: Value of a = %d and value of b = %d\n", a, b );
}
Here is the result produced by the above example. Here the values of a and b are
changes after calling swap function.
#include <stdio.h>
int main() {
char array1[50];
char *array2;
printf("Now enter another string less than 50");
printf(" characters with spaces: \n");
gets(array1);
printf("\nYou entered: ");
puts(array1);
printf("\nTry entering a string less than 50");
printf(" characters, with spaces: \n");
scanf("%s", array2);
printf("\nYou entered: %s\n", array2);
return 0;
}
Ø char *strtok(char *s, const char *delim) - Parse the string s into
tokens
using delim as delimiter.
C - Structured Datatypes
Ø A structure in C is a collection of items of different types. You can think of
a structure as a "record" is in Pascal or a class in Java without methods.
Ø Structures, or structs, are very useful in creating data structures larger
and more complex than the ones we have discussed so far.
Ø Simply you can group various built-in data types into a structure.
Ø Object conepts was derived from Structure concept. You can achieve few
object oriented goals using C structure but it is very complex.
struct student {
char firstName[20];
char lastName[20];
char SSN[9];
float gpa;
};
Now you have a new datatype called student and you can use this datatype
define your variables of student type:
struct {
char firstName[20];
char lastName[20];
char SSN[10];
float gpa;
} student_a, student_b;
All the variables inside an structure will be accessed using these values as
student_a.firstName will give value of firstName variable. Similarly we can
aqccess other variables.
Structure Example:
Try out following example to understand the concept:
#include <stdio.h>
struct student {
char firstName[20];
char lastName[20];
char SSN[10];
float gpa;
};
main()
{
struct student student_a;
strcpy(student_a.firstName, "Deo");
strcpy(student_a.lastName, "Dum");
strcpy(student_a.SSN, "2333234" );
student_a.gpa = 2009.20;
printf( "First Name: %s\n", student_a.firstName );
printf( "Last Name: %s\n", student_a.lastName );
printf( "SNN : %s\n", student_a.SSN );
printf( "GPA : %f\n", student_a.gpa );
}
printf("%s\n", student_a->SSN);
typedef Keyword
There is an easier way to define structs or you could "alias" types you create. For
Example:
typedef struct{
char firstName[20];
char lastName[20];
char SSN[10];
float gpa;
}student;
Now you can use student directly to define variables of student type without using
struct keyword. Following is the example:
student student_a;
Unions Datatype
Unions are declared in the same fashion as structs, but have a fundamental
difference. Only one item within the union can be used at any time, because
the
memory allocated for each item inside the union is in a shared memory
location.
union Shape {
int circle;
int triangle;
int ovel;
};
We use union in such case where only one condition will be applied and only one
variable will be used.
Conclusion:
Ø You can create arrays of structs.
Ø Structs can be copied or assigned.
Ø The & operator may be used with structs to show addresses.
Ø Structs can be passed into functions. Structs can also be returned from
functions.
Ø Structs cannot be compared!
Ø Structures can store non-homogenous data types into a single collection,
much like an array does for common data (except it isn't accessed in the
same manner).
Ø Pointers to structs have a special infix operator: -> for dereferencing the
pointer.
Ø typedef can help you clear your code up and can help save some
keystrokes.
C - Working with Files
When accessing files through C, the first necessity is to have a way to access the
files. For C File I/O you need to use a FILE pointer, which will let the program
keep track of the file being accessed. For Example:
FILE *fp;
To open a file you need to use the fopen function, which returns a FILE pointer.
Once you've opened a file, you can use the FILE pointer to let the compiler
perform input and output functions on the file.
Here filename is string literal which you will use to name your file and mode can
have one of the following values
Note that it's possible for fopen to fail even if your program is perfectly correct:
you might try to open a file specified by the user, and that file might not exist (or
it might be write-protected). In those cases, fopen will return 0, the NULL pointer.
Here's a simple example of using fopen:
FILE *fp;
fp=fopen("/home/tutorialspoint/test.txt", "r");
This code will open test.txt for reading in text mode. To open a file in a binary
mode you must add a b to the end of the mode string; for example, "rb" (for the
reading and writing modes, you can add the b either after the plus sign - "r+b" -
or before - "rb+")
To close a function you can use the function:
fclose(fp);
To work with text input and output, you use fprintf and fscanf, both of which are
similar to their friends printf and scanf except that you must pass the FILE pointer
as first argument.
#include <stdio.h>
main()
{
FILE *fp;
fp = fopen("/tmp/test.txt", "w");
fprintf(fp, "This is testing...\n");
fclose(fp;);
}
Thsi will create a file test.txt in /tmp directory and will write This is testing in that file.
#include <stdio.h>
main()
{
FILE *fp;
char buffer[20];
fp = fopen("/tmp/test.txt", "r");
fscanf(fp, "%s", buffer);
printf("Read Buffer: %s\n", %buffer );
fclose(fp;);
}
The fgetc returns an int. What this actually means is that when it reads a normal
character in the file, it will return a value suitable for storing in an unsigned char
(basically, a number in the range 0 to 255). On the other hand, when you're at
the very end of the file, you can't get a character value--in this case, fgetc will
return "EOF", which is a constnat that indicates that you've reached the end of
the file.
The fputc function allows you to write a character at a time--you might find this
useful if you wanted to copy a file character by character. It looks like this:
Note that the first argument should be in the range of an unsigned char so that it
is a valid character. The second argument is the file to write to. On success, fputc
will return the value c, and on failure, it will return EOF.
Binary I/O
There are following two functions which will be used for binary input and output:
C - Bits Manipulation
The shift operators perform appropriate shift by operator on the right to the
operator on the left. The right operator must be positive. The vacated bits are
filled with zero.
For example: x << 2 shifts the bits in x by 2 places to the left.
Bit Fields
Bit Fields allow the packing of data in a structure. This is especially useful when
memory or data storage is at a premium. Typical examples:
Packing several objects into a machine word. e.g. 1 bit flags can be
compacted.
Reading external file formats -- non-standard file formats could be read
in. E.g. 9 bit integers.
C allows us do this in a structure definition by putting :bit length after the
variable.For example:
struct packed_struct {
unsigned int f1:1;
unsigned int f2:1;
unsigned int f3:1;
unsigned int f4:1;
unsigned int type:4;
unsigned int my_int:9;
} pack;
Here the packed_struct contains 6 members: Four 1 bit flags f1..f3, a 4 bit type
and a 9 bit my_int.
C automatically packs the above bit fields as compactly as possible, provided
that the maximum length of the field is less than or equal to the integer word
length of the computer. If this is not the case then some compilers may allow
memory overlap for the fields whilst other would store the next field in the next
word.
C - Pre-Processors
The C Preprocessor is not part of the compiler, but is a separate step in the
compilation process. In simplistic terms, a C Preprocessor is just a text
substitution tool. We'll refer to the C Preprocessor as the CPP.
#define MAX_ARRAY_LENGTH 20
Tells the CPP to replace instances of MAX_ARRAY_LENGTH with 20. Use #define
for constants to increase readability.
#include <stdio.h>
#include "myheader.h"
Tells the CPP to get stdio.h from System Libraries and add the text to this file.
The next line tells CPP to get myheader.h from the local directory and add the
text to the file
#undef FILE_SIZE
#define FILE_SIZE 42
#ifndef MESSAGE
#define MESSAGE "You wish!"
#endif
Tells the CPP to define MESSAGE only if MESSAGE isn't defined already.
#ifdef DEBUG
/* Your debugging statements here */
#endif
Tells the CPP to do the following statements if DEBUG is defined. This is useful if
you pass the -DDEBUG flag to gcc. This will define DEBUG, so you can turn
debugging on and off on the fly!
Stringize (#):
The stringize or number-sign operator ('#'), when used within a macro definition,
converts a macro parameter into a string constant. This operator may be used
only in a macro that has a specified argument or parameter list.
When the stringize operator immediately precedes the name of one of the macro
parameters, the parameter passed to the macro is enclosed within quotation
marks and is treated as a string literal. For example:
#include <stdio.h>
#define message_for(a, b) \
printf(#a " and " #b ": We love you!\n")
int main(void)
{
message_for(Carole, Debra);
return 0;
}
This example results in the following actual output from the preprocessor:
This example shows the concatenation of token##n into token34. Both the
stringize and the token-pasting operators are used in this example.
Parameterized Macros:
One of the powerful functions of the CPP is the ability to simulate functions using
parameterized macros. For example, we might have some code to square a
number:
int square(int x) {
return x * x;
}
Macros with arguments must be defined using the #define directive before they
can be used. The argument list is enclosed in parentheses and must immediately
follow the macro name. Spaces are not allowed between and macro name and
open parenthesis. For example:
Macro Caveats:
Ø Macro definitions are not stored in the object file. They are only active for
the duration of a single source file starting when they are defined and
ending when they are undefined (using #undef), redefined, or when the
end of the source file is found.
Ø Macro definitions you wish to use in multiple source files may be defined in
an include file which may be included in each source file where the macros
are required.
Ø When a macro with arguments is invoked, the macro processor substitutes
the arguments into the macro body and then processes the results again
for additional macro calls. This makes it possible, but confusing, to piece
together a macro call from the macro body and from the macro
arguments.
Ø Most experienced C programmers enclose macro arguments in
parentheses when they are used in the macro body. This technique
prevents undesired grouping of compound expressions used as arguments
and helps avoid operator precedence rules overriding the intended
meaning of a macro.
Ø While a macro may contain references to other macros, references to
itself are not expanded. Self-referencing macros are a special feature of
ANSI Standard C in that the self-reference is not interpreted as a macro
call. This special rule also applies to indirectly self-referencing macros (or
macros that reference themselves through another macro).
C - Useful Concepts
Error Reporting:
Many times it is useful to report errors in a C program. The standard library
perror() is an easy to use and convenient function. It is used in conjunction with
errno and frequently on encountering an error you may wish to terminate your
program early. We will meet these concepts in other parts of the function
reference chapter also.
Ø void perror(const char *message) - produces a message on standard
error output describing the last error encountered.
errno: - is a special system variable that is set if a system call cannot perform
its set task. It is defined in #include <errno.h>.
Predefined Streams:
UNIX defines 3 predefined streams ie. virtual files
They all use text a the method of I/O. stdin and stdout can be used with files,
programs, I/O devices such as keyboard, console, etc.. stderr always goes to the
console or screen.
The console is the default for stdout and stderr. The keyboard is the default for
stdin.
Dynamic Memory Allocation:
Dynamic allocation is a pretty unique feature to C. It enables us to create data
types and structures of any size and length to suit our programs need within the
program. We use dynamic memory allocation concept when we don't know how
in advance about memory requirement.
#include <stdio.>h
main( int argc, char *argv[] )
{
if( argc == 2 )
printf("The argument supplied is %s\n", argv[1]);
else if( argc > 2 )
printf("Too many arguments supplied.\n");
else
printf("One argument expected.\n");
}
Note that *argv[0] is the name of the program invoked, which means that
*argv[1] is a pointer to the first argument supplied, and *argv[n] is the last
argument. If no arguments are supplied, argc will be one. Thus for n arguments,
argc will be equal to n + 1. The program is called by the command line:
$myprog argument1
When this command is executed, the command interpreter calls the main()
function of the myprog program with 4 passed as the argc argument and an
array of 4 strings as the argv argument.
argv[0] - "myprog"
argv[1] - "aaa"
argv[2] - "bbb"
argv[3] - "ccc"
Multidimensional Arrays:
The array we used in the last example was a one dimensional array. Arrays can
have more than one dimension, these arrays-of-arrays are called
multidimensional arrays. They are very similar to standard arrays with the
exception that they have multiple sets of square brackets after the array
identifier. A two dimensional array can be though of as a grid of rows and
columns.
#include <stdio.h>
const int num_rows = 7;
const int num_columns = 5;
int
main()
{
int box[num_rows][num_columns];
int row, column;
for(row = 0; row < num_rows; row++)
for(column = 0; column < num_columns; column++)
box[row][column] = column + (row * num_columns);
for(row = 0; row < num_rows; row++)
{
for(column = 0; column < num_columns; column++)
{
printf("%4d", box[row][column]);
}
printf("\n");
}
return 0;
}
0 1 2 3 4
5 6 7 8 9
10 11 12 13 14
15 16 17 18 19
20 21 22 23 24
25 26 27 28 29
30 31 32 33 34
The above array has two dimensions and can be called a doubly subscripted
array. GCC allows arrays of up to 29 dimensions although actually using an array
of more than three dimensions is very rare
EXAMPLES
Try following example to understand if statement. You can put the following code into a test.c
file and then compile it and then run it.
#include <stdio.h>
main()
{
int cows = 6;
if (cows > 1)
printf("We have cows\n");
if (cows == 5 )
{
printf("We have 5 cows\n");
}
else if( (cows == 6 )
{
printf("We have 6 cows\n");
}
}
Try following example to understand ternary operator. You can put the following code into a
test.c file and then compile it and then run it .
#include <stdio.h>
main()
{
int a , b;
a = 10;
printf( "Value of b is %d\n", (a == 1) ? 20: 30 );
Try following example to understand switch statement. You can put the following code into a
test.c file and then compile it and then run it .
#include <stdio.h>
main()
{
int Grade = 'A';
switch( Grade )
{
case 'A' : printf( "Excellent\n" );
case 'B' : printf( "Good\n" );
case 'C' : printf( "OK\n" );
case 'D' : printf( "Mmmmm....\n" );
case 'F' : printf( "You must do better than this\n" );
default : printf( "What is your grade anyway?\n" );
}
}
You can come out of the switch block if your condition is met. This can be achieved using break
statement. Try out following example:
#include <stdio.h>
main()
{
int Grade = 'B';
switch( Grade )
{
case 'A' : printf( "Excellent\n" );
break;
case 'B' : printf( "Good\n" );
break;
case 'C' : printf( "OK\n" );
break;
case 'D' : printf( "Mmmmm....\n" );
break;
case 'F' : printf( "You must do better than this\n" );
break;
default : printf( "What is your grade anyway?\n" );
break;
}
}
If none of the conditions is met then default condition is executed. Try out following example to
understand default condition.
#include <stdio.h>
main()
{
int Grade = 'L';
switch( Grade )
{
case 'A' : printf( "Excellent\n" );
break;
case 'B' : printf( "Good\n" );
break;
case 'C' : printf( "OK\n" );
break;
case 'D' : printf( "Mmmmm....\n" );
break;
case 'F' : printf( "You must do better than this\n" );
break;
default : printf( "What is your grade anyway?\n" );
break;
}
}
Try following example to understand while loop. You can put the following code into a test.c file
and then compile it and then run it.
#include <stdio.h>
main()
{
int i = 10;
while ( i > 0 )
{
printf("Hello %d\n", i );
i = i -1;
}
}
You can make use of break to come out of while loop at any time.
#include <stdio.h>
main()
{
int i = 10;
while ( i > 0 )
{
printf("Hello %d\n", i );
i = i -1;
if( i == 6 )
{
break;
}
}
}
Try following example to understand for loop. You can put the following code into a test.c file
and then compile it and then run it.
#include <stdio.h>
main()
{
int i;
int j = 10;
for( i = 0; i <= j; i ++ )
{
printf("Hello %d\n", i );
}
}
You can make use of break to come out of for loop at any time.
#include <stdio.h>
main()
{
int i;
for( i = 0; i <= j; i ++ )
{
printf("Hello %d\n", i );
if( i == 6 )
{
break;
}
}
}
Try following example to understand do...while loop. You can put the following code into a
test.c file and then compile it and then run it.
#include <stdio.h>
main()
{
int i = 10;
do{
printf("Hello %d\n", i );
i = i -1;
}while ( i > 0 );
}
You can make use of break to come out of do...while loop at any time.
#include <stdio.h>
main()
{
int i = 10;
do{
printf("Hello %d\n", i );
i = i -1;
if( i == 6 )
{
break;
}
}while ( i > 0 );
}
A program that prints out pointers and the values they point to:
#include <stdio.h>
int main()
{
int a = 10, b = 2;
int *p1, *p2;
p1 = &a;
p2 = &b;
printf("%p %p \n", p1, p2);
printf("%d %d \n", *p1, *p2);
return 0;
}
0xbfffdac4 0xbfffdac0
10 2
A program that swaps integers using a function that accepts pointers to the integers to be
swapped:
#include <stdio.h>
int main()
{
int a = 10, b = 2, x = 3, y = 5;
printf("a,b,x,y: %d,%d,%d,%d\n", a, b, x, y);
swap(&x, &y);
swap(&a, &b);
printf("a,b,x,y: %d,%d,%d,%d\n", a, b, x, y);
}
a,b,x,y: 10,2,3,5
a,b,x,y: 2,10,5,3
#include <stdio.h>
void rectangle(int a, int b, int * area, int * perim);
int main()
{
int x, y;
int area, perim;
printf("Enter two values separated by space: " );
scanf("%d %d", &x, &y);
rectangle(x, y, &area, &perim);
printf("Area is %d Perimeter is %d\n", area, perim);
return 0;
}
A program that reads elements and assign them in an array. Then, it prints the array elements
out.
#include <stdio.h>
#define SIZE 10
int main()
{
int a[SIZE];
get_array(a, SIZE);
prt_array(a, SIZE);
printf("\n");
return 0;
}
A function is a module or block of program code which deals with a particular task. Making
functions is a way of isolating one block of code from other independent blocks of code.
They allow a programmer to say: `this piece of code does a specific job which stands by itself
and should not be mixed up with anyting else',
*
Second they make a block of code reusable since a function can be reused in many different
contexts without repeating parts of the program text.
A function can take a number of parameters, do required processing and then return a value.
There may be a function which does not return any value.
You already have seen couple of built-in functions like printf(); Similar way you can define your
own functions in C language.
To turn it into a function you simply wrap the code in a pair of curly brackets to convert it into a
single compound statement and write the name that you want to give it in front of the
brackets:
Demo()
{
int total = 10;
printf("Hello World");
total = total + l;
}
curved brackets after the function's name are required. You can pass one or more paramenters
to a function as follows:
By default function does not return anything. But you can make a function to return any value
as follows:
return total;
}
A return keyword is used to return a value and datatype of the returned value is specified
before the name of function. In this case function returns total which is int type. If a function
does not return a value then void keyword can be used as return value.
Once you have defined your function you can use it within a program:
main()
{
Demo();
}
Each function behaves the same way as C language standard function main(). So a function will
have its own local variables defined. In the above example total variable is local to the function
Demo.
A global variable can be accessed in any function in similar way it is accessed in main() function.
Declaration and Definition
When a function is defined at any place in the program then it is called function definition. At
the time of definition of a function actual logic is implemented with-in the function.
A function declaration does not have any body and they just have their interfaces.
A function declaration is usually declared at the top of a C source file, or in a separate header
file.
A function declaration is sometime called function prototype or function signature. For the
above Demo() function which returns an integer, and takes two parameters a function
declaration will be as follows:
Pass by Value: mechanism is used when you don't want to change the value of passed
paramters. When parameters are passed by value then functions in C create copies of the
passed in variables and do required processing on these copied variables.
*
Pass by Reference mechanism is used when you want a function to do the changes in passed
parameters and reflect those changes back to the calling function. In this case only addresses of
the variables are passed to a function so that function can work directly over the addresses.
Here are two programs to understand the difference: First example is for Pass by value:
#include <stdio.h>
int main()
{
int a = 10;
int b = 20;
t = p2;
p2 = p1;
p1 = t;
printf("Value of a (p1) = %d and value of b(p2) = %d\n", p1, p2 );
}
Here is the result produced by the above example. Here the values of a and b remain
unchanged before calling swap function and after calling swap function.
#include <stdio.h>
int main()
{
int a = 10;
int b = 20;
t = *p2;
*p2 = *p1;
*p1 = t;
printf("Value of a (p1) = %d and value of b(p2) = %d\n", *p1, *p2 );
}
Here is the result produced by the above example. Here the values of a and b are changes after
calling swap function.
C - Pre-Processors
The C Preprocessor is not part of the compiler, but is a separate step in the compilation
process. In simplistic terms, a C Preprocessor is just a text substitution tool. We'll refer to the C
Preprocessor as the CPP.
Pre-Processors Examples:
#define MAX_ARRAY_LENGTH 20
Tells the CPP to replace instances of MAX_ARRAY_LENGTH with 20. Use #define for constants
to increase readability.
#include <stdio.h>
#include "myheader.h"
Tells the CPP to get stdio.h from System Libraries and add the text to this file. The next line tells
CPP to get myheader.h from the local directory and add the text to the file.
#undef FILE_SIZE
#define FILE_SIZE 42
Tells the CPP to undefine FILE_SIZE and define it for 42.
#ifndef MESSAGE
#define MESSAGE "You wish!"
#endif
Tells the CPP to define MESSAGE only if MESSAGE isn't defined already.
#ifdef DEBUG
/* Your debugging statements here */
#endif
Tells the CPP to do the following statements if DEBUG is defined. This is useful if you pass the -
DDEBUG flag to gcc. This will define DEBUG, so you can turn debugging on and off on the fly!
Stringize (#):
The stringize or number-sign operator ('#'), when used within a macro definition, converts a
macro parameter into a string constant. This operator may be used only in a macro that has a
specified argument or parameter list.
When the stringize operator immediately precedes the name of one of the macro parameters,
the parameter passed to the macro is enclosed within quotation marks and is treated as a string
literal. For example:
#include <stdio.h>
#define message_for(a, b) \
printf(#a " and " #b ": We love you!\n")
int main(void)
{
message_for(Carole, Debra);
return 0;
}
The token-pasting operator (##) within a macro definition combines two arguments. It permits
two separate tokens in the macro definition to be joined into a single token.
If the name of a macro parameter used in the macro definition is immediately preceded or
followed by the token-pasting operator, the macro parameter and the token-pasting operator
are replaced by the value of the passed parameter.Text that is adjacent to the token-pasting
operator that is not the name of a macro parameter is not affected. For example:
tokenpaster(34);
This example results in the following actual output from the preprocessor:
This example shows the concatenation of token##n into token34. Both the stringize and the
token-pasting operators are used in this example.
Parameterized Macros:
One of the powerful functions of the CPP is the ability to simulate functions using
parameterized macros. For example, we might have some code to square a number:
int square(int x) {
return x * x;
}
Macros with arguments must be defined using the #define directive before they can be used.
The argument list is enclosed in parentheses and must immediately follow the macro name.
Spaces are not allowed between and macro name and open parenthesis. For example:
Macro Caveats:
Macro definitions are not stored in the object file. They are only active for the duration of a
single source file starting when they are defined and ending when they are undefined (using
#undef), redefined, or when the end of the source file is found.
*
Macro definitions you wish to use in multiple source files may be defined in an include file
which may be included in each source file where the macros are required.
*
When a macro with arguments is invoked, the macro processor substitutes the arguments
into the macro body and then processes the results again for additional macro calls. This makes
it possible, but confusing, to piece together a macro call from the macro body and from the
macro arguments.
*
Most experienced C programmers enclose macro arguments in parentheses when they are
used in the macro body. This technique prevents undesired grouping of compound expressions
used as arguments and helps avoid operator precedence rules overriding the intended meaning
of a macro.
*
While a macro may contain references to other macros, references to itself are not
expanded. Self-referencing macros are a special feature of ANSI Standard C in that the self-
reference is not interpreted as a macro call. This special rule also applies to indirectly self-
referencing macros (or macros that reference themselves through another macro).