C强制类型转换总结

C强制类型转换


阅读目录:

一、强制类型转换目的、基本格式

二、C中变量的本质含义

三、普通变量强制类型转换

四、指针变量类型转换


一、强制类型转换目的、基本格式:

1.1、强转目的
C语言是强类型语言,如果一个运算符两遍的运算数据类型不同,先要将其转换为相同的类型,强制类型转换可以消除程序中的警告,即确保写代码的程序员自己清楚类型转换,允许丢失一定精度,或做类型匹配。
例如:

//1、丢失精度情况
int main(void)
{
    float f=5.75;
    printf("f=%d,f=%f\n",(int)f,f);
}
//结果:f=5,f=5.750000

//2、做类型匹配
int main(void)
{
    int a[5] = {1, 2, 3, 4, 5};
    int *p = (int *)(&a + 1);
    printf("*(p-1) = %d.\n", *(p-1));
}
//结果:*(p-1) = 5
//这里面&a + 1中+1,地址加sizeof(a)即一个数组长度
//(int *)将指向数组类型指针,转换为指向整形,赋值运算符左右对应

1.2、C中强转的基本格式

方法比较简单,如下所示:
TYPE1 A;
TYPE2 B;
A = (TYPE1)B; // 强制转换。


二、C中变量的本质含义:

(1)c中所有类型的数据存储在内存中,都是按照二进制格式存储的。所以内存中只知道有0和1,不知道是int的、还是float的还是其他类型。

(2)int、char、short等属于整形,他们的存储方式(数转换成二进制往内存中放的方式)是相同的,只是内存格子大小不同(所以这几种整形就彼此叫二进制兼容格式);而float和double的存储方式彼此不同,和整形更不同。

(3)C语言中的数据类型的本质,就是决定了这个数在内存中怎么存储的问题,也就是决定了这个数如何转成二进制的问题。一定要记住的一点是内存只是存储1010的序列,而不管这些1010怎么解析。所以要求我们平时数据类型存取一致。

int a =5;

存放:编译器给a分配4字节空间,并且将5按照int类型的存储方式转成二进制存到a所对应的内存空间中去(a做左值的);

取值:我们printf去打印a的时候(a此时做右值),printf内部的vsprintf函数会按照格式化字符串(就是printf传参的第一个字符串参数中的%d之类的东西)所代表的类型去解析a所对应的内存空间,解析出的值用来输出。

解析规则:存进去时是按照这个变量本身的数据类型来存储的(譬如本例中a为int所以按照int格式来存储);但是取出来时是按照printf中%d之类的格式化字符串的格式来提取的。此时虽然a所代表的内存空间中的10101序列并没有变(内存是没被修改的)但是怎么理解(怎么把这些1010转成数字)就不一定了。譬如我们用%d来解析,那么还是按照int格式解析则值自然还是5;但是如果用%f来解析,则printf就以为a对应的内存空间中存储的是一个float类型的数,会按照float类型来解析,值自然是很奇怪的一个数字了。

//数据类型存取差异
int main(void)
{
    //1、按照int类型存却按照float类型取 一定会出错
    int a = 5;
    printf("a = %d.\n", a);     // 5
    printf("a = %f.\n", a);     // 0.000000


    //2、按照int类型存却按照char类型取,

    int a = 6;
    char *p1 = &a;
    printf("*p1 = %d.\n", *p1);        //6在char范围,未出错

    short *p2 = &a;        
    printf("*p2 = %d.\n", *p2);        //6


    int a = 66666;
    char *p1 = &a;
    printf("*p1 = %d.\n", *p1);        //106超出char与short范围出错

    short *p2 = &a;        
    printf("*p2 = %d.\n", *p2);        //1130

}


小结:
(1)int和char类型都是整形,类型兼容的。所以互转的时候有时候错有时候对。

(2)int和char的不同在于char只有1个字节而int有4个字节,所以int的范围比char大。在char所表示的范围之内int和char是可以互转的不会出错;但是超过了char的范围后char转成int不会错(向大方向转就不会错,就好比拿小瓶子的水往大瓶子倒不会漏掉不会丢掉),而从int到char转就会出错(就好象拿大瓶子水往小瓶子倒一样)


三、普通变量强转规则:

3.1、普通变量转换规则
当较低类型的数据转换为较高类型时,一般只是形式上有所改变, 而不影响数据的实质内容, 而较高类型的数据转换为较低类型时则可能有些数据丢失。如果一个运算符两边的运算数类型不同,先要将其转换为相同的类型,即将运算符右侧数值转换为运算符左侧类型,同侧低精度类型转换为高精度类型,然后再参加运算,转换规则如下图所示。

double ← ── float 高

long

unsigned

int ← ── char,short 低

横向箭头表示必须的转换,如两个float型数参加运算,虽然它们类型相同,但仍要先转成double型再进行运算,结果亦为double型。

纵向箭头表示当运算符两边的运算数为不同类型时的转换,如一个long 型数据与一个int型数据一起运算,需要先将int型数据转换为long型,然后两者再进行运算,这些转换可以说是自动的,但尽量(确保自己允许转换发生)使用以显式的形式强制转换类型的机制。

3.2、具体转换细节

(1) 浮点型与整型

将浮点数(单双精度)转换为整数时,将舍弃浮点数的小数部分, 只保留整数部分。 将整型值赋给浮点型变量,数值不变,只将形式改为浮点形式,即小数点后带若干个0。注意:赋值时的类型转换实际上是强制的。

(2) 单、双精度浮点型

由于c语言中的浮点值总是用双精度表示的,所以float型数据只是在尾部加0延长为doub1e型数据参加运算,然后直接赋值。doub1e型数据转换为float型时,通过截尾数来实现,截断前要进行四舍五入操作。

(3) char型与int型

int型数值赋给char型变量时,只保留其最低8位,高位部分舍弃。 char型数值赋给int型变量时,一些编译程序不管其值大小都作正数处理,而另一些编译程序在转换时,若char型数据值大于127,就作为负数处理。对于使用者来讲,如果原来char型数据取正值,转换后仍为正值;如果原来char型值可正可负,则转换后也仍然保持原值,只是数据的内部表示形式有所不同。

(4) int型与1ong型

long型数据赋给int型变量时,将低16位值送给int型变量,而将高16 位截断舍弃。(这里假定int型占两个字节)。 将int型数据送给long型变量时,其外部值保持不变,而内部形式有所改变。

(5) 无符号整数

将一个unsigned型数据赋给一个占据同样长度存储单元的整型变量时(如:unsigned→int、unsigned long→long,unsigned short→short) ,原值照赋,内部的存储方式不变,但外部值却可能改变

用强制类型转换是一个好习惯,这样,至少从程序上可以看出想干什么。


四、指针变量强转规则:

(1)指针变量强转,改变了指针取址能力。

(2)一个指针涉及2个变量:一个是指针变量自己本身,一个是指针变量指向的那个变量

(3)int p;定义指针变量时,p(指针变量本身)是int 类型,*p(指针指向的那个变量)是int类型的。

(4)int 类型说白了就是指针类型,只要是指针类型就都是占4字节,解析方式都是按照地址的方式来解析(意思是里面存的32个二进制加起来表示一个内存地址)的。结论就是:所有的指针类型(不管是int 还是char * 还是double *)的解析方式是相同的,都是地址。

(5)对于指针所指向的那个变量来说,指针的类型就很重要了。指针所指向的那个变量的类型(它所对应的内存空间的解析方法)要取决于指针类型。譬如指针是int *的,那么指针所指向的变量就是int类型的。

#include <stdio.h>
int main(void)
{

    int a[3] = {0x11223344, 0x55667788, 0};

    int *p1 = a;
    printf("*p1 = 0x%x\n", *p1);         //*p1 = 0x11223344

    char *p2 = (char *)a;              //改变取址能力         
    printf("*p2 = 0x%x\n", *p2);      //*p2 = 0x44
    printf("*p2 = 0x%x\n", *(p2+1));   //*p2 = 0x33
    printf("*p2 = 0x%x\n", *(p2+2));   //*p2 = 0x22
    printf("*p2 = 0x%x\n", *(p2+3));   //*p2 = 0x11
    printf("*p2 = 0x%x\n", *(p2+4));   //*p2 = 0xffffff88printf("*p2 = 0x%x\n", *(p2+5));  //*p2 = 0x77

    return 0;
}
### 回答1: 在 C 语言中,强制类型转换是将一个数据类型转换为另一个数据类型的过程。它可以通过在需要转换的变量或表达式前面加上括号并在括号内指定需要转换的类型来完成。 例如,将一个 int 类型的变量 x 转换为 float 类型,可以使用以下语法: ``` float y = (float)x; ``` 这将把 x 的值转换为浮点数,并将结果赋给 y。需要注意的是,在进行强制类型转换时,可能会导致精度丢失或数据溢出等问题,因此应该谨慎使用。 ### 回答2: C语言中的强制类型转换是通过在需要转换的值或变量前面加上括号,并在括号中指明所需转换的目标类型来实现的。 强制类型转换可以用于将一种数据类型转换为另一种数据类型。当需要在不同的数据类型之间进行赋值或运算时,需要进行类型转换,以保证程序的正确性。 C语言强制类型转换有两种形式:显示类型转换和隐式类型转换。 显示类型转换是通过在需要转换的值或变量前面加上括号,并在括号中指明所需转换的目标类型来实现的。例如,如果想将一个整数转换为浮点数,可以使用以下语法: float result = (float) 10; 隐式类型转换是在不显式指定类型转换的情况下,由编译器自动进行的类型转换。例如,当将一个整数和一个浮点数相加时,编译器会将整数隐式地转换为浮点数,然后进行相加操作。 需要注意的是,在进行强制类型转换时,可能会出现精度丢失或数据溢出的情况,因此在进行类型转换时需要谨慎,确保转换的结果符合预期。 总结来说,C语言中的强制类型转换是通过在需要转换的值或变量前面加上括号,并在括号中指明所需转换的目标类型来实现的。强制类型转换可以用于不同数据类型之间的赋值和运算,但需要注意可能会出现精度丢失或数据溢出的情况,因此在进行类型转换时需要谨慎。 ### 回答3: C语言中的强制类型转换是一种将一种数据类型转换为另一种数据类型的操作。在某些情况下,我们可能需要将一个变量或表达式的数据类型修改为另一个类型,以便能够进行一些特定的运算或实现特定的功能。 强制类型转换可以通过在要转换的值或表达式之前放置所需的目标类型,使用圆括号进行实现。例如,如果我们想将一个整数类型的变量强制转换为浮点类型,在变量前面放置浮点类型即可实现转换。 强制类型转换的一些使用场景包括: 1. 当我们进行除法运算时,将整数类型的操作数转换为浮点类型,以避免整数除法的截断问题。 2. 当我们将一个较大的数据类型转换为较小的数据类型时,以确保不会出现数据丢失或溢出的问题。 3. 在某些函数调用或表达式计算中,可能需要将一种数据类型的值转换为另一种数据类型,以满足函数或表达式的要求。 需要注意的是,强制类型转换有可能导致精度损失或数据截断,因此在进行转换时需要谨慎操作。为了保证程序的可读性和可维护性,我们应该尽量避免过多的强制类型转换,尽量使用适当的数据类型来处理数据。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值