- Will this code run? If yes what is it going to return?
int main(){
printf("Hello World!");
}
Answer: Yes this will run in C (not C++). C compiler will create an implicit declaration for the function printf(), compile this code into an object file. This is going to return 0. You can check it using echo $?
.
- Swap 2 variables without using temporary variable
Using '+' and '-':
a = a + b;
b = a - b;
a = a - b;
OR
Using '*' and '/':
a = a * b;
b = a / b;
a = a / b;
OR
Using ^ (ex-or):
a = a ^ b;
b = a ^ b;
a = a ^ b;
- Implementation of strlen
int impStrlen (char *str) {
// pointer to the starting address of str
char *temp= str ;
while (*temp++)
// You can print value of temp to get an idea of what is happening
//printf("%d\n", temp);
;
return (temp - str - 1 );
}
- Check if string is palindrome or not
int palindromeCheck (char *str){
char *p1 = str; // pointer pointing to starting of the string
char *p2 = str + strlen (str) - 1; // pointer pointing to ending of the string
while (p1 < p2){
if (*p1++ != *p2--)
return 0; // not a palindrome
}
return 1; // it is a palindrome
}
- Set or reset a specific bit in a given variable
e.g.
8 -> 1000
we want to set 0th position so that it become 1001 (i.e. 9)
3 -> 0011
we want to reset 0th position so that it becomes 0010 (i.e. 2)
#include <stdio.h>
// We can do a bitwise AND (&) or OR (|) to reset or set the bit
#define BITWISE_OP(x) (0x1 << x)
int set(int val, int pos){
return (val |= BITWISE_OP(pos)) ;
}
int reset(int val, int pos) {
return (val &= ~BITWISE_OP(pos)) ;
}
int main (){
printf("%d\n", set(8,0));
printf("%d\n", reset(3,0));
return 0;
}
Output is:
9
2
- We can use bitwise operators in many situation
Rewriting int c = a * 16 + b / 2;
int c = (a << 4) + (b >> 1)
- Complement of integer without directly using ~
refer to this: https://stackoverflow.com/questions/791328/how-does-the-bitwise-complement-operator-work
#include <stdio.h>
void complement(int i) {
printf( "Method 1: %d\n", i^(~0));
printf( "Method 2: %d\n" , -(i+1)) ;
}
int main (){
complement(5);
return 0;
}
- Multiline macros
It is best practice to set a pointer to NULL after freeing it. We can use a macros for this. Below is the code for the same.
#include <stdio.h>
#include <stdlib.h>
#define freeNULL(p) {free(p); p = NULL; printf("FIN");}
int main (){
int *ptr = (int *) malloc( 1 * sizeof(int *) );
if(1)
freeNULL(ptr);
else
printf("Nothing to do here...");
return(0);
}
But this code gives an error message:
In function 'main':
12:5: error: 'else' without a previous 'if'
else
^~~~
why so?
The reson is that after puting the macros the if statement becomes something like this:
if(1){
free(p);
p = NULL;
printf("FIN");
};
The semicolon at the end cause the error. The Best way to overcome this is to use a do-while(0) loop. This loop runs only once, and after using this loop the if statement becomes something like this:
if(1)
do {
free(p);
p = NULL;
printf("FIN");
}while(0);
So our program looks like:
#include <stdio.h>
#include <stdlib.h>
#define freeNULL(p) do{free(p); p = NULL; printf("FIN");}while(0)
int main (){
int *ptr = (int *) malloc( 1 * sizeof(int *) );
if(1)
freeNULL(ptr);
else
printf("Nothing to do here...");
return(0);
}
- Find size of a variable without using sizeof operator
#include <stdio.h>
#define mySizeof(x) (((x *) 0) + 1)
int main(){
printf("%zd\n", mySizeof(char));
printf("%zd\n", mySizeof(int));
printf("%zd\n", mySizeof(float));
printf("%zd\n", mySizeof(double));
}
- Swap two nibbles in a byte
before swap:
18 -> 0001 0010
after swap:
0010 0001 -> 33
#include <stdio.h>
#define swap(x) (((x & 0xF) << 4 ) | ((x&0xF0) >> 4))
int main(){
printf("%d\n", swap(18));
}
- Find middle node in a singly linked list
The idea is to take two pointers pointing to head. Now increament 1st pointer by 1 and 2nd pointer by 2 in each iteration. When the 2nd pointer reach the end of the linked list then the first pointer is pointing to middle node.
Node *middleNode(Node *head){
Node *p1 = head, *p2 = head;
while(p2 != 0){
p1 = p1->next;
p2 = (p2->next) ? p2->next->next : 0;
}
return p1;
}
- Check if the number is power of 2
Any number which is power of 2 and the number 1 less returns 0 when we apply a bitwise AND (&) operation.
e.g.
8 -> 1000
7 -> 0111
and (1000 & 0111) = 0000
checkIfPowerOf2(int n){
return(!(n & (n-1)));
}
- Delete a node in a linked list, pointer to which is given
void deleteNode(Node *node){
// traverse the list till the end keep on copying the next node to the current node
while(node != NULL){
node.data = node->next.data;
node = node->next;
}
}
This way the time complexity is O(n). Another way of doing this is just copy the next node detail in the current node and free the next node. But in this way you need to make sure that a next node is present in the list. This way the time complexity is O(1).
void deleteNode(Node *node){
if(node->next != NULL){
// copy data of next node
node.data = node->next.data;
// get pointer to the next node
Node *freeNode = node->next;
// copy pointer to the next to next node
node->next = node->next->next;
// free next node
free(freeNode);
}
}
- How would you know whether your system is Big-endian or Little-endian
The order in which the variable is stored in machine referred to as endian.
Sequence order -> big endian
Reverse order -> little endian
Eg. take hex number 0x0304
Machine A stores 2 bytes at a time and the sequence of store is: 0x03 followed by 0x04
Machine B stores 2 bytes at a time and the sequence of store is: 0x04 followed by 0x03
Therefore
machine A -> big endian
machine B -> little endian
#include <stdio.h>
int main(){
// This is a very nice example to demostrate the use of union
// In union all member shares same memory location
union{
short value;
char store[sizeof(short)];
} endianCheck;
endianCheck.value = 0x0304;
// Here endianCheck share same memory location as of value
if(endianCheck.store[0] == 0x03 && endianCheck.store[sizeof(short) - 1] == 0x04)
printf("big-endian system");
else
printf("little-endian system");
return 0;
}
- Find if a number is power of two
The logic is very straight forward.
2 => 0010; 2-1 (i.e. 1) => 0001; 2 & 1 => 0010 & 0001 = 0000
4 => 0100; 4-1 (i.e. 3) => 0011; 4 & 3 => 0100 & 0011 = 0000
8 => 1000; 8-1 (i.e. 7) => 0111; 8 & 7 => 1000 & 0111 = 0000 ...
int isPowerOf2(int num){
return(!(num & (num -1)));
}
- Very interesting code. Print binary representation of any number. Let's do it recursively.
#include <stdio.h>
void decimalToBinary(int num){
if(num == 0)
return;
decimalToBinary(num >> 1);
(num & 01) ? printf("1") : printf("0");
}
int main(){
decimalToBinary(965);
return 0;
}
Output: 1111000101
- Check if implemetation of shift operator (>>) follows logical shift or arithematic shift.
The results are implementation-dependent for right shifts of signed values.
Logical right shift: 1 bit shifts all the bits to the right and fills in the leftmost bit with a '0'.
Arithmetic right shift: 1 bit shifts all the bits to the right and fills in the leftmost bit with original value of the right mst bit shifted.
e.g.
1 bit Logical Right Shift
unsigned char x = x >> 1;
Before: 00010000 ( = 16 decimal )
After: 00001000 ( = 8 decimal )
Before: 01001001
After: 00100100 ( one bit is "lost". LSB of original data)
Before: 10000001 ( = 129 decimal )
After: 01000000 ( = 64 decimal - one bit is "lost")
1 bit Arithmetic Right Shift
signed char x = x >> 1;
Before: 00001000 ( = 8 decimal )
After: 00000100 ( = 4 decimal )
Before: 10000011 ( = -125 decimal in two's complement )
After: 11000001 ( = -63 decimal )
After: 11100000 ( = -32 decimal )
After: 11110000 ( = -16 decimal )
After: 11111000 ( = -8 decimal )
After: 11111100 ( = -4 decimal )
After: 11111110 ( = -2 decimal )
After: 11111111 ( = -1 decimal )
After: 11111111 ( = -1 decimal )
More info: http://teaching.idallen.com/dat2343/09f/notes/04bit_operations_shift.txt
Use:
Logical shift: Used in serial data communication to transfer the data bit by bit or for multiplying unsigned numbers by power of 2.
Arithematic shift: Used in signed arithmetic computations.
So the program?
int main(){
((-1 >> 1) < 0) ? printf("Arithematic shift") : printf("Logical shift");
}
- Output of the code?
int main(){
printf("%d\n", 1 < 2 < 3);
printf("%d\n", 3 > 2 > 1);
}
Output:
1
0
For 1st printf(), ( 1 < 2 ) is 1 and ( 1 < 3 ) is true hence 1 is printed. For 2nd printf(), ( 3 > 2) is 1 and (1 > 1) is false (0), and hence 0 is printed.
- Output of the code?
int main(){
int i = 1;
printf("%d", ++i++);
}
- error: lvalue required as increment operand
(++i) is a rvalue expression and postfix ++ requires an lvalue, hence error.
Generally lvalue is objects that occupies some identifiable memory location.
eg.
int a = 7;
a is a lvalue
4 = b; // error!
(b + 1) = 4; // error!
b+1 and 4 both are not lvalue which makes them rvalue.
- Output of the code?
int main(){
int i = 1;
int *ptr = &i;
printf("%d", ++*ptr++);
}
Output: 2
When we combine prefix ++ with dereferencing operator * we get a result which is a lvalue i.e (++*ptr) and the value is now 2. Now postfix ++ needs a lvalue which is available so it's fine. Though there is no immediate effect of postfix ++, hence the value printed is still 2.
- Output of the code?
int main(){
int i = 3;
printf("%d", ~!-+~i);
}
Output: -1
Reason:
~i = -4
+~i does not have any effect because + is dummy operator in C.
-+~i = 4
!-+~i = 0 i.e. negation of any non zero value is 0
~!-+~i = -1 i.e. ~0 = -1
#include <stdio.h>
int main(){
int * const i = 20;
printf("%d", *++i);
return 0;
}
- error: increment of read-only variable 'i'
Reason: We are trying to change the value of constant pointer.
#include <stdio.h>
int main(){
float f = 2.0 * 3.0 - 6.0 + 1.0;
switch(f){
case 1.0:
printf("%f", 1.0);
break;
case 2.0:
printf("%f", 2.0);
break;
default:
printf("%f", f);
}
}
- error: switch quantity not an integer
- error: case label does not reduce to an integer constant
Reason: Switch/case can only accept integers.