咱们在写程序的时候经常会遇到一些不好找的bug,有的并不是很难,只是大家容易忽略,今天咱们就来看一个,关于C语言自动类型转换的bug。

先看一段代码:

void getNext(int * next,string str){

int i=0,j=-1;

next[0]=-1;

while(i < str.length()-1){

if(j==-1 || str[i]==str[j]){

i++;

j++;

next[i]=j;

}

else{

j=next[j];

}

}

}

int KMPMatch(string buffer,string p){

int next[100];

getNext(next,p);

int i=0,j=0;

while(i < buffer.length() && j < p.length() ){

if(j == -1 || buffer[i] == p[j]){

i++;

j++;

}

else

j=next[j];

}

if(j == p.length())

return i-j;

else

return -1;

}

咋看之下似乎一切都没有问题,可是实际运行起来都是返回 -1

debug看了下,每次当在 i=0,j=-1时,对于

while(i < buffer.length() && j < p.length() )

判定条件不成立,跳出while循环,直接执行27行的 if(j == p.length()) ?

再看看length函数的声明,返回类型是size_t !! 是一个无符号数!!

size_t length() const;

《C++ Primer》里面提示道:切勿混用带符号和无符号类型

如果表达式里面既有带符号类型又有无符号类型,当带符号类型取值为负时会出现异常结果,这是因为带符号数会自动转换成无符号数。

当有符号数 j 和无符号数 string::length()比较时,j被自动转换成无符号整数,j 直接被转换成一个很大的数!!使得while循环直接跳出,返回错误的结果。

接下来咱们就来总结一下:

自动转换遵循以下规则:

1、若参与运算量的类型不同,则先转换成同一类型,然后进行运算。

2、转换按数据长度增加的方向进行,以保证精度不降低。如int型和long型运算时,先把int量转成long型后再进行运算。

a.若两种类型的字节数不同,转换成字节数高的类型

b.若两种类型的字节数相同,且一种有符号,一种无符号,则转换成无符号类型

3、所有的浮点运算都是以双精度进行的,即使仅含float单精度量运算的表达式,也要先转换成double型,再作运算。

4、char型和short型参与运算时,必须先转换成int型。

5、在赋值运算中,赋值号两边量的数据类型不同时,赋值号右边量的类型将转换为左边量的类型。如果右边量的数据类型长度左边长时,将丢失一部分数据,这样会降低精度,丢失的部分按四舍五入向前舍入。

关于隐式转换

隐式类型转换分三种,即算术转换、赋值转换和输出转换。

1.算术转换

进行算术运算(加、减、乘、除、取余以及符号运算)时,不同类型数招必须转换成同一类型的数据才能运算,算术转换原则为:

在进行运算时,以表达式中最长类型为主,将其他类型位据均转换成该类型,如:

(1)若运算数中有double型或float型,则其他类型数据均转换成double类型进行运算。

(2)若运算数中最长的类型为long型.则其他类型数均转换成long型数。

(3)若运算数中最长类型为int型,则char型也转换成int型进行运算。算术转换是在运算过程中自动完成的。

2.赋值转换

进行赋值操作时,赋值运算符右边的数据类型必须转换成赋值号左边的类型,若右边的数据类型的长度大于左边,则要进行截断或舍入操作。

下面用一实例说明:

char ch;

int i,result;

float f;

double d;

result=ch/i+(f*d-i);

(1)首先计算 ch/i,ch → int型,ch/i → int型。

(2)接着计算 f*d-i,由于最长型为double型,故f→double型,i→double型,f*d-i→double型。

(3)(ch/i) 和(f*d-i)进行加运算,由于f*d-i为double型,故ch/i→double型,ch/i+(f*d-i)→double型。

(4)由于result为int型,故ch/i+(f*d-i)→double→int,即进行截断与舍入,最后取值为整型。

3.输出转换

在程序中将数据用printf函数以指定格式输出时,当要输出的盐据类型与输出格式不符时,便自动进行类型转换,如一个long型数据用整型格式(%d)输出时,则相当于将long型转换成整型(int)数据输出;一个字符(char)型数据用整型格式输出时,相当于将char型转换成int型输出。

注意:较长型数据转换成短型数据输出时,其值不能超出短型数据允许的值范围,否则转换时将出错。如:

long a=80000;

printf("%d",a);

运行结果为14464,因为int型允许的最大值为32767,80000超出此值,故结果取以32768为模的余数,即进行如下取余运算:

(80000-32768)-32768=14464;

输出的数据类型与输出格式不符时常常发生错误,如:

int d=9;

printf("%f",d);

float c=3.2;

printf("%d",c);

将产生错误的结果。

同一句语句或表达式如果使用了多种类型的变量和常量(类型混用),C 会自动把它们转换成同一种类型。以下是自动类型转换的基本规则:

1、在表达式中,char 和 short 类型的值,无论有符号还是无符号,都会自动转换成 int 或者 unsigned int(如果 short 的大小和 int 一样,unsigned short 的表示范围就大于 int,在这种情况下,unsigned short 被转换成 unsigned int)。因为它们被转换成表示范围更大的类型,故而把这种转换称为“升级(promotion)”。

2、按照从高到低的顺序给各种数据类型分等级,依次为:long double, double, float, unsigned long long, long long, unsigned long, long, unsigned int 和 int。这里有一个小小的例外,如果 long 和 int 大小相同,则 unsigned int 的等级应位于 long 之上。char 和 short 并没有出现于这个等级列表,是因为它们应该已经被升级成了 int 或者 unsigned int。

3、在任何涉及两种数据类型的操作中,它们之间等级较低的类型会被转换成等级较高的类型。

4、在赋值语句中,= 右边的值在赋予 = 左边的变量之前,首先要将右边的值的数据类型转换成左边变量的类型。也就是说,左边变量是什么数据类型,右边的值就要转换成什么数据类型的值。这个过程可能导致右边的值的类型升级,也可能导致其类型降级(demotion)。所谓“降级”,是指等级较高的类型被转换成等级较低的类型。

5、作为参数传递给函数时,char 和 short 会被转换成 int,float 会被转换成 double。使用函数原型可以避免这种自动升级。

c语言编程车,C语言编程之自动类型转化相关推荐

  1. c语言自动转化,C语言编程之自动类型转化

    咱们在写程序的时候经常会遇到一些不好找的bug,有的并不是很难,只是大家容易忽略,今天咱们就来看一个,关于C语言自动类型转换的bug. 先看一段代码: void getNext(int * next, ...

  2. 【Groovy】Groovy 动态语言特性 ( Groovy 中的变量自动类型推断以及动态调用 | Java 中必须为变量指定其类型 )

    文章目录 前言 一.Groovy 动态语言 二.Groovy 中的变量自动类型推断及动态调用 三.Java 中必须为变量指定其类型 前言 Groovy 是动态语言 , Java 是静态语言 ; 一.G ...

  3. 在哪儿能找c语言编程题,C语言程序设计的试题及答案

    大家在考程序员时,C语言程序设计大家有了解吗?下面小编为大家分享了,供大家参考. 第一章 基础知识 一.填空 1. 每个 C 程序都必须有且仅有一个________ 函数. 2. C 语言程序开发到执 ...

  4. 3.4 学编程不拘于语言,学语言不限于平台——《逆袭大学》连载

    返回到[全文目录] 目录 3.4 学编程不拘于语言,学语言不限于平台 编程语言 软件和硬件--计算机要作为一个整体看待 语言的江湖 语言不是回事 多平台上的精彩 3.4 学编程不拘于语言,学语言不限于 ...

  5. 第1章 JVM语言家族概览 《Kotin 编程思想·实战》

    第1章 JVM语言家族概览 天地和而万物生,阴阳接而变化起.<荀子·礼记> 1.1 编程语言简述 1.1.1 编程语言是什么 所谓编程语言只是一个抽象的规范,而编译器是这个规范的实现,它是 ...

  6. 编程每种语言有什么用_每种情况的最佳编程语言

    编程每种语言有什么用 有一个问题,很多人都认为提问者根本不理解该主题. 有些人甚至觉得它很生气. 问题的形式通常是什么是最佳X? 什么是最好的车? 最好的编程语言是什么? 但是同时,这是一个问题,我们 ...

  7. 学单片机需要多少C语言基础,C语言单片机编程教程

    很多学员和朋友经常找我给他们推荐C语言的书,他们可能想和我当年一样,闷头大干一场,学个几个月的C语言,把C语言吃透,再开始单片机程序开发! 别!别!别! 大家千万别这么干,我当年自学了3个月的C语言, ...

  8. 聊聊在博客园写博客的这两年《Unity 3D脚本编程:使用C#语言开发跨平台游戏》正式出版...

    版本状态: 2016.9 第一次印刷 (2016.11 输出到台湾) 2017.1 第二次印刷 2017.5 第三次印刷 2017.5 电子书上线:Unity 3D脚本编程--使用C#语言开发跨平台游 ...

  9. c语言图形方式初始化,c语言图形方式下的编程.doc

    c语言图形方式下的编程.doc C语言图形方式下的编程 学习目标 ·了解PC显示系统的结构 ·C语言图形初始化的一般方法 ·C语言常用的图形处理函数 显示系统简介 PC机显示系统一般是由显示器和显示卡 ...

最新文章

  1. android定义空字符串数组,android – retrofit:处理可以是空字符串或数组的属性
  2. 赵雅智_android多线程下载带进度条
  3. Outlook2010 Bug 一则
  4. Reactjs 踏坑指南3:一些例子(未完成)
  5. Web Service 学习笔记(2)
  6. 金蝶显示中间服务器忙,金蝶显示云服务器繁忙怎么回事
  7. Qt Creator常问问题FAQ
  8. 关于自然语言处理,有一本通关手册待接收
  9. Linux计划任务Crontab介绍
  10. 离职总结:大公司与小公司的个人体验
  11. Description Resource Path Location Type Project configuration is not up-to-date with pom.xml. Select
  12. 张一鸣批评的互联网“语言腐败”,危害到底有多大?
  13. 一个CSS3滤镜Drop-shadow阴影效果
  14. vs 外部依赖项、附加依赖项以及如何添加依赖项目
  15. 近期的技术问题让云供应商进行预设加密
  16. SQL*Loader 和 Data Pump
  17. JAVA中public protected default private访问权限
  18. 手机4g接台式计算机,我用4G手机开热点连接电脑网速很慢怎么回事
  19. 网络蠕虫一般利用计算机系统,网络蠕虫是什么
  20. 2013年节假日放假安排时间表

热门文章

  1. 【Spring】Spring boot 可以通过集成jolokia来使用HTTP形式访问mbean
  2. Spark Structured : HIve jdbc方式访问待下划线的表,找不到表的错误
  3. 95-150-080-源码-Sink-StreamingFileSinkBucketingSink
  4. 浅谈 Mybatis中的 ${ } 和 #{ }的区别
  5. Spark的ShuffleManager
  6. erc20怎么查询代币交易记录_信用卡在pos机上刷卡手续费怎么算?信用卡刷卡记录如何查询?...
  7. 分布式缓存 Redis 集群搭建,这里一次性帮你搞定!
  8. JavaScript小效果的实现(笔记)
  9. linux改变时间 find,Find命令查找最近几天修改的文件
  10. leetcode题解-买卖股票的最佳时机