c89、c99、c11

2024-06-24 01:28
文章标签 c99 c11 c89

本文主要是介绍c89、c99、c11,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C99 标准开始引入了 // 单行注释。在此之前,C语言只支持 /* ... */ 多行注释。

具体说明:

  • // 单行注释:在C99标准(ISO/IEC 9899:1999)引入之前,C语言中没有单行注释。C99标准借鉴了C++的注释风格,引入了这种单行注释。
  • /* ... */ 多行注释:自C语言的早期版本(包括C89和ANSI C)以来就已经存在。

示例:

在C99之前(例如,使用ANSI C标准的编译器):

int main() {/* 这是一个多行注释 */int a = 10;/* 这是一个多行注释 */return 0;
}

在C99及之后:

int main() {// 这是一个单行注释int a = 10;// 这是另一个单行注释return 0;
}

在现代C编译器中(C99及之后),可以混合使用这两种注释:

int main() {// 这是一个单行注释int a = 10; /* 这是一个多行注释 */return 0;
}

总结:

  • // 单行注释是从C99标准开始引入的。
  • /* ... */ 多行注释从C语言的早期版本(包括C89和ANSI C)就已经存在。

C89(也称为ANSI C)和C99是两个主要的C语言标准,它们之间有许多显著的区别。以下是一些关键区别及相应的示例:

1. 单行注释

  • C89:不支持 // 单行注释。
  • C99:支持 // 单行注释。

示例

// 这是一个C99支持的单行注释
int a = 10; // 这是一个单行注释

2. 变量声明

  • C89:变量必须在函数块的开头声明。
  • C99:变量可以在代码块中的任意位置声明。

示例

// C89
void example() {int a = 10;/* 必须在这里声明所有变量 */int b = 20;a = a + b;
}// C99
void example() {int a = 10;a = a + 10;int b = 20; // 可以在任意位置声明变量a = a + b;
}

3. 复合字面量

  • C89:不支持复合字面量。
  • C99:支持复合字面量。

示例

// C99
struct Point {int x, y;
};
struct Point p = (struct Point){.x = 1, .y = 2}; // 复合字面量

4. 变长数组

  • C89:不支持变长数组。
  • C99:支持变长数组。

示例

// C99
void example(int n) {int arr[n]; // 变长数组for (int i = 0; i < n; i++) {arr[i] = i;}
}

5. 内联函数

  • C89:不支持 inline 关键字。
  • C99:支持 inline 关键字。

示例

// C99
inline int add(int a, int b) {return a + b;
}

6. 数据类型扩展

  • C89:没有 long long 数据类型。
  • C99:引入了 long long 数据类型(至少64位)。

示例

// C99
long long bigNumber = 123456789012345LL;

7. __func__ 预定义标识符

  • C89:没有 __func__ 预定义标识符。
  • C99:引入了 __func__ 预定义标识符,用于获取当前函数的名称。

示例

// C99
#include <stdio.h>void example() {printf("Function name: %s\n", __func__);
}

8. 初始化增强

  • C89:初始化结构体和数组时必须按顺序进行。
  • C99:允许使用指定初始化器(designated initializers)。

示例

// C99
struct Point {int x, y;
};
struct Point p = {.y = 2, .x = 1}; // 指定初始化器

这些示例展示了C99对C89的多方面增强,使C语言变得更为灵活和功能强大。


C11 是 C 语言的一个标准,全称为 ISO/IEC 9899:2011,相对于 C99 进一步引入了一些新特性和改进。以下是 C11 相对于 C99 和 C89 的一些主要区别及相应的示例:

1. _Generic 选择表达式

  • C89/C99:不支持 _Generic 选择表达式。
  • C11:引入了 _Generic 选择表达式,用于泛型编程。

示例

// C11
#include <stdio.h>#define type_of(x) _Generic((x), \int: "int", \float: "float", \double: "double", \default: "other")int main() {int i = 0;float f = 0.0;printf("i is %s\n", type_of(i)); // 输出:i is intprintf("f is %s\n", type_of(f)); // 输出:f is floatreturn 0;
}

2. 匿名结构体和联合体

  • C89/C99:不支持匿名结构体和联合体。
  • C11:支持匿名结构体和联合体。

示例

// C11
struct {union {int i;float f;};
} u;int main() {u.i = 10;printf("%d\n", u.i); // 输出:10u.f = 5.5;printf("%f\n", u.f); // 输出:5.500000return 0;
}

3. 静态断言

  • C89/C99:不支持静态断言。
  • C11:引入了 _Static_assert 关键字,用于在编译时进行静态断言。

示例

// C11
#include <assert.h>_Static_assert(sizeof(int) == 4, "int size is not 4 bytes");int main() {return 0;
}

4. 线程支持

  • C89/C99:不提供标准化的线程支持。
  • C11:引入了 threads.h,提供对多线程的标准支持。

示例

// C11
#include <stdio.h>
#include <threads.h>int thread_func(void *arg) {printf("Hello from thread!\n");return 0;
}int main() {thrd_t t;thrd_create(&t, thread_func, NULL);thrd_join(t, NULL);return 0;
}

5. 对齐支持

  • C89/C99:不提供对齐支持。
  • C11:引入了 _Alignof_Alignas 关键字,用于指定和查询类型对齐要求。

示例

// C11
#include <stdio.h>
#include <stdalign.h>struct S {char c;_Alignas(16) int i;
};int main() {printf("Alignment of char: %zu\n", alignof(char)); // 输出:1printf("Alignment of int: %zu\n", alignof(int)); // 输出:4printf("Alignment of struct S: %zu\n", alignof(struct S)); // 输出:16return 0;
}

6. 预定义宏

  • C89/C99:不支持新的预定义宏。
  • C11:引入了新的预定义宏,例如 __STDC_VERSION__

示例

// C11
#include <stdio.h>int main() {#if __STDC_VERSION__ >= 201112Lprintf("C11 or later\n");#elseprintf("Before C11\n");#endifreturn 0;
}

7. 可选特性宏

  • C89/C99:不支持可选特性宏。
  • C11:引入了可选特性宏,例如 __STDC_NO_THREADS__

示例

// C11
#include <stdio.h>int main() {#ifdef __STDC_NO_THREADS__printf("Threads are not supported\n");#elseprintf("Threads are supported\n");#endifreturn 0;
}

8. 更严格的类型检查

  • C89/C99:类型检查不如 C11 严格。
  • C11:引入了更严格的类型检查和标准库改进。

C11 标准通过引入这些新特性和改进,使得 C 语言变得更加现代化、灵活和安全,特别是在泛型编程、静态断言、多线程和内存对齐等方面提供了显著的增强。

这篇关于c89、c99、c11的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1088811

相关文章

【转】keil(arm)中配置c99方法 及 C99特性

配置方法:option->c/c++->misc controls:--c99 附c99特性: 在ANSI的标准确立后,C语言的规范在一段时间内没有大的变动,然而C++在自己的标准化创建过程中继续发展壮大。《标准修正案一》在1994年为C语言创建了一个新标准,但是只修正了一些C89标准中的细节和增加更多更广的国际字符集支持。不过,这个标准引出了1999年ISO 9899:1999的发表。它通常

关于Xcode6编译变更 “Implicit declaration of function 'sysctl' is invalid in C99” 报错问题

之前代码在Xcode5.1上面跑的好好的,但是自从升级到6.0之后,就经常出现编译报错问题。后来查阅的相关资料,好像是Xcode为了兼容Swift语言,更换来编译系统(名字忘记了)。于是就报了一个C语言的C99编译错误,不说了,直接上代码。 一般出现该问题是因为通过C调用了unix/linux 底层接口,所以需要调整c语言的编译选项,设置方法见下图:(根据实际情况选择相应的编译选项

C语言C99标准下结构体赋值

12.25 今天又算学到了一种新的方式,不过感觉好像,这个东西我记过。 有关结构体的初始化,名字上叫做:标记结构初始化语法 struct file_operations scull_fops ={.owner = THIS_MODULE,.llseek = scull_llseek,.read = scull_read,.write = scull_write,.ioctl = scull_io

(C11) 泛型表达式

文章目录 ⭐语法⭐举例🚩判断对象类型🚩判断指针🚩函数重载🚩嵌套使用 END ⭐语法 Ref: 泛型选择 (C11 起) - cppreference.com 关键词: Genericdefault _Generic(控制表达式 , 关联列表) (C11 起) 关联列表 类型名:表达式 类型名:任何并非可变修改的完整对象类型(即既非 VLA 亦非指向 VL

Linux 之父终于被劝动:用了 30 年的 Linux 内核将升级至 C11

星标/置顶 公众号👇,硬核文章第一时间送达! 来源丨量子位 还在使用89年版C语言的Linux内核,现在终于要做出改变了。 今天,Linux开源社区宣布,未来会把内核C语言版本升级到C11,预计5.18版之后生效,也就是今年5月。 这个决定很突然,从发起问题到官方声明,不过才一个星期,要知道说服固执的Linux之父 Linus Torvalds可不是件容易的事。 事情的原因,说起来还有那么

C89与C99标准比较

1、增加restrict指针 C99中增加了公适用于指针的restrict类型修饰符,它是初始访问指针所指对象的惟一途径,因此只有借助restrict指针表达式才能访问对象。restrict指针指针主要用做函数变元,或者指向由malloc()函数所分配的内存变量。restrict数据类型不改变程序的语义。 如果某个函数定义了两个restrict指针变元,编译程序就假定它们指向两个不同的对象,

gcc/g++ 如何支持c11 / c++11标准编译

gcc/g++ 如何支持c11 / c++11标准编译 linux中的编译环境默认可能不支持C++11语法,如R“(abc)”原始字符串。 那么如果一定要编译呢? 通过命令man g++可以得知以下方法: g++ -o main main.cpp -std=c++11 (2017.3.1亲测 gcc version 5.4.0 20160609 (Ubuntu 5.4

如何确定gcc是否支持c11,c14,c17

实际工作中,可能会遇到c++的一些高级特性,例如std::invoke,此函数是c++17才引入的,如何判断当前的gcc是否支持c++17呢,这里提供两种办法。 1.根据gcc的版本号来推断 gcc --version,可以查看版本号,笔者的电脑,gcc的版本号是8.3.0,然后查看8.3.0是什么时候发布的,查出是2019年发布的,从而推断出是支持,但是不是十分肯定,所以接着看第二种方法 2.直

C语言 C99标准与C11标准部分更改函数对比

文章目录 1. scanf 函数与scanf_s 函数1.1 scanf函数1.2 scanf_s函数1.3 两者区别1.4 注意事项 2. gets 函数与gets_s 函数2.1 gets函数2.2 gets_s 函数2.3 两者区别2.4 注意事项 3. fopen 函数与fopen_s 函数3.1 fopen 函数3.2 fopen_s 函数3.3 两者区别

C11 列表初始化、左/右值引用、移动语义、可变参数模版

目录 一、统一的列表初始化 1、{}初始化 2、std::initializer_list 3、模拟实现vector花括号操作 二、声明 1、自动类型推断 - auto 2、类型推导关键字 - decltype 3、空指针常量 - nullptr 4、范围for循环 三、右值引用和移动语义 1、左值引用和右值引用 左值与左值引用 右值与右值引用 2、左值引用与右值引用比