1. 变量定义和声明
让我们更为仔细的看一下变量的定义和变量声明的区别,考虑如下例子:
int tern = 1; //定义tern
main()
{
extern int tern;//使用在其他地方定义的tern变量
}
这里,tern声明两次。第一次声明为变量留出了存储空间。它构成了变量的定义。第二次声明只是告诉编译器要使用先前定义的变量tern,因此不是定义。第一次声明称为定义声明。第二次声明称为引用声明,关键字extern表明该声明不是一个定义,因为它指示编译器参考其他地方。一个外部变量只可进行一次初始化,而且一定是在变量被定义时进行。下面的语句是错误的:extern char str = ‘a’; /*错误的*/。
2. extern的使用
ANSI c通过在一个文件中定义变量,在其他文件中引用声明这个变量来实现共享,也就是说,除了在一个声明(定义声明)外,其他所有声明都必须使用关键字extern。
注意:除非在第二个文件中也声明了该变量(通过使用extern),否则在一个文件中定义的外部变量不可以在第二个文件。
3. 存储类与动态内存分配
大多数人搞不清楚存储类型和动态内存之间的关系,我们来看一个理想模型。可以认为程序将可用内存分成三个独立的部分:(1)具有外部链接的,具有内部链接的以及具有空链接的静态变量;(2)自动变量;(3)动态分配的内存。
在编译期时就已经知道了静态存储时期存储类变量所需的内存数量,存储在这一部分的数据在整个程序运行期间都可用。这一类型的每个变量在程序开始时就已经存在,到程序结束时终止。
一个自动变量在程序进入包含该变量定义的代码块时产生,在退出这一代码块时终止。伴随着程序对函数调用和终止,自动变量使用的内存数量也在增加和减少。将这一部分内存处理为一个堆栈。说明在内存中,新变量在创建时按顺序加入,在消亡时按相反顺序移除。
动态分配的内存在调用malloc()或相关函数产生,在调用free()时释放。而且这块内存由程序员管理。因此内存块可在一个函数中创建,另一个函数中释放。由于这一点,动态内存分配所用的内存部分可能变成碎片状,不管咋样,使用动态内存往往导致进程比使用堆栈内存慢。
4. /*(1)一个c变量的作用域可以是代码块作用域,文件作用域,函数原型作用域;(2)代码块就是从'{'开始,到这个'}'结束。(3)如果外部变量使用了static修饰,则为内部链接,否则为外部链接。例如:8行a变量,9行b变量.具有代码块作用域和函数原型作用域的变量有空链接。(4)无论内部链接,还是外部链接,都具有静态存储时期。如果一个变量具有静态存储时期,它在程序执行期间一直存在。具有文件作用域的变量具有静态存储时期。例如:9行的变量b。如果一个变量具有代码块作用域一般情况具有自动存储时期,例如:53行i变量,当input函数结束了,i变量也就结束。
*/
#include<stdio.h>
#include<string.h>
int login(char *str1, char *str2);
void input(char *str1, char *str2);
int a = 88; /*具有文件作用域,外部链接。和该文件属于同一程序的其他文件可以使用a变量。*/
static int b = 0;/*具有文件作用域,内部链接。可以被该文件中的任一函数使用。*/
int main()
{
char str1[30];
char str2[30];
input(str1, str2);
return 0;
}
int c = 0; /*c的生存期是17~69*/
int login(char *str1, char *str2) /*str1,str2,i,a都具有代码块作用域。str1的生存期18~49.*/
{
printf("c= %d\n", c);
if(!strcmp(str1, "admin\n") && !strcmp(str2, "123456\n"))
{
int i = 0; /*i的生存期23~41*/
int a = 0;
for(i = 0; i < 3; i++)
{
printf("请输入验证码!\n");
scanf("%d", &a);
if(a == 234)
{
printf("登录成功!\n");
break;
}
else
{
printf("输入的验证码错误!\n");
continue;
}
}
return 1;
}
else
{
printf("登录失败!\n");
printf("输入了错误的用户名或密码,请重新输入!\n");
return 0;
}
}
void input(char *str1, char *str2)
{
int i = 0;
static int d = 4;/*d变量在整个程序执行期间一起存在,直到程序执行结束才结
束,而不因input函数的结束而结束。*/
int p = 0;
for(i = 0; i < 3; i++)
{
printf("please input username,str1的长度5\n");
// scanf("%s", str1);
fgets(str1, 30, stdin);
printf("please input secret ,str2的长度6\n");
//scanf("%d", &secret);
fgets(str2, 30, stdin);
p = login(str1, str2);
if(p == 1)
break;
}
printf("a= %d\n", a);
}
/*上述程序中有的变量是多余的,定义这些变量是为了说明存储类型,链接,内存管理这些知识点*/