惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
H
Hacker News: Front Page
P
Palo Alto Networks Blog
T
ThreatConnect
Apple Machine Learning Research
Apple Machine Learning Research
博客园_首页
T
True Tiger Recordings
P
Privacy & Cybersecurity Law Blog
B
Blog
IT之家
IT之家
Last Week in AI
Last Week in AI
F
Full Disclosure
Hacker News: Ask HN
Hacker News: Ask HN
C
Comments on: Blog
Microsoft Azure Blog
Microsoft Azure Blog
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
博客园 - 【当耐特】
N
News and Events Feed by Topic
NISL@THU
NISL@THU
腾讯CDC
雷峰网
雷峰网
Security Latest
Security Latest
李成银的技术随笔
M
Microsoft Research Blog - Microsoft Research
L
LangChain Blog
L
Lohrmann on Cybersecurity
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
C
Check Point Blog
Y
Y Combinator Blog
Recent Announcements
Recent Announcements
博客园 - Franky
N
News | PayPal Newsroom
V
V2EX
A
About on SuperTechFans
The Register - Security
The Register - Security
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Google Online Security Blog
Google Online Security Blog
MyScale Blog
MyScale Blog
Cisco Talos Blog
Cisco Talos Blog
Vercel News
Vercel News
WordPress大学
WordPress大学
C
Cyber Attacks, Cyber Crime and Cyber Security
The Hacker News
The Hacker News
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
爱范儿
爱范儿
A
Arctic Wolf
L
LINUX DO - 最新话题
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More

Dejavu's Blog

使用 auto-cpufreq 平衡 Linux 性能功耗 哪吒监控面板部署教程 NodeBB 论坛部署教程 Scaleway VPS 安装 Alpine Linux 甲骨文 ARM 实例部署 Gemma 4 模型 Headscale + Tailscale 组建虚拟专用网 在 Linux 上使用 Yubikey OpenPGP 应用 BuyVM VPS 块存储挂载教程 Alpine Linux 服务器配置指南 Alpine Linux 安装 Cloudflared Docker 多容器共享中心数据库 安装 Komari 服务器监控工具 Scaleway VPS 安装 Debian Linux Debian 13 下部署 AsmBB 论坛 使用 Kopia 自动化备份服务器数据 给 Docker 启用 IPv6 支持 Netcup 服务器安装自定义 ISO 镜像 在 Debian 13 上自托管 Mox 邮局 使用 Stalwart 自托管邮局服务 烽火 HG5582A 光猫开启桥接模式 Docker 自托管 Shlink 短链服务 部署 Obsidian LiveSync 实时同步服务指南 我的 2025 年不完全回顾 我的 2025 年度自托管服务报告 Linux 下 Intel 核显驱动配置与硬件加速 Fedora Linux 安装配置记录 2025 年优雅地自托管 RSS 服务 自托管部署 Pocket ID 与 Tinyauth 完全指南 Woodpecker CI 和 Gitea 实现 Hugo 自动部署 Gitea/Forgejo 集成 Woodpecker CI/CD 在 Blinko 中使用 Ollama 作为 AI 供应商 Docker 部署 Gitea/Forgejo Plausible CE 启用城市级地理位置识别 Blinko 开源 AI 知识库 Docker 部署指南 Netcup 免税账号注册及购买服务器全记录 新到手的 Linux 服务器,我这样设置 Docker 自托管 Cloudreve Pro 私有网盘服务 GiffGaff SIM 卡使用体验和注意事项 在 n8n 中使用 Ollama 的本地 LLM 模型 简体中文互联网在变得糟糕吗? 如何低成本申请 S/MIME 证书用于个人邮件服务 迁移到 NameCrane Mail 域名邮箱套件服务 香烟与咖啡 联系我 Now 近况 在 Windows 中使用 Yubikey 的 OpenPGP 应用 Windows 11 IoT LTSC 养老指南 巧用浏览器生成 HAR 文件批量下载所需网页资源 聊聊静态网站的评论系统 在 Hugo 站点中优雅地展示实况照片 使用 snac 部署一个简洁优雅的 ActivityPub 实例 当 25 岁来临的时候 断网的这两年 从 Bitwarden/Vaultwarden 到 KeePassXC 更新日志 也许是 Android 上最好的本地播放器 - 椒盐音乐 2023 半年小结 写一个好玩的 Hugo 货币汇率转换短代码 写一个 Hugo 短代码将不同类别的总字数统计转换为书名显示 海信 Touch Lite 墨水屏音乐阅读器的简单体验 从 Debian 11(bullseye) 无感升级到 Debian 12(bookworm) 使用 GitHub Actions 自动提交 URL 到 IndexNow 批量重写 Git 历史提交记录 使用 cgit 托管小小的也很可爱的 Git 服务器 在 Debian 上使用 stagit 和 Nginx 的简单 Git 服务器 托管简单 Git 服务器的一些尝试 使用 stagit 自托管纯静态 Git 存储库 使用 Docker 快速部署单人 Fediverse 实例 我是如何建立自己的个人博客的? 简单、隐私友好的谷歌分析替代品,Plausible 自托管部署指南 服务器使用 Cloudflare CDN 的最佳实践 甲骨文服务器分配并启用 IPv6 地址 Git 版本控制学习笔记(二) Cloudflare WAF 防护策略简易指南 Mastodon 服务器批量导入自定义表情 低成本自托管 Mastodon 实例简明指南 自托管服务指南──有了服务器可以做什么? 使用 Miniflux + RSSHub 打造个人 RSS 阅读器 构建支持多种 CPU 架构的 Docker 镜像 一个可自托管的静态短链接应用 别了,二〇二二! Duplicacy CLI 进阶使用指南 Duplicacy CLI 备份工具的基本使用 初探 Cloudflare 零信任 - 通过 Docker 部署 Bitwarden 密码管理器 Yubikey 转运日志及上手指南 初探 Cloudflare 零信任 — 通过 Cloudflare Tunnel 搭建 SSH 聊天室 初探 Cloudflare 零信任 - 通过 Cloudflare Tunnel 访问服务 某科学的 PGP 算号指南 版权声明 2022 年的 MacBook Air M1 使用体验及应用推荐 使用 Vercel 和 Supabase 自托管 Cusdis 评论系统 为什么你的搜索引擎不好用? 个人在 Windows 上常用软件清单 macOS 基础开发环境设置指南 在 macOS 上使用 Canokey 的 OpenPGP 应用 提取 Windows UWP 应用商店的安装包以供离线安装 Git 版本控制学习笔记(一) 使用 Notion 管理 GitHub Star 项目 使用 Vercel 免费部署 Giscus 评论系统 将博客评论从 Utterances 无缝迁移到 Giscus
C 学习笔记(二)
2019-11-09 · via Dejavu's Blog

编程机制

C 具有良好的可移植性,可以在很多操作系统下使用,在 C 程序的编写过程中,源代码一般以文本文件存储在以 .c 为文件拓展名的文件中。

C 的基本机制是 源代码文件 编译为可执行文件(包括可直接运行的机器语言代码)的过程,典型步骤是 编译链接运行,其中编译器把源代码转换成中间代码,链接器把中间代码和其他代码合并,生成可执行文件,这种策略使得 C 可以方便地对程序进行 模块化 设计。

模块化 的好处是:对于一个项目,可以先独立编译单独的模块,然后再用 链接器 合并已编译的模块,如果需要更改某个模块,不必因此重新编译其他模块。此外 链接器 还将您编写的程序和预编译的库代码进行合并。

C 的编译

Unix C

C 最开始为 Unix 系统开发而生,下面是 C 在 Unix 下的编译过程

Unix C

GNU&&LLVM

GNU 编译器集合也就是 GCC,其中包含 C 的编译器 GCC C,在 UNIX、Linux、Windows 等操作系统下都可以使用,它跟随 C 标准的改动,对新标准的支持比较好。一般地,使用 gcc 命令调用 GCC C 编译器,一些使用 gcc 的系统又使用 cc 作为 gcc 的别名。

LLVM 项目是 cc 的另一个替代品,它使用 Clang 编译器处理 C,Clang 对最新的 C 标准的支持也很好。

对于 gcc 和 clang:

gcc-v # 查看使用的编译器及其版本

gcc [options][filenames] # GCC 最基本的用法:(options 是所需参数、filenames 是文件名)

gcc -std=c99 inform.c # gcc/clang 选择调用不同 C 标调,比如调用 C99 标准
gcc -std=clx inform.c # 调用 C11 之前草案标准
gcc -std=c11 inform.c # 调用 C11 标准

令牌(Tokens)

令牌 是程序的最小元素,令牌的类型有:关键字标识符常量字符串运算符

关键字(Keywords)

关键字是 C 中预定义的保留字,每个保留字均有特定功能或与特定功能与之关联,它们对编译器有特殊意义

C 中共有 32 个关键字

autodoubleintstruct
breakelselongswitch
caseenumregistertypedef
charexternreturnunion
continueforsignedvoid
doifstaticwhile
defaultgotosizeofvolatile
constfloatshortunsigned

printf 函数

printf() 函数是刚开始使用比较多的一个函数,它负责打印输出信息到屏幕。printf 由单词 printformat 组成

#include <stdio.h>

int main()
{
    printf("%i \n", 6);
    return 0;
}
# 输出结果是 6

printf() 可以在输出文本中指定 占位符,上面的 %i 就是一个 占位符;占位符在一个半角逗号后面的就是实际替换占位符的值,比如上面用 6 替换 %i,根据替换值的类型,需要用不同的占位符:

  • %a:浮点数
  • %A:浮点数
  • %c:字符
  • %d:十进制整数
  • %e:使用科学计数法的浮点数,指数部分的 e 为小写
  • %E:使用科学计数法的浮点数,指数部分的 E 为大写
  • %i:整数,基本等同于 %d
  • %f:小数(包含 float 类型和 double 类型)
  • %g:6 个有效数字的浮点数。整数部分一旦超过 6 位,就会自动转为科学计数法,指数部分的 e 为小写
  • %G:等同于 %g,唯一的区别是指数部分的 E 为大写
  • %hd:十进制 short int 类型
  • %ho:八进制 short int 类型
  • %hx:十六进制 short int 类型
  • %huunsigned short int 类型
  • %ld:十进制 long int 类型
  • %lo:八进制 long int 类型
  • %lx:十六进制 long int 类型
  • %luunsigned long int 类型
  • %lld:十进制 long long int 类型
  • %llo:八进制 long long int 类型
  • %llx:十六进制 long long int 类型
  • %lluunsigned long long int 类型
  • %Le:科学计数法表示的 long double 类型浮点数
  • %Lflong double 类型浮点数
  • %n:已输出的字符串数量。该占位符本身不输出,只将值存储在指定变量之中
  • %o:八进制整数
  • %p:指针
  • %s:字符串
  • %u:无符号整数 unsigned int
  • %x:十六进制整数
  • %zdsize_t 类型
  • %%:输出一个 %

标识符(Identifiers)

C 中 标识符 用于命名 变量函数数组 等,可以由用户定义这些标识符,由 字母数字下划线 组成

标识符命名规则

  • 必须以字母或者下划线开头
  • 只能使用字母、数字和下划线,不允许使用其他特殊字符、标点符号
  • 不能包含空格
  • 不能是 C 的关键字(Keywords)
  • 最长为 31 个字符

常量(Constant)

常量是固定值,就像是常规的变量,在定义后不能进行修改并且在程序执行期间不会改变。这些固定的值,又叫做 字面量

常量可以是任何 基本数据类型,比如:

  • 整数常量
  • 浮点常量
  • 字符常量
  • 字符串字面值
  • 枚举常量

下面是常量的数据类型的一些实例

整数常量

整数常量 可以是八进制、十进制、十六进制的常量

指定基数:前缀0x/0X 表示十六进制,前缀 0 表示八进制,十进制一般不带前缀

后缀是 U 和 L 的组合:U 是无符号整数(unsigned),L 是长整数(long)UL 可以是大写/小写的任意顺序,比如:

85         /* 十进制 */
0213       /* 八进制 */
0x4b       /* 十六进制 */
30         /* 整数 */
30u        /* 无符号整数 */
30l        /* 长整数 */
30ul       /* 无符号长整数 */
212        /* 合法的 */
215u       /* 合法的 */
0xFeeL     /* 合法的 */
078        /* 非法的:8 不是八进制的数字 */
032UU      /* 非法的:不能重复后缀 */

浮点变量

浮点常量 由整数部分、小数点、小数部分和指数部分组成,使用小数或者指数形式来表示

  • 小数形式表示:必须包含小数点、指数或者同时包含两者;
  • 指数形式表示:必须包含整数部分、小数部分,或同时包含两者,带符号的指数使用 e 或者 E 引入,比如:
3.14159       /* 合法的 */
314159E-5L    /* 合法的 */
510E          /* 非法的:不完整的指数 */
210f          /* 非法的:没有小数或指数 */
.e55          /* 非法的:缺少整数或分数 */

字符常量

字符常量 使用 '' 包起来,比如: s 可存储在 char 类型的简单变量中,字符常量可以是一个普通字符( 如 s )、一个转义序列( 如 \n )或者一个通用字符( 如 \u02C0

在 C 中,一些特定的字符前面有反斜杠 \\ 时,它们就会变成有特殊含义的 转义序列 (Escape Sequence),转义序列用于代表难以表示或无法输入的字符,像 t 代表 Tab 键,\b 代表 Backspace

转义序列含义
\\ 字符
\'' 字符
\"" 字符
?? 字符
\a警报铃声
\b退格键
\f换页符
\n换行符
\r回车
\t水平制表符
\v垂直制表符
\ooo一到三位的八进制数
\xhh . . .一个或多个数字的十六进制数

字符串常量

字符串字面值或常量是括在双引号 "" 中的,一个字符串包含类似于字符常量的字符:普通的字符、转义序列和通用的字符

可以使用空格做分隔符,把一个很长的字符串常量进行分行,并不影响编译程序识别

# 第一种
"hello,world"
# 第二种    
"hello,\
dear"
# 第三种
"hello, " "w" "orld"

定义常量

在 C 中,有两种简单的定义常量的方式( #defineconst 关键字),定义常量时经常以 大写英文字母 形式

下面是两个实例:

  1. 使用 #define 预处理
#include <stdio.h>

#define LENGTH 10   
#define WIDTH  5
#define NEWLINE '\n'

int main()
{

   int area;  
  
   area = LENGTH * WIDTH;
   printf("value of area : %d", area);
   printf("%c", NEWLINE);

   return 0;
}
  1. 使用 const 关键字
#include <stdio.h>

int main()
{
   const int  LENGTH = 10;
   const int  WIDTH  = 5;
   const char NEWLINE = '\n';
   int area;  
   
   area = LENGTH * WIDTH;
   printf("value of area : %d", area);
   printf("%c", NEWLINE);

   return 0;
}

两个例子的结果一样

value of area : 50

字符串(String)

字符串是以 null 空字符 \\0 结尾的一维字符数组,此空字符表示字符串已结束,字符串始终用双引号 "" 引起来,C 编译器会在初始化数组时,自动把 \0 放在字符串的末尾,比如:

char greeting[6] = {'H', 'e', 'l', 'l', 'o', '\0'};

也可以写成下面的:(数组初始化规则)

char greeting[] = "Hello";

上面的字符串在内存中可以如下图所示:

内存位置

C 中操作字符串的函数:

函数目的
strcpy(s1, s2);复制字符串 s2 到字符串 s1
strcat(s1, s2);连接字符串 s2 到字符串 s1 的末尾
strlen(s1);返回字符串 s1 的长度
strcmp(s1, s2);s1 和 s2 相同,返回 0;s1<s2 返回小于 0;s1>s2 返回大于 0
strchr(s1, ch);返回一个指针,指向字符串 s1 中字符 ch 第一次出现的位置
strstr(s1, s2);返回一个指针,指向字符串 s1 中字符串 s2 第一次出现的位置

比如:

#include <stdio.h>
#include <string.h>
 
int main ()
{
   char str1[12] = "Hello";
   char str2[12] = "World";
   char str3[12];
   int  len ;
 
   /* 复制 str1 到 str3 */
   strcpy(str3, str1);
   printf("strcpy( str3, str1):%s\n", str3 );
 
   /* 连接 str1 和 str2 */
   strcat( str1, str2);
   printf("strcat( str1, str2):%s\n", str1 );
 
   /* 连接后,str1 的总长度 */
   len = strlen(str1);
   printf("strlen(str1):%d\n", len );
 
   return 0;
}

则会有

strcpy( str3, str1 ):Hello
strcat( str1, str2 ):HelloWorld
strlen( str1 ):10

运算符(Operators)

运算符 是一种告诉编译器执行特定数学或逻辑操作的符号,C 中提供了以下类型的运算符:

  • 算数运算符
  • 关系运算符
  • 逻辑运算符
  • 位运算符
  • 赋值运算符
  • 杂项运算符

算数运算符

给定变量 AB,在 C 中常用的算数运算符罗列如下:

运算符描述结果
+两操作数相加A+B
-前一个操作数减去后一个操作数A-B
*两操作数相乘A*B
/两操作数相除(分子/分母)A/B
%取模,整除后的余数A/B 后的余数
++自增运算符,整数值 +1A+1B+1
--自减运算符,整数值 -1A-1B-1

特别地,对于 a++和++a,相同点都是给 a 的值+1,不同点是 a++ 是先赋值再+1,而 ++a 则是先 +1 再赋值

#include <stdio.h>
int main()
{
	int c;
	int a = 10;
	c = a++;
    
	printf( "先赋值后运算:\n" );
	printf( "Line 1 - c 的值是 %d\n", c );
	printf( "Line 2 - a 的值是 %d\n", a );
    
	a = 10;
	c = a--;
    
	printf( "Line 3 - c 的值是 %d\n", c );
	printf( "Line 4 - a 的值是 %d\n", a );
	printf( "先运算后赋值:\n" );
    
	a = 10;
	c = ++a;
    
	printf( "Line 5 - c 的值是 %d\n", c );
	printf( "Line 6 - a 的值是 %d\n", a );
    
	a = 10;
	c = --a;
    
	printf( "Line 7 - c 的值是 %d\n", c );
	printf( "Line 8 - a 的值是 %d\n", a );
}

则会有:

先赋值后运算:
Line 1 - c 的值是 10
Line 2 - a 的值是 11
Line 3 - c 的值是 10
Line 4 - a 的值是 9
先运算后赋值:
Line 5 - c 的值是 11
Line 6 - a 的值是 11
Line 7 - c 的值是 9
Line 8 - a 的值是 9

关系运算符

给定变量 AB,在 C 中常见的关系运算符有:

运算符描述实例
==检查两个操作数是否相等,相等则条件为真A==B
!=检查两个操作数是否相等,不相等则条件为真A!=B
>检查左操作数是否大于右操作数,是则条件为真A>B
<检查左操作数是否小于右操作数,是则条件为真A<B
>=检查左操作数是否大于或等于右操作数,是则条件为真A>=B
<=检查左操作数是否小于或等于右操作数,是则条件为真A<=B

比如:

#include <stdio.h>

int main(){
    int a = 21;
    int b = 10;
    int c;

    if( a == b ){
        printf("Line 1 - a 等于 b\n");
    }else{
        printf("Line 1 - a 不等于 b\n");
    }

    if( a < b){
        printf("Line 2 - a 小于 b\n");
    }else{
        printf("Line 2 - a 不小于 b\n");
    }

    if( a > b ){
        printf("Line 3 - a 小大于 b\n");
    }else{
        printf("Line 3 - a 小大于 b\n");
    }

    // 改变 a 和 b 的值

    a = 5;
    b = 20;

    if( a <= b){
        printf("Line 4 - a 小于等于 b\n");
    }

    if( b >= a){
        printf("Line 5 - b 大于或等于 b\n")
    }
}

输出结果

Line 1 - a 不等于 b
Line 2 - a 不小于 b
Line 3 - a 大于 b
Line 4 - a 小于或等于 b
Line 5 - b 大于或等于 b

逻辑运算符

给定变量AB,在 C 中常见的逻辑运算符有:

运算符描述实例
&&逻辑与运算符A&&B
||逻辑或运算符A||B
!逻辑非运算符!(A&&B)

位运算符

所谓 位运算,就是对一个比特(Bit)位进行操作。比特(Bit)是一个电子元器件,8 个比特构成一个字节(Byte),它已经是粒度最小的可操作单元了,C 中提供了 6 种位运算符:

运算符说明
&按位与运算
|按位或运算
^按位异或运算
~取反运算
<<左移运算
>>右移运算

赋值运算符

给定变量 ABC,在 C 语言中支持的赋值运算符:

运算符描述例子
=简单赋值运算符C=A+B 即把 A+B 的值赋给 C
+=加且赋值运算符C+=AC=C+A
-=减且赋值运算符C-=AC=C-A
*=乘且赋值运算符C*=AC=C*A
/=除且赋值运算符C/=AC=C/A
%=求模且赋值运算符C%=AC=C%A
<<=左移且赋值运算符C<<=2C=C<<2
>>=右移且赋值运算符C>>=2C=C>>2
&=按位与且赋值运算符C&=2C=C&2
^=按位异或且赋值运算符C^=2C=C^2
|=按位或且赋值运算符C|=2C=C|2

杂项运算符

运算符描述例子
sizeof()返回变量的大小sizeof(a) 返回 4,其中 a 是整数
&返回变量的地址&a; 返回变量 a 在内存中的实际地址
*指向一个变量*a; 指向一个变量(指针)
?:条件表达式如果条件为真 ? 则值为 X : 否则值为 Y

运算符优先级

类别运算符结合性
后缀() [] -> . ++ - -从左到右
一元+ - ! ~ ++ - - (type)* & sizeof从右到左
乘除* / %从左到右
加减+ -从左到右
移位<< >>从左到右
关系< <= > >=从左到右
相等== !=从左到右
位与 AND&从左到右
位异或 XOR^从左到右
位或 OR|从左到右
逻辑与 AND&&从左到右
逻辑或 OR||从左到右
条件?:从右到左
赋值= += -= *= /= %=>>= <<= &= ^= |=从右到左
逗号,从左到右

参考资料