Hello everyone, I am Xiaoyu. Today we are going to talk about a super basic yet extremely important topic in the C language – data types and variables, especially integers, floating-point numbers, characters, and pointer types.
Don’t underestimate these basics; they actually have a profound impact on your programming world. So today, let’s dive deep into understanding them, ensuring you go from “Wow, cool” to “Haha, I get it!”

1. Data Types and Variables: The Basic Building Blocks of C Language
In C language, data types are the foundation for defining variables, meaning they determine how our variables are stored and what operations can be performed on them. Just like everyone has their own unique personality and abilities, different data types in C have different uses.
Data types are divided into two main categories:
- 
Basic Data Types: including integer types, floating-point types, character types, etc. 
- 
Derived Data Types: such as arrays, structures, unions, and pointer types. 
Today, we will focus on the basic data types: integers, floating-point numbers, characters, and pointers.
2. Integer Types: Don’t Underestimate This Guy
2.1 What is an Integer Type?
Integer types in C language are used to store whole numbers, meaning numbers without decimal parts. Integer types can be divided into different types based on the size of memory they occupy, commonly including int, short, long, and long long.
2.2 Code Example:
#include <stdio.h>
int main() {
    int a = 100;         // 4 bytes, range from -2,147,483,648 to 2,147,483,647
    short b = 20;        // 2 bytes, range from -32,768 to 32,767
    long c = 100000L;    // 4 or 8 bytes, range from -2,147,483,648 to 2,147,483,647 (system-dependent)
    long long d = 123456789012LL; // 8 bytes, typically from -9,223,372,036,854,775,808 to 9,223,372,036,854,775,807
    printf("int: %d\n", a);
    printf("short: %hd\n", b);
    printf("long: %ld\n", c);
    printf("long long: %lld\n", d);
    return 0;
}
2.3 Detailed Analysis
- 
int: the most commonly used integer type, default size is 4 bytes, storage range is -2,147,483,648 to 2,147,483,647 (typically the size on a 32-bit system).
- 
short: typically 2 bytes, suitable for storing smaller range integers, occupying less memory.
- 
long: larger thanint(typically 4 bytes or 8 bytes, depending on system architecture), suitable for situations requiring larger range storage.
- 
long long: the largest integer type, typically 8 bytes.
You might ask, “Which type should I use?” Simply put, when choosing a type, consider:
- 
Memory Usage: If you only need to store small range integers, use short.
- 
Range Requirements: If you need to store large range numbers, longorlong longis more suitable.
- 
Platform Dependency: On 32-bit and 64-bit systems, the size of longmay differ, so be careful to choose the appropriate type.
2.4 Advanced: Understanding Integer Overflow
Integer overflow occurs when the value you store exceeds the maximum range of the data type. For example, in a 32-bit system, if you set an int variable to 2147483648, it will overflow, resulting in a negative number.
Let’s look at an example of overflow:
#include <stdio.h>
int main() {
    int max_int = 2147483647; // max value for int
    printf("Before overflow: %d\n", max_int);
    max_int = max_int + 1; // overflow occurs
    printf("After overflow: %d\n", max_int);
    
    return 0;
}
The output will surprise you:
Before overflow: 2147483647
After overflow: -2147483648
The reason for the overflow is that the storage of integers is limited, exceeding this limit can lead to unpredictable results. You can define an unsigned integer type using unsigned to avoid negative interference.
unsigned int ui = 4294967295; // max value for unsigned int
3. Floating-Point Types: This Type Can Help You Solve All “Decimal” Problems
3.1 What is a Floating-Point Type?
Floating-point types are used to represent numbers with decimals, such as 3.14, -0.5, etc. The C language provides two common floating-point types:
- 
float: single precision floating-point number, typically occupying 4 bytes, with a precision of about 6~7 significant digits.
- 
double: double precision floating-point number, typically occupying 8 bytes, with a precision of about 15~16 significant digits.
- 
long double: extended precision floating-point number, typically occupying 12~16 bytes, with a precision of about 18~19 significant digits.
3.2 Code Example:
#include <stdio.h>
int main() {
    float f = 3.14159f;
    double d = 3.141592653589793;
    long double ld = 3.14159265358979323846264338327950288L;
    printf("float: %.6f\n", f);
    printf("double: %.15f\n", d);
    printf("long double: %.30Lf\n", ld);
    return 0;
}
3.3 Precision Issues and Errors
A common issue with floating-point numbers is precision errors. Since computer memory stores floating-point numbers in binary, some decimal fractions cannot be accurately represented in binary, leading to small errors.
For example:
#include <stdio.h>
int main() {
    double x = 0.1 + 0.2;
    printf("x = %.20f\n", x);  // 0.30000000000000004
    return 0;
}
What you see as 0.1 + 0.2 is actually not 0.3; its value becomes 0.30000000000000004, which reflects the precision of floating-point numbers. When high precision calculations are needed, consider using double or long double, or utilize specialized libraries (like MPFR) to avoid precision loss.
3.4 Advanced: How to Avoid Floating-Point Comparison Issues?
Due to the precision limitations of floating-point numbers, directly comparing two floating-point numbers for equality may lead to errors. We can use a small tolerance range (commonly referred to as epsilon) for comparison:
#include <stdio.h>
#include <math.h>
#define EPSILON 1e-6
int main() {
    double a = 0.1 + 0.2;
    double b = 0.3;
    
    if (fabs(a - b) < EPSILON) {
        printf("a and b are considered equal\n");
    } else {
        printf("a and b are not equal\n");
    }
    return 0;
}
This code will output:
a and b are considered equal
This way, we avoid the errors that may arise from directly comparing two floating-point numbers.
4. Character Types: Your “Single Character” Friend
4.1 What is a Character Type?
Character types are used to store a single character. Characters in C language are based on ASCII codes and typically occupy 1 byte. You can assign values through character constants (like 'A').
4.2 Code Example:
#include <stdio.h>
int main() {
    char c = 'A'; // store character 'A'
    printf("Character: %c\n", c);
    printf("ASCII value: %d\n", c); // print the ASCII value of the character
    return 0;
}
4.3 Characters and ASCII Codes
In C language, the actual storage of character variables is their ASCII code. For example, the character 'A' corresponds to the ASCII value 65, while the character 'a' corresponds to the ASCII value 97.
#include <stdio.h>
int main() {
    char c = 'A';
    printf("Character 'A' is stored as: %d\n", c); // outputs 65
    return 0;
}
4.4 Advanced: Character Arrays and Strings
In C language, strings are actually in the form of character arrays. You can handle multiple characters through arrays, and strings always end with '\0' (NULL character).
#include <stdio.h>
int main() {
    char str[] = "Hello, World!";
    printf("String: %s\n", str);
    return 0;
}
5. Pointer Types: Manipulating Memory with Pointers
5.1 What is a Pointer Type?
Pointer types are used to store the memory addresses of variables rather than their values. Pointers are a powerful tool in C language that enable efficient memory management and dynamic memory allocation.
5.2 Code Example:
#include <stdio.h>
int main() {
    int a = 10;
    int *p = &a; // p pointer stores the address of a
    printf("a = %d\n", a);  // output the value of a
    printf("Address of a: %p\n", (void*)p);  // output the address of a
    printf("Value at address: %d\n", *p);  // access the value of a through pointer
    return 0;
}
5.3 Advanced: Tips for Using Pointers
Pointers can be used to implement many advanced features, such as dynamic memory allocation, returning multiple values from functions, etc. For example:
#include <stdio.h>
#include <stdlib.h>
int* create_array(int size) {
    return (int*)malloc(size * sizeof(int));
}
int main() {
    int* arr = create_array(5);
    
    for (int i = 0; i < 5; i++) {
        arr[i] = i * 2;
    }
    for (int i = 0; i < 5; i++) {
        printf("arr[%d] = %d\n", i, arr[i]);
    }
    free(arr); // free dynamically allocated memory
    return 0;
}
With pointers, you can dynamically allocate memory and handle large amounts of data without being limited to stack memory constraints.
Conclusion
Today we have analyzed in detail the integer, floating-point, character, and pointer types in C language. You should now understand the essence and usage of these basic data types and be able to flexibly choose the appropriate data type to solve problems during programming. Although pointers are a bit complex, they offer powerful functionality, especially in dynamic memory management and efficiently handling large amounts of data, making them indispensable.
Remember, in C language, each data type has its unique charm. Choosing the right one can make your program run more efficiently and your code more elegant!