当前移动开发的趋势已经势不可挡,这个系列希望浅谈一下个人对IOS开发的一些见解,这个IOS系列计划从几个角度去说IOS开发:
这么看下去还有大量的内容需要持续补充,但是今天我们从最基础的C语言开始,C语言部分我将分成几个章节去说,今天我们简单看一下C的一些基础知识,更高级的内容我将放到后面的文章中。
今天基础知识分为以下几点内容(注意:循环、条件语句在此不再赘述):
既然是IOS开发系列首先看一下在Mac OS X中的C的运行
打开Xcode
选择命令行程序
填写项目名称并选择使用C语言
选择保存目录
自动生成如下代码
OK,在Xcode上我们编写自己的程序如下
// // main.c // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // #include <stdio.h> void showMessage(){ printf("Hello,World!\n"); } int main(int argc, const char * argv[]) { showMessage(); return 0; }
在上面的程序中我们需要解释几点:
注意:#include 包含文件时有两种方式:使用<>和””。区别就是<>包含只会查找编译器库函数文件,因此适用于包含库函数;而“”包含则首先查找程序当前目录,如果没有找到则查找库函数路径,因此适用于自定义文件;
C语言的运行分为两大步:编译和链接
在大型项目开发中程序中所有的代码都写到一个文件中是不现实的,我们通常将一个子操作分为两个文件:.c文件和.h文件。在.c文件中实现对应的函数,在.h中进行函数声明,这样只要在主函数上方包含对应的头文件就可以将子操作分离出来而且不用考虑顺序问题。例如改写“Hello World”的例子(注意message对应的.c和.h文件名完全可以不相同,但是出于规范的目的我们还是取相同的文件名):
message.h
// // Message.h // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // void showMessage();
message.c
// // Message.c // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // #include <stdio.h> void showMessage(){ printf("Hello,World!\n"); }
main.c
// // main.c // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // #include <stdio.h> #include "Message.h" int main(int argc, const char * argv[]) { showMessage(); return 0; }
可以发现程序仍然可以正常运行,但是我们思考一个问题:如果我们不分成两个文件,直接在主函数文件中包含message.c是否也可以正常运行呢?答案是否定的,原因是由于编译生成的两个文件main.obj和 message.obj在链接时会发现main.obj中已经有message.obj中定义的showMessage函数,抛出“标示符重复”的错误。
从上图我们可以清晰的看到C语言的数据类型结构,当然对于这些类型我们还有一些类型修饰符(或叫限定符)
对于类型修饰符需要做如下解释
当然有时候我们必须清楚每个类型占用的字节,下表列出常用数据类型占用的存储空间
注意:char类型是最小的数据类型单位,在任何类型的编译器下都是占用1个字节,char类型的变量赋值可以直接赋值等于某个字符也可以赋值为整数(对应的ASCII值);可以使用两个long来修饰一个整形(就是经常使用的8字节的整形long long),但是两个long不能修饰double而且也不存在两个short,否则编译警告;一个浮点型常量如果后面加上f编译器认为它是float类型,否则认为double类型,例如10.0是double类型,10.0f是float类型。
C语言中有34中运算符,同C#、Java等语言没有太大的区别,这里指列出一些注意事项
针对上面几点看以下例子
// // main.c // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // #include <stdio.h> int main(int argc, const char * argv[]) { int a=2>1,b=2<1,c=99,d=0; int f=0,g=0,h=0,e=(f=3,g=4,h=5); a>0;//没有保存运算结果 printf("%d,%d\n",a,b);//结果:1,0 if(c){//可以通过 printf("true.\n"); } if(d){//无法通过 printf("false\n"); } printf("%d\n",e);//结果:5 return 0; }
printf()函数用于向标准输出设备输出数据,配合格式符可以完成强大的输出功能,上面的例子中我们已经使用了这个函数。
通常我们的输出不是固定内容而是包含某些变量,此时需要用到格式符,常用格式符如下
对于格式符的输出宽度和浮点数的小数位我们可以进行精确的控制
// // main.c // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // #include <stdio.h> int main(int argc, const char * argv[]) { int a=16; float b=79.3f; printf("[a=%4d]\n",a); printf("[a=%-4d]\n",a); printf("[b=%10f]\n",b); printf("[b=%.2f]\n",b); printf("[b=%4.2f]\n",b); return 0; }
运行结果如下
从运行结果我们不难发现格式符%前的正数可以设置前端补齐,负数设置后端对齐,如果数据的总长度超过设置的修饰长度,则按照实际长度显示;小数点后的整数用于控制小数点后保留小数位的长度。
scanf()函数用于从标准输入设备接收输入数据
// // main.c // C语言基础 // // Created by Kenshin Cui on 14-7-12. // Copyright (c) 2014年 cmjstudio. All rights reserved. // #include <stdio.h> int main(int argc, const char * argv[]) { int a,b,c; scanf("%d,%d,%d",&a,&b,&c);//此时需要输入:1,2,3 然后回车 printf("a=%d,b=%d,c=%d\n",a,b,c); return 0; }
对于scanf()函数我们需求强调几点
//C语言中的数据类型
//整型
//64bits
//char short int long long long
// 1 2 4 8 8
//32bits
// 1 2 4 4 8
//sizeof运算符
//求变量或者常量占用内存空间大小
//int main(int argc, const char *argv[])
//{
// printf("char = %lu\n",sizeof(char));//unsigned
// printf("short = %lu\n",sizeof(short));//%u %lu打印无符号整形数据的占位符
// printf("int = %lu\n", sizeof(int));
// printf("long = %lu\n",sizeof(long));
// printf("long long = %lu\n",sizeof(long long));
//
// return 0;
//}
//char
//正数 : 0000 0000(补码) ~ 0111 1111(补码) 0 ~ 127
//负数 : 1000 0001(补码) ~ 1111 1111(补码) -127 ~ -1
//1000 0000(补码) = -128
//1 即表示符号位 又表示绝对值大小
//short 1000 0000 0000 0000 ~ 0111 1111 1111 1111
/*int main(int argc, const char *argv[])
{
// char a,b;
// a= 0x80;
// b= 0x7f;
// printf("%d ~~ %d\n",a,b);
//
// short c,d;
// c=0x8000;
// d =0x7fff;
// printf("%d ~~ %d\n", c,d);
//
// int e,f;
// e = 0x80000000;
// f = 0x7fffffff;
// printf("%d ~~ %d\n",e,f);
long l1,l2;
l1 = 0x8000000000000000;
l2 = 0x7fffffffffffffff;
printf("%ld ~~ %ld\n",l1,l2);
return 0;
}*/
//char
//-128 ~ 127(-2^7 ~ 2^7-1)
//-32768 ~~ 32767(-2^15 ~ 2^15-1)
//int
//-2147483648 ~~ 2147483647 (-2^31 ~ 2^31-1)
//long
//-9223372036854775808 ~~ 9223372036854775807 (-2^63 ~ 2^63-1)
//long long
//-9223372036854775808 ~~ 9223372036854775807
//unsigned
//无符号
//unsigned char
//0 ~ 255
//0 ~ 2^8-1
//unsigned short
//0 ~ 2^16-1
//unsigned int
//0 ~ 2^32-1
//unsigned long
//0 ~ 2^64-1
//unsigned long long
//0 ~ 2^64-1
//实型数据
// float(单精度) double(双精度) long double(长双精度)
// 4 8 16
// 3.14(常用)
// 3.5 * 10^3(数学 科学计数法)
// 3.7e3;(指数法)
//5.6e-6 ==> 0.56e-5
//
//浮点类型数据存储分为三部分
//1.符号位 0
//2.指数位 -5
//3.数据位 56
//float (6-7)
//double (15 ~16)