C语言 - 最初的源点

从Hello World说起

本文为这么些年大家追过的言语之C语言篇。第三个C语言程序是Hello
World, 创笔者Brian W. Kernighan, The C Programming
Language

的小编之一。同时, 那是本文推荐的首先本关于C语言的书籍,
它大概涵盖C中兼有的底蕴语法和注意事项. 以往,
请随小编一起重温那段经典代码.
#include <stdio.h>

int main(int argc, char* argv[]){
  printf("Hello World\n");
  return 0;
}

对此逐个C程序, 系统运作的入口就是main. 那么,
从系统接口传入的参数自然要传至其参数int argc, char* argv[]中. 其中,
argc意味着参数个数, argv以字符指针数组的样式保留各种参数.
为证实真实传参情状, 编写如下程序test.c以输出argcargv:
#include <stdio.h>

int main(int argc, char* argv[]){
  int i;
  printf("%d argument(s):", argc);
  for(i=0; i<argc; ++i) printf(" %s", argv[i]);
  return 0;
}

继而, 大家编译test.c为可执行文件test, 并带参运维它:

yogy@Kali:~/PL$ gcc -o test test.c
yogy@Kali:~/PL$ ./test foo bar baz qux
5 argument(s): ./test foo bar baz qux

结果评释, 系统将收受的多少个字符串都当做参数传入了main函数, 包涵./test.

面向进度的C

有人说, C语言之于编程语言, 好似内功心法之于武学. 在此,
大家只谈对私家的影响, 不论江湖地位. C语言作为国内大学周边入门语言
(国外多为Python和Java), 也是本身上学的首先门编程语言. 当年采取的书是
Programming in
C
,
对于新人还是着力推介它, 通俗易懂, 性价比高. 作为一门面向经过的言语,
C对自身那四年的编程思路影响太深, 以至于无论选择C++依旧Python,
普遍服从了C的风格, 失去了面向对象的特征. 当然,
那里面富含一向学习算法的关系. 小编始终认为,
算法是一种是面向进程的逻辑思考方法, 使用封装一定水准上捐躯了它的频率.
显而易见, 想入门C, 踏实看书, 学会像程序一样思考, 思路比语法紧要. 另,
C进阶书籍推荐:

后两本只是为着让你不会深感无趣, 同时一定水平满意IT面试的须要,
并非系统算农学习.

黑魔法

1. Hello World

#define _________ } 
#define ________ putchar 
#define _______ main 
#define _(a) ________(a); 
#define ______ _______(){ 
#define __ ______ _(0x48)_(0x65)_(0x6C)_(0x6C) 
#define ___ _(0x6F)_(0x2C)_(0x20)_(0x77)_(0x6F) 
#define ____ _(0x72)_(0x6C)_(0x64)_(0x21) 
#define _____ __ ___ ____ _________ 
#include<stdio.h> 
_____

地方这段Hello World代码, 使用预处理#define的迭代操作,
结合了十六进制的ASICC码.
#include <stdio.h>
main(){
int
i,n[]={(((1<<1)<<(1<<1)<<(1<<1)<<(1<<(1>>1)))+((1<<1)<<(1<<1))),
(((1<<1)<<(1<<1)<<(1<<1)<<(1<<1))-((1<<1)<<(1<<1)<<(1<<1))+((1<<1)<<(1<<(1>>1)))+(1<<(1>>1))),(((1<<1)<<(1<<1)<<(1<<1)<<
(1<<1))-((1<<1)<<(1<<1)<<(1<<(1>>1)))-
((1<<1)<<(1<<(1>>1)))),(((1<<1)<<(1<<1)<<(1
<<1)<<(1<<1))-((1<<1)<<(1<<1)<<(1<<(1>>1)))-((1<<1)<<(1<<(1>>1)))),(((1<<1)<<
(1<<1)<<(1<<1)<<(1<<1))-((1<<1)<<(1<<1)<<(
1<<(1>>1)))-(1<<(1>>1))),(((1<<1)<<(1<<1
)<<(1<<1))+((1<<1)<<(1<<1)<<(1<<(1>>1)))
-((1<<1)<<(1<<(1>>1)))),((1<<1)<<
(1<<1)<<(1<<1)),(((1<<1)<<(1<<1)<<(1<<1)<<(1<<1))-((1<<1)<<(1<<1))-(1<<(1>>1))),(((1<<
1)<<(1<<1)<<(1<<1)<<(1<<1))-((1<<1)<<
(1<<1)<<(1<<(1>>1)))-(1<<(1>>1))),
(((1<<1)<<(1<<1)<<(1<<1)<<(1<<1))-
((1<<1)<<(1<<1)<<(1<<(1>>1)))+(1<<1)),
(((1<<1)<<(1<<1)<<(1<<1)<<
(1<<1))-((1<<1)<<(1<<1)<<(1<<(1>>1)))-((1<<1)<<(1<<(1>>1)))),
(((1<<1)<<(1<<1)<<(1<<1)<<(1<<1))-((1<<1)<<(1<<1)<<(1<<1))+((1<<1)<<(1<<(1>>1)))),
(((1<<1)<<(1<<1)<<(1<<1))+(1<<(1>>1))),(((1<<1)<<(1<<1))+((1<<1)<<(1<<(1>>1)))

  • (1<<(1>>1)))};
    for(i=(1>>1);i<=((1<<1)<<(1<<1))+((1<<1)<<(1<<(1>>1)));++i)
    printf(“%c”,n[i]);
    }
    移动操作<<>>是很棒的操作, 相比于乘除2的幂, 功能高很多. 那段Hello
    World, 凑各个壹回幂也是微醉, 小编不详. 主要提醒:
    移位操作符优先级低于+-, 请时刻牢记是不是需要括号.

2. ++–

先看stackoverflow上的2个座谈What is the name of the “–>”
operator?
.
#include <stdio.h>
int main() {
int x = 10;
while (x –> 0) {
printf(“%d “, x);
}
}
上述代码输出结果为9 8 7 6 5 4 3 2 1 0, 提问者对第陆行中的-->代表质疑,
是或不是它应精晓为”趋向于”. 难点自个儿很好通晓, x --> 0有道是作为(x--) > 0,
即先与0比较再自减.

自加++和自减--最经典的座谈就是其职责放置变量的前或后. 接下去,
让我们从汇编角度做出分析. 原分析我不详, 如不平常请联系小编.

i = i2++的反汇编代码分析:

  1. 将dword ptr [i2](即i2中的内存单元)中的数据拷贝到eax寄存器中
  2. 将eax寄存器中的数据(即i2)拷贝到dword ptr [i](即i的内存单元)中
  3. 将dword ptr [i2](即i2中的内存单元)中的数据拷贝到ecx寄存器中
  4. 将ecx寄存器中的内容自增1
  5. 将ecx寄存器中的内容(ecx加1后的数据)拷贝到dword ptr [i2]

i = ++i2的反汇编代码分析:

  1. 将dword ptr [i2](即i2中的内存单元)中的数据拷贝到eax寄存器中
  2. 将eax寄存器中的内容自增1
  3. 将eax寄存器中的内容(eax加1后的数目)拷贝到dword ptr
    [i2](即i2中的内存单元)
  4. 将dword ptr [i2](即i2中的内存单元)中的数据拷贝到ecx寄存器中
  5. 将ecx寄存器中的内容(ecx加1后的数码)拷贝到dword ptr
    [i](即i所在的内存单元)

由上述分析可见, 自加自减对内建数据类型的意况,功能没有区分。

另, 与自加自减功用相近, -~i表示i+1, ~-i表示i-1.
但出于它们是经过位运算办到的, 放在变量之后并未意义.

3. Quine

Quine以国学家威拉德 van Orman Quine命名,
表示一个足以输出他自个儿的一心源代码的主次,
详见Quine_(computing)).
上边, 请欣赏彩蛋:

上述代码应用C编译 (gcc -ansi) 均得到youmu, 而使用C++编译 (g++)
都输出yuyuko. 那种程序名为Polyglot,
详见Polyglot_(computing)).
上述彩蛋便是应用一定的3字符片段来区分C和C++.

结束语

“Hello, World!” 开始, 从 World:” Hello”! 走向另1个开头.

图片 1

C