开题报告第一版写完发给老师了,熬了两周终于搞出来了,等着被怼了之后再改吧。晚上选了Leetcode一道简单的题,整数反转,就是将一个int类型的数反转。原本确实很简单,最后出现个问题有意思——整数溢出。

溢出

题目给出的要求是给出一个 32 位的有符号整数,因此可以确定是int类型,但是32位int类型的范围是[$-2^{31}, 2^{31}-1$], -2147483648-2147483647,那么问题来了,我提交代码后,提示错误:

Exception in thread "main" java.lang.NumberFormatException: For input string: "9646324351"

意思就是无法将9646324351转为int,很明显给出的原始数字是1534236469,这是一个规范的int类型数值,然而反转后9646324351很明显超过了范围。溢出了。当时我取巧了,既然是这个异常,直接在catch后return 0,机智如我,是个很人,提交,完美通过。

判断整数溢出

说到底还是需要知道怎么做判断啊,我的第一直觉是转为BigDecimal,很明显,这很方便且有效:

 /**
* 方案二:转为BigDecimal
*/
BigDecimal b = new BigDecimal(sb.toString());
if(b.compareTo(new BigDecimal(String.valueOf(Integer.MAX_VALUE))) > 0) {
// 溢出了
}else{ }

sb.toString()就是需要判断的字符串。

那么除了转为BigDecimal,还有什么方式呢,肯定还有啦;

整数可以看作是相应位数上的数字做如下操作,比如123

  1. 0*10 + 1 = 1
  2. 1*10 + 2 = 12
  3. 12 * 10 + 3 = 123

    每次增加10倍,并加上各位数字,因此可以在每一次sum*10+digit之前做判断。
if(Integer.MAX_VALUE/10 < sum || (Integer.MAX_VALUE/10 == sum && Integer.MAX_VALUE % 10 < digit)) {
// 说明溢出
return true;
}

完整判断代码如下:

private boolean outOfRange(String str) {
int length = str.length();
if(length == 0){
return false;
}
int index = 0;
int sum = 0; // 记录累加结果
while(index < length) {
int digit = str.charAt(index) - '0';
// 这里假定str是合法的字符串,不需要进行digit合法性判断 if(Integer.MAX_VALUE/10 < sum || (Integer.MAX_VALUE/10 == sum && Integer.MAX_VALUE % 10 < digit)) {
// 说明溢出
return true;
}
// 说明还没有溢出
sum = sum * 10 + digit;
index ++;
}
return false;
}

参数str是需要判断的字符串,这里没有处理符号位,如果有-直接去掉,按照Integer.MAX_VALUE做比较同样可以。同时函数也没有考虑str中含有不合法的数字字符,重点关注溢出嘛。

LeetCode整数反转

最后还是贴上整数反转的代码

package com.jiajia.m5;

import java.math.BigDecimal;

/**
* @ClassName: ReverseInteger
* @Author: fanjiajia
* @Date: 2019/5/17 下午8:00
* @Version: 1.0
* @Description:
*/
public class ReverseInteger { public static void main(String[] args) {
ReverseInteger r = new ReverseInteger();
System.out.print(r.reverse(1534236469));
} public int reverse(int x) {
if(x == 0) {
return 0;
}
char[] ch_arr = String.valueOf(x).toCharArray();
StringBuilder sb = new StringBuilder();
int end_index = ch_arr[0] == '-'? 1 : 0;
boolean flag = ch_arr[ch_arr.length - 1] == '0'; // 最后一位为0
for(int i = ch_arr.length - 1; i >= end_index; i--) {
if(flag && ch_arr[i] == '0') {
continue;
}
flag = false;
sb.append(ch_arr[i]);
}
/**
* 方案一:异常里面返回,也是够了
*/
// int result;
// try {
// result = end_index == 1? -Integer.valueOf(sb.toString()) : Integer.valueOf(sb.toString());
// }catch(NumberFormatException e) {
// return 0;
// }
// return result;
/**
* 方案二:转为BigDecimal
*/
// BigDecimal b = new BigDecimal(sb.toString());
// if(b.compareTo(new BigDecimal(String.valueOf(Integer.MAX_VALUE))) > 0) {
// return 0;
// }else{
// return end_index == 1? -Integer.valueOf(sb.toString()) : Integer.valueOf(sb.toString());
// } /**
* 方案3:做溢出判断
*/
if(outOfRange(sb.toString())) {
return 0;
}else {
return end_index == 1? -Integer.valueOf(sb.toString()) : Integer.valueOf(sb.toString());
}
} private boolean outOfRange(String str) {
int length = str.length();
if(length == 0){
return false;
}
int index = 0;
int sum = 0; // 记录累加结果
while(index < length) {
int digit = str.charAt(index) - '0';
// 这里假定str是合法的字符串,不需要进行digit合法性判断 if(Integer.MAX_VALUE/10 < sum || (Integer.MAX_VALUE/10 == sum && Integer.MAX_VALUE % 10 < digit)) {
// 说明溢出
return true;
}
// 说明还没有溢出
sum = sum * 10 + digit;
index ++;
}
return false;
}
}

有意思的是写了个溢出判断函数的效率还没有直接catch异常来得高!!!

最后

研三毕业了,羡慕!!!

此致,敬礼!!!

Java判断整数溢出的更多相关文章

  1. Java 判断整数方法

    今天写代码的时候突然想到要怎么来判断整数,然后通过判断是否是整数来处理相关的操作.开始想到了几个方法,比如百度到的 x(int) instanceof Integer,但是这样的话程序会报错,还有一个 ...

  2. java基本数据类型转换溢出问题

    java的基本数据类型有(int.byte.double.float.char.boolean.long.short):这里介绍整型数据 示例1: public class H_Z01 { publi ...

  3. java整数溢出问题及提升为long型

    整数溢出问题 Java 中的 int 用 32 位表示,正数最大值的情况,首位是 0,其他位都可以是 1(就是 2^31-1).但是如果正数过大了,例如 2^31,计算机不得不把首位变成 1,并且计算 ...

  4. Java之整数运算

    Java的整数运算遵循四则运算规则,可以使用任意嵌套的小括号.四则运算规则和初等数学一致.例如: public class Main { public static void main(String[ ...

  5. CVE-2012-0774:Adobe Reader TrueType 字体整数溢出漏洞调试分析

    0x01 TrueType 字体 TTF 字体是 Apple 和 Microsoft 两家公司共同推出的字体格式,现在已经广泛的运用于 Windows 操作系统,其中 PDF 文档也可以嵌入 TTF ...

  6. (转)java判断string变量是否是数字的六种方法小结

    java判断string变量是否是数字的六种方法小结 (2012-10-17 17:00:17) 转载▼ 标签: it 分类: 转发 1.用JAVA自带的函数 public static boolea ...

  7. CTF 两道web整数溢出题目(猫咪银行和ltshop)

    ①猫咪银行: (2018中科大hackgame) 一开始给十个CTB,而flag需要20个CTB,我们需要理财赚够20个. 理财是只能买入TDSU才可以获得收益.我们先上来直接把CTB全部换成TDSU ...

  8. CVE-2013-2551:Internet Explore VML COALineDashStyleArray 整数溢出漏洞简单调试分析

    0x01 2013 Pwn2Own 黑客大赛 在 Pwn2Own 的黑客大赛上,来自法国的 VUPEN 安全团队再一次利用 0day 漏洞攻破 Windows8 环境下的 IE10 浏览器,这一次问题 ...

  9. PWN学习之整数溢出

    目录 PWN学习之整数溢出 整数溢出 溢出和回绕 漏洞多发函数 整数溢出例子 PWN学习之整数溢出 整数溢出 如果一个整数用来计算一些敏感数值,如缓冲区大小或数值索引,就会产生潜在的危险.通常情况下, ...

随机推荐

  1. MacOS X GateKeeper Bypass

    MacOS X GateKeeper Bypass OVERVIEW On MacOS X version <= 10.14.5 (at time of writing) is it possi ...

  2. kubernetes 资源清单定义入门

    k8s中的资源 什么叫资源? k8s中所有的内容都抽象为资源, 资源实例化之后,叫做对象 在k8s中有哪些资源? 工作负载型资源(workload): Pod ReplicaSet Deploymen ...

  3. linux-centos7.6设置固定IP网络方法

    两种方法设置固定IP 本文分别用了虚拟机网络模式桥接模式和Net模式,至于两者直接的区别可查看其他文档. 一.安装时设置固定IP地址 1.在系统设置界面,点击“网络和主机名”选项,可以看到默认是未连接 ...

  4. django操作mysql

    连接mysql 1.安装pymysql 操作指令 : pymsql: pip install pymysql 2.导入库 在项目目录下的__init__.py文件中导入pymysql模块 加入以下两行 ...

  5. 15 Windows编程——系统内置窗口子类型之button

    button子类型BS_3STATE.BS_AUTO3STATE.BS_AUTOCHECKBOX 源码 #include<Windows.h> #include<Windowsx.h ...

  6. windows RabbitMQ Server 环境配置中的一些坑

    原文:https://blog.csdn.net/lindonglian/article/details/55805637 RabbitMQ的服务端基于Erlang语言编写,要在机器上安装Rabbit ...

  7. erase & remove_if 合用

    words_.erase( remove_if( words_.begin(), words_.end(), [&](const entry& e) { return (e.type ...

  8. ubuntu下新立得(synaptic)软件包管理器安装

    1.从ubuntu下的软件中心(面板主页中输入soft即可找到)搜索安装synaptic后,打开新立得一闪就自动关了.解决办法为: 1.1命令行下卸载,命令行下重新安装: 卸载: #purge表示卸载 ...

  9. jquery统计输入文字的个数并对其进行判断

    <textarea placeholder="该产品满足你的期待吗?说说你的使用心得,分享给 同样看中的他们吧"></textarea> <span ...

  10. jmeter HTTP请求之content-type

    对于初次接触接口的同学来说,自己在发送一个http请求时,总会遇到这样那样的问题,比如必传参数不存在啊 出现这样类似问题的问题首先排除的应该是content-type是否正确,那什么是content- ...