Operators
The basic mathematical operators in the C Langauge are likely much like the basic operators in whatever language you've programmed in the past, e.g. +, -, *, /, %, =, >, <, >=, <=, and ==. It is worth noting that = is used for assignment and == is used for comparison.One curious thing about the C Language is that it was originally designed without a boolean type. Instead, 0 is interpreted to be false and any non-zero value is interpreted to be true.
Another curious thing is that, in C, assignments are expressions and evaluate to the assigned value, e.g. "x = 5" has a value of 5. For this reason, it is considered good coding style to put a constant on the left side of an assignment, e.g. one might choose to write "if (5 == x)" instead of "if ( x == 5)". Under normal circumstances, there is no difference, but if one goofs up and writes "=" instead of "==", the first expression will produce a compilation error (since 5 is not a variable and can't be reasigned), but the second expression will fail silently at runtime, reassigning x to be 5.
It is probably also worth mentioning the increment and decrement operators ++ and --. One nuance of the C language is that they behave differntly depending upon which side of the variable they are placed. Consider this example:
  
  x = 0;
  if (x++) {
    printf ("Doesn't get executed\n");
  } else {
    printf ("Bet this is a surprise: %d\n", x); // Prints 0
  }
  printf ("x is %d\n", x); // Prints 1
  
  
  As compared to this example:
  
  x = 0;
  if (++x) {
    printf ("Bet this is what you expected: %d\n", x); // Prints 1
  } else {
    printf ("Doesn't get executed\n");
  }
  printf ("x is %d\n", x); // Prints 1
  
  
  Notice that when a ++ (or --) comes after the variable that is being incremented, the expression has the value of the new, incremented (or decremented) value, but if ++ (or --) comes before the variable that is being incremented, the expression has the value of the new, incremented (or decremented) value. In any case, after the expression is evaluated and before it can otherwise be used it has the updated value.
C's Bit-Wise Operators
C has several operators that are defined in terms of manipulations upon the individual bits that compose values, rather than upon the whole values. These operators include the following:
Left Shift
The effect of the left shift operator is pretty easy to understand. At a high-level, it acts to multiply numbers by powers of two. For example:
int x = 5; int y; y = x << 1; /* Multiply x by 2^1 */ printf ("%d\n", y); /* 10 */ y = x << 3; /* Multiply x by 2^3 */ printf ("%d\n", y); /* 40 */The reason that this works is that numbers are stored in memory in what is, in effect, a binary format. In decimal, each position represents a power of 10: 100=10^2, 10=10^1, 1=10^0. Each binary digit (bit) represents a power of 2.
So, each time we shift by one posititon to the left, back-filling the vacated positions with 0s, we are, making each of the shifted bits worth twice as much:
Right Shift
In theory a right-shift is the opposite of a left shift. It has the effect of dividing by two, as shown below:
And, what we've got above is exactly true for unsigned types and also for any non-negative value. But, the Right Thing to do gets a bit confused when it comes to signed types. In my experience, the effect will always be the same as dividing by two. But, this is absolutely not guaranteed by the standard -- the standard leaves the effect of any right-shift of a negative value completely up to the compiler.
The reason for this you'll study, in detail, in 15-213. It has to do with the fact that most hardware does not represent negative numbers in the same way as it does positive numbers. It uses an encoding known as two's compliment. This encoding makes, when a negative number is involved, simple math much easier to implement in hardware.
But, it also means that a so-called "arithmetic right shift" is harder to do. Instead of always shifting a 0 in on the right side, a 1 must be shifted in if the exisitng left-mst bit is a 1.
Although I think the standard should have been for the right shift \ to be "arithmetic", meaning have the same effect as dividing by a power of two, the standard leaves this to the compiler.
The Bit-Wise AND and OR Operators
The bit-wise AND and bit-wise OR operators act exactly as do their so-called logical version, but they operate at a per-bit, bit-wise level upon corresponding bits of the two operands, as shown below:
  
  
  /* 
   * Notice that only those bits that are 1 in both of the corresponding
   * input bits are 1 in the result 
   */
  x  = 9;         /* 1001 */
  y  = 5;         /* 0101 */
  z = x & y;      /* 0001 */
  /* 
   * Notice that those bits that are 1 in EITHER (or both) of the corresponding
   * input bits are 1 in the result
   */
  x  = 9;         /* 1001 */
  y  = 5;         /* 0101 */
  z = x | y;      /* 1101 */
  
  
  
  Bit-wise Exclusive OR and Bit-wise Negation
Although there is no logical XOR in C, there is a bit-wise version. It does exactly as one might expect. Each result bit is true if, and only if, exactly one of the corresponding input bits is on. A result bit is false in the case that both correspnding bits are in the same state, whether true or false, as shown in the example below:
The negation operator simply flips each bit. Each one becomes a zero, and vice-versa: