C和指针学习
原则:此文尽量短小精悍,实用,需要扩充阅读都是以链接形式出现
1.编译
#cc program.c
生成a.out文件
这个名字是编译器默认的输出名。如果要修改可执行文件的名字可以加-o参数:gcc -o myexec main.c
这样就把main.c编译连接生成的可执行文件命名为myexec
gcc编译器的编译自定义格式
#cc-ohellohello.c
#gcc-ohellohello.c
使用gcc编译器就会为我们生成一个hello的可执行文件
扩充阅读:Linux编译器GCC的使用
http://blog.csdn.net/21aspnet/article/details/1534108
http://blog.csdn.net/21aspnet/article/details/167420
补充说明:如果你实在基础很差,那么需要在windows下借助VS2010这样的可视化工具来调试了,这样可以很清晰的看出内存中的指针
http://blog.csdn.net/21aspnet/article/details/6723758
2.执行
#./a.out
编译多个源文件
#cc a.c b.c c.c
3.产生目标文件
#cc -c program.c
产生program.o的目标文件以后供链接用
产生多个目标文件
#cc -c program.c a.c b.c
编译指定名称文件
#cc -o sea a.c
4.链接几个目标文件
#cc a.o b.o c.o
5.转义字符
\\代表\
\\\\代表\\
注意不是\\\代表\\
\'代表'
\"代表"
\n换行
\r回车
扩充阅读:C语言 格式控制符 和 转义字符
http://blog.csdn.net/21aspnet/article/details/1535459
6.注释
方法一 /* */
方法二 //
7.保留字
http://blog.csdn.net/21aspnet/article/details/1539252
8.整型
singed 有符号
unsigned是无符号的意思,也就是说如果你的编译系统给int分配的存储单元的长度是2个字节的话,有符号的int 取值范围是-32768(即2^15)——32767(即2^15-1),而无符号的unsigned int就是0-65535(2^16-1)
类型 | 最小范围 |
char | 0-127 |
signed char | -127-127 |
unsinged char | 0-255 |
int | -32767-32767 |
longint | -2147483647-2147483647 |
unsinged int | 0-65536 |
int a=8;
int a=012;//8进制是0开头
int a=0x0a;//16进制是0x开头
3种输出printf("%d",a);都是10
i++和++i
9.枚举类型
enum A{a,b,c,d};
10.浮点
float double long double
11.指针
指针只能指向地址!
指针的指针是为了取得“地址”,因为指针只是指向“主体”
int a;
int *b=&a;
int **c=&b;
a=1;
*b=2;
指针要对应级别,一级b对应一级&a,二级c对应二级&b。
如果传值原代码块无星
main()
{
swap(a,b,c)
}
swap(int a,int *b,int **c)
{
a=*b+**c;//这样不能改变值
*b=a+**c;//可以改变值
}
函数里要改变原值只有指针
printf("%d",*b);//输出2
printf("%d",a);//输出2
//指针和字符串
char *ch="abc";
char * cp=ch;
printf("%s",cp);//输出abc
//指针和字符
char ch=‘a’;
char * cp=&ch;
printf("%c",*cp);//输出a
//指针和数组
int a1[]={1,2,3,4,5};
printf("%d",&a1[2]);
int *b=&a1[2];
扩展阅读:
把指针说透
http://blog.csdn.net/21aspnet/article/details/317866
C指针本质
http://blog.csdn.net/21aspnet/article/details/1539652
还看不懂的建议去看:《彻底搞定C指针》
http://download.csdn.net/source/3491131
12.typedef
http://www.yesky.com/SoftChannel/72342371928899584/20040913/1853309.shtml
typedef为C语言的关键字,作用是为一种数据类型定义一个新名字。这里的数据类型包括内部数据类型(int,char等)和自定义的数据类型(struct等)。
在编程中使用typedef目的一般有两个,一个是给变量一个易记且意义明确的新名字,另一个是简化一些比较复杂的类型声明。
typedef与结构结合使用
typedef struct tagMyStruct
{
int iNum;
long lLength;
} MyStruct;
这语句实际上完成两个操作:
1) 定义一个新的结构类型
struct tagMyStruct
{
int iNum;
long lLength;
};
2) typedef为这个新的结构起了一个名字,叫MyStruct。
typedef struct tagMyStruct MyStruct;
因此,MyStruct实际上相当于struct tagMyStruct,我们可以使用MyStruct varName来定义变量。
typedef int k;
k k1=10;
此声明定义了一个 int 的同义字,名字为k。注意 typedef 并不创建新的类型。它仅仅为现有类型添加一个同义字。你可以在任何需要 int 的上下文中使用k
int a=0;
typedef int * p;
p b=&a;
不可以p *b 因为p=int *
13.const常量
http://www.yesky.com/SoftChannel/72342371928899584/20040916/1854856.shtml
const是一个C语言的关键字,它限定一个变量不允许被改变。使用const在一定程度上可以提高程序的健壮性,另外,在观看别人代码的时候,清晰理解const所起的作用,对理解对方的程序也有一些帮助。
const int a=15;//必须声明时赋值
int const a=15;//必须声明时赋值,const在前在后都可以
声明好以后再赋值会报错
const int a;
a=15
其次:在函数中声明为const的形参在函数被调用时会得到实参的值
int ax(const int aa)
{
int a=aa;
return a+1;
}
扩充阅读:http://blog.csdn.net/21aspnet/article/details/160197
14.define常量
#defined MMX 50
printf("%d",MMX);
15.链接属性
external外部
internal内部
none无
16.变量类型
取变量的值可以直接=变量
给变量赋值一定要&
auto 变量
是用堆栈(stack)方式占用储存器空间,因此,当执行此区段是,系统会立即为这个变量分配存储器空间,而程序执行完后,这个堆栈立即被系统收回.在大括号{}内声明.
static 变量
是C程序编译器以固定地址存放的变量,只要程序不结束,内存不被释放.
static int a=5;
扩充阅读:static
http://blog.csdn.net/21aspnet/article/details/1535573
external 变量
外部变量 定义在程序外部,所有的函数和程序段都可以使用.
extern可以置于变量或者函数前,以标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义。
另外,extern也可用来进行链接指定。
http://www.yesky.com/SoftChannel/72342371928899584/20040920/1855898.shtml
头文件f1.h
int a=5;//定义
主文件f2.c
#include "f.h" //引用,注意不能<"f.h">
extern int a;//声明
printf("%d",a);
static external 变量
静态外部变量和外部变量差别在于,外部变量生命可以同时给多个文件使用,而静态外部变量则只能给声明此变量的文件使用.
register 变量
寄存器变量,是由寄存器分配空间,访问速度比访问内存快,加快执行速度.寄存器大小有限.
注意:register int a=1;只能函数内 而不能函数外,由于寄存器的数量有限(不同的cpu寄存器数目不一),不能定义任意多个寄存器变量
extern 和static可以函数外
扩充阅读:寄存器register介绍
http://blog.csdn.net/21aspnet/article/details/257511
扩充阅读:变量属性
http://blog.csdn.net/21aspnet/article/details/2560072
17.语句
if...else...
while循环
i=0;a=0;
while(i<10)
{
a=a+i;
i++;
}
break:退出大循环
continue:退出本次循环
fo循环
for(int i=0;i<2;i++)
{
}
do....while
switch
switch(ch){
case "A":
i+1;
break;
case "B":
i+2;
break;
default:
i=0;
}
goto跳转
goto AA;
AA:if...else
18.操作符
+-*/%
19.位移和位操作符
<<左移
>>右移
&位与
|位或
^位非
扩展阅读:http://blog.csdn.net/21aspnet/article/details/160037
20.逻辑运算符
&& ||
21.条件运算符
a>5?b-6:c/2
a大于5就执行b-6否则执行c/2
22.++
K++ :k不变
++K :K+1
23.布尔值
C语言中没有布尔类型。任何一个整型的变量都可以充当布尔变量,用0表示False,其它数(默认为1)表示True。
a = 1;
if(a) {printf("a is T\n");}else{printf("a is F\n");}//输出T
a = 2;
if(a) {printf("a is T\n");}else{printf("a is F\n");}//输出T
a = -1;
if(a) {printf("a is T\n");}else{printf("a is F\n");}//输出T
a = 0;
if(a) {printf("a is T\n");}else{printf("a is F\n");}//输出F
if(!a) {printf("a is T\n");}else{printf("a is F\n");}//输出T
如果你想像Pascal一样使用true和false,那么你可以包含头文件stdbool.h。这样你可以定义变量为bool类型并赋值为true或false。例如:
#include "stdbool.h"bool a = true;
if (a) printf("a is true");
也可以使用
#define False 0
#defineTrue 1
24.数组
int a[]={1,2,3,4,5}
a[0]//用序号输出数组元素,默认下标从0开始
求数组长度:printf("%d\n",sizeof(a)/sizeof(a[0]));
int *ap=a+2;
ap[0];//输出3
int a[]={1,2,3,4,5,6};//自动计算数组长度
循环数组可以用2种方法
int i;
for(int i;i<6;i++)
{
a[i];这样输出
或者*(a+i);因为*(a)指向数组第一个元素的指针
}
指针访问数组
int *p=a;注意不是&a;
可以三种方法输出
p[i];方法一就是数组元素
*(p+i);方法二
for()
{
printf("%d",*p);//方法三
p++;
}
如果 int const p=a;
用上述代码linux下一样可以编译
指针的本质
int a[]={1,2,3,4,5};
int b[]={6,7,8,9,10};
int *p;p=b;
int i;
for(i=o;i<10;i++)
{
prinft("d%",*p);
p++;
}
输出678910 12345
如果改为p=a;
输出12345后面其实是其他未知地址
但是如果改为
p=a;
for(i=-5;i<5;i++)
{
prinft("d%",*p);
p++;
}
输出678910 12345
由此可以看到指针不是孤立的!
多维数组
int a[2][3]={1,2,3,4,5,6};
a[0][0]=1;
a[0][1]=2;
a[0][2]=3;
int a[2][3]={
{1,2,3},
{4,5,6}
};
多维数组只有第一维可以缺省,其余维都要显示写出
int a[][3]={1,2,3,4,5,6};
//声明指针数组,每个指针元素都初始化为指向各个不同的字符串常量
char const keyword[]={
"do";
"doo";
"doooo";
"do";
"doo";
"do";
"doooooo";
}
//声明矩阵,每一维长度都是9,不够用0补齐
char const keyword[][9]={
"do";
"doo";
"doooo";
"do";
"doo";
"do";
"doooooo";
}
扩展阅读:C语言中字符数组和字符串指针分析
http://blog.csdn.net/21aspnet/article/details/1539928
25.字符串
类型一:字符串
char day[15] = "abcdefghijklmn";//定义一
char day[] = "abcdefghijklmn";//定义一
char * str="abcdefghijklmn";//定义二
windows下VS2010调试窗口看区别
char str[20]="0123456789";//str是编译期大小已经固定的数组
int a=strlen(str); //a=10;//strlen()在运行起确定
int b=sizeof(str); //而b=20;//sizeof()在编译期确定
sizeof 运算符是用来求内存容量字节的大小的。而strlen是用来求字符串实际长度的。
下面2种声明完全一样
类型二:字符数组
char msg[]={'a','b','c'}
char msg[]={abc}
char *ans;
ans=strchr(day,'c');
printf("%s",ans);//输出从c开始的部分
printf("%d",*ans);//输出找到的指针
ans=strrchr(day,'c');
printf("%s",ans);//输出找到的指针
printf("%d",*ans);//输出找到的指针
char temp[100]="hello";
char *t="world";
strcat(temp,t);
输出 temp
要注意的是 第一个参数数组长度要=自身原长+合并字符长
类型三:字符串数组
char *a[4]={"this","is","a","test"};
char **p=a;
扩展阅读:C语言字符串处理库函数
http://blog.csdn.net/21aspnet/article/details/1539970
26.结构
注意结构一定要};结束
声明一
struct simple{
int a;
int b;
};
使用
struct simple s1;
s1.a=1;
声明二
typedef struct {
int a;
int b;
};simple
使用
simple s1;
s1.a=1;
注意:方法二比一少写一个struct
结构初始化
typedef struct {
int a;
int b;
};simple{1,2}
使用
simple.a;//输出1
指向结构的指针
struct data{
int a;
int b;
};data1{1,2}
struct data *p;
p=&data1;
(*p).a;//输出1
p->b;//输出2
扩展阅读:
结构的成员访问 http://blog.csdn.net/21aspnet/article/details/150259
27.内联函数
首先要声明
inline int 函数名(参数)
后面要有一个真实的函数体
inline int 函数名(参数)
{
}
只有三二行的代码,建议使用内联!这样运行速度会很快,因为加入了inline 可以减少了,跳入函数和跳出函数的步骤!
28.宏
#define SQUARE(x) x*x
程序中写SQUARE(3) 实际等于3*3
29.函数
int a=1;
int b=2;
ext1(a,b);
ext1(int x.int y)
{
int temp=x;
x=y;
y=temp;
prinft("x=%d,y=%d",x,y)
}
30.动态内存分配
动态内存分配是为了链表,否则只能用数组
使用malloc和free分配和释放
#include <malloc.h>//malloc
#include <stdlib.h>//exit’
int * p;
p=malloc(100);
if(p==NULL)
{
printf("Out of memory!\n");
exit(1);
}
实际中p=malloc(sizeof(int));获得足够1个整数的内存
实际中p=malloc(10*sizeof(int));获得足够10个整数的内存
扩展阅读:C语言内存动态分配
http://blog.csdn.net/21aspnet/article/details/146968
31.链表
//先定义一个结构体
typedef struct node{
int data;//数据域
struct node * next;//指针域 ,是struct node的成员,又指向struct node类型的的数据,这里存放下个结点的地址
}Node;
建立链表
//返回类型是之前定义的结构体
Node * creat()
{
//需要先定义3个节点
Node * head=NULL;//头结点
Node *p=NULL;//指向原链表的最后一个结点
Node *s=NULL;//指向新节点
head=(Node*)malloc(sizeof(Node));//动态分配内存空间
head->next=NULL;//开始头结点也就是尾结点
p=head;//p一开始指向头结点
//用while循环
int c=1;//为0是退出循环的条件
int d=0;
while(c)
{
//使用scanf接受用户数据数据
scanf("%d",&d);
if(d!=0)//继续输入
{
s==(Node*)malloc(sizeof(Node));//定义新节点s,和定义head一样
s->data=d;//给新节点赋值
p->next=s;//将第一个结点链接在第二个结点后面
p=s;//p继续指向当前节点
}
else
{
c=0;//用户输入0结束 输入
}
}
p->next=NULL;//结束输入时将最后结点定义为尾结点
return head;
}
//输出结点
output(Node * head)
{
//新定义一个指针用于遍历
//其实如果head->data会是0;单链表巧妙的从第二个结点开始,头结点没用,但是并不代表没用值!
Node * p=head->next;
while(p!=NULL)
{
printf("%d",p->data);
p=p->next;//注意while循环里一定要让P指针不断往下指向
}
}
//查找制定元素的顺序
searchdata(Node * head)
{
int t=0;
printf("请输入您要查找的值:\n");
scanf("%d",&t);
Node * p=head->next;
int k=0;
while(p!=NULL)
{
k++;
if(p->data==t)
{
printf("%d是第%d个元素:\n",t,k);
break;
}
p=p->next;
}
if(p==NULL)
{
printf("没找到");
}
}
//元素之间插入值
insertdata(Node * head)
{
Node *s=NULL;//指向新节点
s=(Node*)malloc(sizeof(Node));//动态分配内存空间
int t=0;
int i=0;
printf("请输入您要查找的值:\n");
scanf("%d",&t);
printf("请输入您要插在那个元素后面:\n");
scanf("%d",&i);
Node * p=head->next;
while(p!=NULL)
{
if(p->data==i)
{
s->data=t;
s->next=p->next;//注意是先指向新节点,再改旧节点
p-next=s;
printf("插入成功:\n");
break;
}
p=p->next;
}
if(p==NULL)
{
printf("没找到\n");
}
}
扩展阅读:
链表的C语言实现http://blog.csdn.net/21aspnet/article/details/146968
单链表功能大全 http://blog.csdn.net/21aspnet/article/details/159053
32.scanf
int d=0;
注意是scanf("%d",&d)
这里千万要注意如果写为scanf("d%",&d)一样可以编译通过,但是会引起隐含的bug!!!
注意是&d而不是d,因为是改变变量的地址对应的内容,而不是直接取其值
33.union
联合也叫共同体,和结构struct很像
struct可以定义一个包含多个不同变量的类型,每一个变量在内存中占有自己独立的内存空间,可以同时存储不同类型的数据。
uniion也可以定义一个包含多个不同变量类型,但这些变量只共有同一个内存空间,每次只能使用其中的一种变量存储数据。
问题是这样做有什么意义?
union{
int a;
float b;
char c;
}a;
a.b=10.123;printf("%d",a.a);
虽然没有初始化a,一样可以使用!union相当于pascal中的变体记录。就像c++中的重载这么方便。
34.register寄存器
register int a=1;
但是如果int *b=&a;这样会报错,因为不可以取到寄存器变量的地址。
但是使用VS2010调试时可以用&a.
35.enum枚举
就是为了定义一组同属性的值,默认的最前面的是0,后面的元素依次+1;
但是注意,每个枚举都唯一定义一个类型,里面的元素的值不是唯一的,枚举成员的初始化只能通过同一枚举的成员进行!!
之所以不用整数而用枚举是为了提高程序的可读性!
36.条件编译
形式一
#ifdef
#else
#endif
形式二
#ifndef
#else
#endif
形式三
#if
#else
#endif
非0 为真
选择功能是否加入,在编译阶段。
作用一:调试开关。
#define DEBUG//定义可调试 不定义则不
#ifnedf DEBUG
#ifdef
作用二:便于确定哪些头文件没有编译
#ifnedf FILE_H_
#define FILE_H_
总结:
#define 定义宏
#undef 取消已定义的宏
#if 如果给定条件为真,则编译下面代码
#ifdef 如果宏已经定义,则编译下面代码
#ifndef 如果宏没有定义,则编译下面代码
#elif 如果前面的#if给定条件不为真,当前条件为真,则编译下面代码
#endif 结束一个#if……#else条件编译块
#error 停止编译并显示错误信息
扩展阅读:http://blog.csdn.net/21aspnet/article/details/6737612
37.size_t
是为了方便系统之间的移植而定义的
在32位系统上 定义为 unsigned int
在64位系统上 定义为 unsigned long
更准确地说法是 在 32位系统上是32位无符号整形
在 64位系统上是64位无符号整形
size_t一般用来表示一种计数,比如有多少东西被拷贝等
38.perror
perror("fff");
perror ()用 来 将 上 一 个 函 数 发 生 错 误 的 原 因 输 出 到 标 准 设备 (stderr) 。参数 s 所指的字符串会先打印出,后面再加上错误原因字符串。此错误原因依照全局变量error 的值来决定要输出的字符串。
# ./a.out
fff: Success
39.main函数参数
argc:命令行参数个数;
argc:命令行参数数组;
env:环境变量数组;
#include <stdio.h>
int main(int argc,char *argv[],char *env[])
{
int i;
for(i=0; i<argc; i++)
printf("argv[%d]=%s\n",i,argv[i]);
for(i=0; env[i]!=NULL; i++)
printf("env[%d]:%s\n",i,env[i]);
return 5;
}
40.内存分配
程序中局部变量在栈上分配空间,系统自动分配管理;
动态分配内存在堆上,需要用户释放。
都在<stdlib.h>
申请内存:malloc
重新申请:realloc
释放内存:free
#include <stdio.h>
#include <stdlib.h>
main()
{
int i;
char c,*p;
p = (char *)malloc(10);//分配十个字节的缓存区
for(i=0;;i++)
{
c = getchar();//从键盘读取单个字符数据
if(i>9)
{
p = (char *)realloc(p,1);//重新增加申请一个字节的空间
}
if(c == '\n')
{
p[i] = '\0';
break;
}
else
{
p[i] = c;//将输入的字符保存到分配的缓存区
}
}
printf("%s\n",p);
free(p);//释放内存
}
41.memset
void *memset(void *s, int ch, size_t n);
函数解释:将s中前n个字节替换为ch并返回s;
memset:作用是在一段内存块中填充某个给定的值,它是对较大的结构体或数组进行清零操作的一种最快方法。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
main(){
char a[5];
memset(a,'1',sizeof(a));
printf("%s\n",a);
char buffer[] = "Hello world\n";
printf("Buffer before memset: %s\n", buffer);
memset(buffer, '*', strlen(buffer) );
printf("Buffer after memset: %s\n", buffer);
}
输出:
42.随机数
srand():生成随机数种子
rand():生成随机数
#include <stdio.h>
#include <stdlib.h>
main()
{
int i,j;
srand((int)time(0));//开始注释这里
for(i=0;i<10;i++)
{
j=1+(int)(10.0*rand()/(RAND_MAX+1.0));
printf("%d ",j);
}
}
输出:
说明:没有调用srand每次产生的都是一样的!