003-scanf函数使用和表达式-C语言笔记

/ 0

输入函数scanf的基本使用方法

scanf 函数的声明在标准输入输出头文件 stdio.h 中,这个函数用于接受键盘输入的内容。

语法:scanf("格式控制字符串",输入项地址列表);

格式控制字符串:规定数据输入的格式,由格式控制符和普通字符组成,格式控制符和百分号(%)一起使用,用来说明输入数据的数据类型(格式字符)。

输入项地址列表:需要接收数据的变量地址,这些输入项与格式控制字符串在类型和数量上要对应,当有多个输入项时,各个地址名之间以逗号 , 分隔。

在C语言中,变量的地址可以通过地址运算符 & 得到:

int a = 10, b = 20;
&a, &b; //取得变量a、b的地址

常见的scanf格式控制符

QQ20150625-2

#include <stdio.h>

int main(int argc, const char * argv[]) {

    //定义两个int类型的变量用于储存QQ账号和密码,并初始化
    long qqNumber = 0, qqPassWord = 0;

    //提示用户输入QQ账号
    printf("请输入你的QQ账号:\n");

    //接收用户输入的QQ账号并储存到qqNumber变量中
    scanf("%11ld", &qqNumber);

    //提示用户输入QQ密码
    printf("请输入你的QQ密码:\n");

    //接收用户输入的QQ密码
    scanf("%16ld",&qqPassWord);

    printf("你的QQ:%ld登陆成功!\n",qqNumber);
    /*
    请输入你的QQ账号:
    44334512
    请输入你的QQ密码:
    123456
    你的QQ:44334512登陆成功!
    */
    return 0;
}

输入函数scanf运行原理和缓冲区理解

使用 scanf 函数并不是直接将用户输入的数据赋值给变量,而且先将用户输入的数据全部存到缓冲区,注意任何字符都会存到缓冲区(包括回车),然后才给依次给变量赋值。如果缓冲区的数据不为空,调用 scanf 函数会直接从缓冲区中获取数据,而不会要求用户输入。

解决办法,使用 rewind(stdin) 清空缓冲区:

#include <stdio.h>
#include <stdlib.h>

int main(int argc, const char * argv[]) {

    int num = 0,Num = 0;
    char ch = 'a';
    printf("请输入:\n");
    scanf("%d%d",&num,&Num);
    rewind(stdin);//清空缓冲区的所有数据
    printf("请输入:\n");
    scanf("%c",&ch);//这样ch就能正确接收到用户输入的内容

    printf("num = %d,Num = %d,ch = %c\n",num,Num,ch);
    /*
    控制台输出
    请输入:
    34 424 y 34 y
    请输入:
    y
    num = 34,Num = 424,ch = y
    */
    return 0;
}

算术运算符和算术表达式的使用

QQ20150625-7

算术表达式就是算术运算符连接起来的表达式,C 语言里的表达式的结果不处理也不会报错,只是会警告提示表达式结果未被使用。

求余运算中,如果两个数中有负数,余数的正负取决于第一个数的正负:

#include <stdio.h>

int main() {
        int result1 = -10 % 3;
        int result2 = 10 % -3;
        printf("result1 = %d\n",result1);//result1 = -1
        printf("result2 = %d\n",result2);//result2 = 1

    return 0;
}

数据类型自动转换和强制转换

类型转换分为:隐式数据类型转换显式数据类型转换

隐式转换:

int i = 1;
i = i + 9.801;//i转换为double型再与9.801相加,然后将结果转换为整型赋值给i
printf("i = %d\n",i);//i = 10

显式转换:

int i = 1;
i = i + (int)9.801;//将9.801转换为整型再与i相加,然后将结果直接赋值给i
printf("i = %d\n",i);//i = 10

将大范围的数据赋值给小范围变量时,系统会自动做一个强制类型转换的操作,这样容易丢失精度。

int a = 10.7; //隐式类型转换
int b = (int)10.7; //显式类型转换
int c = 198l; //long 转换为 int
char d = 65; //int 转换为 char
int e = 19.5f; //float 转换为 int

赋值运算符和复合赋值运算符

赋值运算符:赋值运算符标记为 =,将赋值运算符右边的表达式的值赋值给左边的变量,赋值运算符都是从右向左赋值。

int a, b, c;
a = b = c = 10;//这里是将10赋值给c,然后将c的值赋值给b,再将b的值赋值给a。

复合赋值运算符:在赋值运算符 = 前加上其它二目运算符可构成复合赋值运算符。

#include <stdio.h>

int main(int argc, const char * argv[]) {

    int a = 10;//声明一个int类型变量a,并将10赋值给变量a

    int b,c,d,e;//声明4个int类型变量,b,c,d,e。

    b = c = d = e = 10;//从右到左为变量赋值。

    a += 10;//等价于 a = a + 10;
    b -= 10;//等价于 b = b - 10;
    c *= 10;//等价于 c = c * 10;
    d /= 10;//等价于 d = d / 10;
    e %= 10;//等价于 e = e % 10;

    printf("a = %d\nb = %d\nc = %d\nd = %d\ne = %d\n",a,b,c,d,e);
    return 0;
}

自增自减运算符的使用注意

在程序设计中,经常遇到 i = i + 1i = i - 1 这两种极为常用的操作。C 语言为这种操作提供了两个更为简洁的运算符,即 ++--,分别叫做自增运算符自减运算符。它们是单目运算符,是从右向左结合的算术运算符。

前缀表达式:先完成变量的自增自减1运算,再用x的值作为表达式的值。即“先变后用”,也就是变量先进行自增自减运算,再将变量参与运算。

++x, --x;

后缀表达式:先用当前值作为表达式的值,再进行自增自减1运算。即“先用后变”,也就是将变量参与运算,然后才对变量进行自增自减运算。

x++, x--;

示例:

#include <stdio.h>

int main(int argc, const char * argv[]) {

    int i = 1;
    int b = 10;
    printf("i++ = %d\n",i++);//这里++在后,所以i++表达式的值是1,但此后i也自增了1
    printf("i = %d\n",i);//这里i就是2了

    b += ++i;//这里++在前,所以++i表达式的值是3,10+3=13
    printf("b = %d\n",b);//b=13
    printf("i = %d\n",i);//i=3

    b += i++;//这里++又在后了,所以i++表达式的值也是3,用完表达式的值后,i自增了1
    printf("b = %d\n",b);//13+3=16
    printf("i = %d\n",i);//i=4

    return 0;
}

逗号表达式

在C语言中逗号 , 也是一种运算符,称为逗号运算符。其功能是把多个表达式连接起来组成一个表达式,逗号表达式的值就是从左到右计算各个表达式,取最后一个表达式的值为逗号表达式的值。

#include <stdio.h>

int main() {
    int a = 10, b = 20, c = 30;
    int result = (a + b, b + c);
    printf("result = %d",result);//result = 50

    return 0;
}

如果你光看了这个实例,认为逗号表达式的值就是只看最后一个表达式,那就大错特错了。

#include <stdio.h>

int main(int argc, const char * argv[]) {
    int i = 1;
    int result = (i++, i++, ++i);//i++ i++了两次,i就变成了3,然后最后一个表达式是++i,所以表达式的值就是4
    printf("result = %d\n",result);//result = 4
    return 0;
}