首页 > 其他分享 >C语言常量和字面量

C语言常量和字面量

时间:2024-09-07 09:53:54浏览次数:17  
标签:const 常量 int C语言 printf define 字面

目录

引言

1. 字面量

1.1 字符字面量

1.2 整型字面量

1.3 浮点字面量

2. 常量

2.1 使用预处理器指令 #define 定义常量

2.1.1 语法格式

2.1.2 使用举例

2.2 使用 const 关键字定义常量

2.3 使用 #define 和 const 定义常量的区别


引言

        字面量是直接在代码中使用的程序运行期间不会改变的没有名字的固定值。所以我们只能用字面量来称呼它,故叫字面量。

        常量是用预处理器指令 #define 或 const 关键字定义的在程序运行期间不会改变的在代码中有名字的固定值。

1. 字面量

int i = 100;

像上面语句中的数字 100,这种在程序中不能修改的无法用名字称呼它的字面值常量,叫做字面量。比如 printf("Hello World\n"); 中的 "Hello World\n" 也是字面量。

1.1 字符字面量

        在 ASCII 码表中,编码 [0, 31] 和 127 是控制字符,共 33 个字符是不可以打印的,其中编码 [0, 31] 这 32 个字符主要用于控制打印机等外围设备。

        编码 [32, 126] 是可显示字符,也就是可以打印在屏幕上。记得以下几个编码规则,在 C语言对字符进行操作的时候,或许挺管用。

  • 大写字母 A ~ Z 的 ASCII码 65 ~ 90
  • 小写字母 a ~ z 的 ASCII码 97 ~ 122
  • 大写字母比小写字母的 ASCII码小 32 (0b00100000,0x20) ,比如: char c = 'a',小写字母变大写字母,c = c - 32 得到,也可以做位与操作,c = c & 0b11011111 得到
  • 数字字符 0 ~ 9 的 ASCII码 48 ~ 57,所以一个数字要转换成数字字符,只要将数字 加上 48 (0x30) 即可
  • 字符串的终止符为 '\0',ASCII 码为 0
  • 换行符 '\n' 的 ASCII码为 10 (0x0A)

        对于这些字符字面量,有4种书写格式

1) 用一对单引号把字符括起来(推荐使用)

#include <stdio.h>

int main() {
    char c1 = 'A';
    printf("%c\n", c1);    // A

    return 0;
}

 对于不可打印字符和特殊字符,用转义字符书写。

\n换行符\t水平制表符
\v纵向制表符\b退格符
\r回车符\f换页符
\a报警响铃符\\反斜杠
\"双引号\'单引号
#include <stdio.h>

int main() {
    char c1 = '\\';

    printf("%c\n", c1);

    return 0;
}

2) 用编码数值书写

#include <stdio.h>

int main() {
    char c1 = 65;     // 十进制,大写字母 A 的ASCII码:65
    char c2 = 0101;   // 八进制
    char c3 = 0x41;   // 十六进制
    printf("%c,%c,%c\n", c1, c2, c3);    // A,A,A

    return 0;
}

3) 对于任何字符,还可以用通用的转义字符书写

  • \ooo,其中 ooo 表示三个八进制数字,注意是三个八进制数字,比如即使是编码为 1 的字符,也不能省掉前面那两个0。
  • \xhh,其中 x 表示十六进制,hh 是二个十六进制数字
#include <stdio.h>

int main() {
    char c1 = '\x41';   // 十六进制
    char c2 = '\101';   // 是3位八进制数字,即使编码为 1 的字符,也要写成 '\001',不能写成 '\1'

    printf("%c,%c", c1, c2);    // A,A

    return 0;
}

1.2 整型字面量

        整型字面量可以用十进制、八进制、十六进制或二进制来表示。

  • 以 0x 或 0X 开头表示十六进制
  • 以数字 0 开头表示八进制
  • 以 0b 开头表示二进制
  • 不带前缀默认是十进制

以下表示同一个整型字面量

15                        // 十进制

017                      // 八进制

0xf                       // 十六进制

0b00001111        // 二进制

#include <stdio.h>

int main() {
    int i1 = 15;
    int i2 = 017;
    int i3 = 0xf;
    int i4 = 0b00001111;
    printf("%d, %d, %d, %d\n", i1, i2, i3, i4);

    return 0;
}

整型字面量还可以带一个由 U 和 L 组合的后缀,U 和 L 不分先后顺序,U 可以大写也可以小写 u,是 unsigned 的意思,L 可以大写也可以小写 l,是 long 的意思。

15u         // 无符号整型字面量

15L         //  长整形字面量,推荐用大写 L,因为小写字面 l 和数字 1 很容易混淆

15ul        // 无符号长整形字面量

0xfu        // 无符号整型字面量

0xfL        // 长整形字面量

0xful       // 无符号长整型字面量

017u      // 无符号整型字面量 

017L      // 长整形字面量

017ul     // 无符号长整型字面量

// limits.h 文件中,int, unsigned int, long, unsigned int 取值范围定义
#define INT_MIN     (-2147483647 - 1)
#define INT_MAX       2147483647
#define UINT_MAX      0xffffffff
#define LONG_MIN    (-2147483647L - 1)
#define LONG_MAX      2147483647L
#define ULONG_MAX     0xffffffffUL
#include <stdio.h>
#include <limits.h>

int main() {
    unsigned int uiVar = 15u;
    long longVar = 15L;
    unsigned long ulVar = 15ul;

    printf("%u\n", uiVar);      // %u 无符号整型格式化输出
    printf("%d\n", longVar);    // %d 以十进制格式化输出有符号整数(在 limits.h 中有定义: long 和 int 的取值范围是一样的)
    printf("%lu\n", ulVar);     // %lu 以无符号长整型格式化输出

    uiVar = 0xfu;
    longVar = 0xfL;
    ulVar = 0xful;
    printf("%u\n", uiVar);      // %u 无符号整型格式化输出
    printf("%d\n", longVar);    // %d 以十进制格式化输出有符号整数(在 limits.h 中有定义: long 和 int 的取值范围是一样的)
    printf("%lu\n", ulVar);     // %lu 以无符号长整型格式化输出

    uiVar = 017u;
    longVar = 017L;
    ulVar = 017ul;
    printf("%u\n", uiVar);      // %u 无符号整型格式化输出
    printf("%d\n", longVar);    // %d 以十进制格式化输出有符号整数(在 limits.h 中有定义: long 和 int 的取值范围是一样的)
    printf("%lu\n", ulVar);     // %lu 以无符号长整型格式化输出

    return 0;
}

注意:

018        是不对的整型字面量,因为八进制没有数码 8

15uu      是不对的整型字面量,后缀 u 不能重复

15Lu      是正确的整型字面量,跟 15uL表达的是同一个字面量,u和l组合时,不分先后顺序

1.3 浮点字面量

        通常用十进制或科学计数法来表示浮点字面量。

  • 默认浮点字面量为 double 类型,比如:3.14
  • 在数值后面加上后缀 f 或 F 表示 float 类型,比如:3.14f 或 3.14F
  • 0 的 double 类型表示为 0. ;0 的 float 类型表示为  0.f 或 0.F
  • 对于只有小数部分的浮点字面量,整数部分的 0 可以省略掉不写,比如:.01 等价于 0.01;.01f 等价于 0.01f
  • 使用科学计数法时,指数用 e 或 E 表示。比如:3.14e0 或 3.14E0 表示的是 double 类型 3.14;而 3.14e0f 或 3.14E0F 表示的是 float 类型 3.14f
  • double 类型 0 用科学计数法表示为 0e0,float 类型 0 用科学计数法表示为 0e0f
#include <stdio.h>

int main() {
    printf("%d, %d\n", sizeof(.01), sizeof(0.01));          // 8, 8
    printf("%d, %d\n", sizeof(.01f), sizeof(0.01f));        // 4, 4
    printf("%.2f, %.2f\n", .01, 0.01);                      // 0.01, 0.01
    printf("%d, %d\n", sizeof(0.), sizeof(0.f));            // 8, 4
    printf("%d\n", sizeof(3.14));                           // 8
    printf("%d, %d\n", sizeof(3.14f), sizeof(3.14F));       // 4, 4
    printf("%d, %d\n", sizeof(3.14e0), sizeof(3.14E0));     // 8, 8
    printf("%d, %d\n", sizeof(3.14e0f), sizeof(3.14E0F));   // 4, 4
    printf("%d, %d\n", sizeof(0e0), sizeof(0e0f));          // 8, 4
    return 0;
}

1.4 字符串字面量

        字符串字面量用一对双引号把所有的字符括在里面,比如:"Hello World\n"

  • 字符串字面量在内存中编译器会自动在末尾加上一个字符串终止符('\0'),所以字符串在内存中占用的字节数会比字符串的长度多1字节。
  • 两个相邻的仅由空格、制表符('\t') 或 换行符('\n') 分开的字符串字面量,可以连接成一个新的字符串字面量。
  • 一个字符串过长,在一行放不下时,可以使用反斜杠 ('\') 进行换行。
  • 空字符串 "" 在内存在占一个字节,这个字节存的是字符串终止符('\0')。
#include <stdio.h>

int main() {
    printf("Hello World\n");      
    printf("Hello"  " world\n");  // 两个字符串字面量连接成一个新的字符串 "Hello World\n"
    printf("Hello \
World\n");                        // 用 '\' 换行
    printf("%d\n", sizeof(""));   // 1
    return 0;
}

2. 常量

        常量是用预处理器指令 #define 或 const 关键字定义的在程序运行期间不会改变的在代码中有名字的固定值。

2.1 使用预处理器指令 #define 定义常量

2.1.1 语法格式
#define 常量名 常量值
// 以下代码定义了一个名为 PI 的常量
#define PI 3.14
2.1.2 使用举例
#include <stdio.h>

#define PI 3.14

// 圆周长
double circumference(const double r) {
	return 2 * PI * r;	// 使用了预处理器指令 #define 定义的常量 PI
}

int main() {
	printf("半径为 %.2f 的圆周长 = %.2f\n", 2.0, circumference(2.0));

	return 0;
}

2.2 使用 const 关键字定义常量

2.2.1 语法格式

const 常量类型 常量名 = 常量值;
// 以下代码定义了一个名为 PI 的常量
const double PI = 3.14;

2.2.2 使用举例

#include <stdio.h>

const double PI = 3.14;

// 圆周长
double circumference(const double r) {
	return 2 * PI * r;	// 使用了用 const 关键字定义的常量 PI
}

int main() {
	printf("半径为 %.2f 的圆周长 = %.2f\n", 2.0, circumference(2.0));

	return 0;
}

注意:用 const 关键字定义的常量,必须同时它进行初始化

/*
// 以下对定义常量 MAX 是错误的,因为常量必须在定义的时同时对它进行初始化,初始化后的常量是不能改变的
const int MAX;
MAX = 100;
*/
// 正确用 const 定义 MAX 常量的方法
const int MAX = 100;

2.3 使用 #define 和 const 定义常量的区别

  • 1. 编译器处理方式不同
  • 使用 #define 定义的常量会在预处理阶段用它对应的字面量替换掉代码中的常量字符标识,在程序运行期间内存中并不存在该常量。
  • 使用 const 定义的常量是具有数据类型和作用域的常量,在程序运行期间内存中是存在该常量的符号标识的
  • 2. 类型和安全检查不同

  • 使用 #define 定义的常量不存在数据类型,从而在编译阶段,编译器不会对它进行类型检查

  • 使用 const 定义的常量是具有数据类型的,在编译阶段,编译器会对它进行类型检查

  • 3. 作用域检查不同

  • 使用 #define 定义的常量没有作用域,它在定义之后的整个代码中都有效

  • 使用 const 定义的常量具有作用域,只有在它所在的作用域内有效

  • 4. 存储方式不同

  • 使用 #define 定义的常量在符号表中不会有相应的条目。

  • 使用 const 定义的常量会在符号表中有相应的条目,有助于调试和可读性。

        使用 #define 与 const 这两种方式都可以用来定义常量,通常情况下,建议使用 const 关键字来定义常量,因为通过 const 关键字定义的常量,具有数据类型和作用域,编译器可以对它进行类型检测和作用域检查,而用 #define 定义的常量,仅在预编译阶段进行简单的文本替换,可能会导致一些没有预料到的问题。

标签:const,常量,int,C语言,printf,define,字面
From: https://blog.csdn.net/d704791892/article/details/141947898

相关文章

  • 全国计算机二级考试C语言篇3——选择题
    C语言部分——C语言概述1.程序模块化的优点程序模块化的优点在于它可以使程序的开发、维护和复用变得更简单。下面是一些主要的优点:降低复杂度:模块化可以将复杂的问题分解成更小的、更易管理的部分。可维护性:模块化使得代码更易于维护,因为修改一个模块的影响被限制在该......
  • c语言内存函数
    今天来学习C语言中的内存函数目录1.memcpy代码形式示例运行结果2.memmove代码形式示例运行结果3.memset代码形式示例运行结果4.memcmp形式示例运行结果![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/b7d8d59577b248deaa7b869d014d8b4f.png#pic_center)5......
  • 学习笔记|鹏哥C语言——关键字
    一、 关键字typedeftypedef顾名思义是类型定义,这里应该理解为类型重命名。比如:二、 关键字static在C语言中:static是用来修饰变量和函数的1.修饰局部变量-称为静态局部变量2.修饰全局变量-称为静态全局变量3.修饰函数-称为静态函数三、 修饰局部变量......
  • E31.【C语言】练习:指针运算习题集(上)
    Exercise1求下列代码的运行结果#include<stdio.h>intmain(){ inta[5]={1,2,3,4,5}; int*ptr=(int*)(&a+1); printf("%d",*(ptr-1)); return0;}答案速查:分析:Exercise2 求下列代码的运行结果//在x86环境下//假设结构体的大小是20个字节......
  • C语言中有关函数的知识
        前言        C语言函数是一种函数,用来编译C语言,一般包括字符库函数,数学函数,目录函数,进程函数,诊断函数,操作函数等    这里这个函数和我们高中时期学的函数类似,高中的函数是这样    F(x)=5x+21                   ......
  • C语言程序设计(初识C语言后部分)
    不要重来,不要白来,不要重来。5.指针和数组数组:一组相同类型元素的集合指针变量:是一个变量,存放的地址要理解数组名大部分情况下是数组的首元素地址6.二级指针先了解一级指针变量二级指针变量(二级指针变量是用来存放一级指针变量的地址的)7.指针数组指针数组是......
  • C语言数据类型和变量
    引言好久不见大家,最近因为在忙开学的事情很久没有更新,很感谢大家的支持,我会继续努力滴!!!前篇链接:http://#小程序://CSDN/Rz9Z9VlUkPV8ttg那我们长话短说:开始我们的这一节--------C语言数据类型和变量数据类型介绍C语⾔提供了丰富的数据类型来描述⽣活中的各种数据。......
  • C语言——使用回调函数模拟实现qsort
    同学们还记得之前我们已经学过一种排序方法叫“冒泡排序“嘛。代码直接附上咯voidbubble_sort(intarr[],intsz){ inti=0;//趟数 for(i=0;i<sz-1;i++) { intj=0; for(j=0;j<sz-i-1;j++) { if(arr[j]>arr[j+1]) { inttmp=......
  • C语言-第七章:字符和字符串函数、动态内存分配
    传送门:C语言-第六章-加餐:其他自定义类型目录第一节:字符和字符串函数    1-1.strlen函数和sizeof关键字    1-2.memcpy内存拷贝函数    1-3.memmove内存拷贝函数    1-4.memset内存设置函数    1-5.strtok字符串切割函数......
  • C语言-第六章-加餐:其他自定义类型
    传送门:C语言-第六章:结构体目录第一节:位段    1-1.位段是什么    1-2.位段的大小第二节:联合体    2-1.联合体是什么    2-2联合体的大小第三节:枚举类型    3-1.枚举是什么第四节:结构体中的柔性数组    4-1.柔性数组......