草庐IT

java - "possible lossy conversion"是什么意思,我该如何解决?

coder 2024-03-15 原文

新的 Java 程序员经常对编译错误消息感到困惑,例如:

"incompatible types: possible lossy conversion from double to int"



对于这行代码:
int squareRoot = Math.sqrt(i);

一般而言,“可能的有损转换”错误消息是什么意思,如何解决?

最佳答案

首先,这是一个编译错误。如果您在运行时在异常消息中看到它,那是因为您运行的程序有编译错误1。
消息的一般形式是这样的:

"incompatible types: possible lossy conversion from <type1> to <type2>"


哪里<type1><type2>都是原始数字类型;即 byte 之一, char , short , int , long , floatdouble .
当您的代码尝试从 <type1> 进行隐式转换时会发生此错误至 <type2>但转换可能是有损的。
在问题中的示例中:
  int squareRoot = Math.sqrt(i);
sqrt方法产生一个 double ,但来自 double 的转换至 int可能有损。
“潜在有损”是什么意思?
好吧,让我们看几个例子。
  • long 的转换到 int是一个潜在的有损转换,因为有 long没有对应的值 int值(value)。例如,任何 long大于 2^31 - 1 的值太大而无法表示为 int .同样,任何小于 -2^31 的数字都太小了。
  • int 的转换到 long不是有损转换,因为每个 int value 有一个对应的 long值(value)。
  • float 的转换到 long是潜在的有损转换,因为存在 float太大或太小而无法表示为 long 的值值。
  • long 的转换到 float不是有损转换,因为每个 long value 有一个对应的 float值(value)。 (转换后的值可能不太精确,但“损失”并不意味着……在这种情况下。)

  • 这些是所有可能有损的转换:
  • shortbytechar
  • charbyteshort
  • intbyte , shortchar
  • longbyte , short , charint
  • floatbyte , short , char , intlong
  • doublebyte , short , char , int , longfloat .

  • 你如何修复错误?
    消除编译错误的方法是添加类型转换。例如;
      int i = 47;
      int squareRoot = Math.sqrt(i);         // compilation error!
    
    变成
      int i = 47;
      int squareRoot = (int) Math.sqrt(i);   // no compilation error
    
    但这真的是修复吗?考虑 47 的平方根是 6.8556546004 ... 但是 squareRoot将得到值 6 . (转换将截断,而不是舍入。)
    而这个呢?
      byte b = (int) 512;
    
    结果是 b获取值 0 .从较大的 int 类型转换为较小的 int 类型是通过屏蔽掉 512 的高位和低 8 位来完成的。都是零。
    简而言之,您不应该简单地添加类型转换,因为它可能不会为您的应用程序做正确的事情。
    相反,您需要了解为什么您的代码需要进行转换:
  • 这是因为您在代码中犯了其他错误吗?
  • <type1>是不同的类型,所以这里不需要有损转换?
  • 如果需要进行转换,类型转换是否会执行正确的行为的无声有损转换?
  • 或者您的代码是否应该进行一些范围检查并通过抛出异常来处理不正确/意外的值?

  • 下标时“可能的有损转换”。
    第一个例子:
    for (double d = 0; d < 10.0; d += 1.0) {
        System.out.println(array[d]);  // <<-- possible lossy conversion
    }
    
    这里的问题是数组索引值必须是 int .所以d必须从 double 转换而来至 int .通常,使用浮点值作为索引没有意义。要么有人认为 Java 数组的工作方式类似于(比如)Python 字典,要么他们忽略了浮点运算通常不准确的事实。
    解决方案是重写代码以避免使用浮点值作为数组索引。 (添加类型转换可能是一个不正确的解决方案。)
    第二个例子:
    for (long l = 0; l < 10; l++) {
        System.out.println(array[l]);  // <<-- possible lossy conversion
    }
    
    这是上一个问题的变体,解决方法是一样的。不同之处在于根本原因是 Java 数组仅限于 32 位索引。如果您想要一个具有超过 231 - 1 个元素的“类似数组”的数据结构,您需要定义或找到一个类来完成它。
    方法或构造函数调用中的“可能的有损转换”
    考虑一下:
    public class User {
        String name;
        short age;
        int height;
    
        public User(String name, short age, int height) {
            this.name = name;
            this.age = age;
            this.height = height;
        }
    
        public static void main(String[] args) {
            User user1 = new User("Dan", 20, 190);
        }
    }
    
    使用 Java 11 编译上面的代码会得到以下结果:
    $ javac -Xdiags:verbose User.java 
    User.java:20: error: constructor User in class User cannot be applied to given types;
        User user1 = new User("Dan", 20, 190);
                     ^
      required: String,short,int
      found: String,int,int
      reason: argument mismatch; possible lossy conversion from int to short
    1 error
    
    问题在于文字 20int ,并且构造函数中的相应参数声明为 short .转换 intshort是有损的。
    分配文字时“可能的有损转换”
    考虑一下:
    int a = 21;
    byte b1 = a;   // <<-- possible lossy conversion
    byte b2 = 21;  // OK
    
    到底是怎么回事?为什么允许一个版本而另一个版本不允许? (毕竟他们“做”同样的事情!)
    首先,JLS 声明 21是类型为 int 的数字文字. (没有 byteshort 文字。)所以在这两种情况下,我们都分配了一个 intbyte .
    第一种情况,报错的原因是不是所有int值将适合 byte .
    在第二种情况下,编译器知道 21是一个总是适合 byte 的值.
    技术上的解释是,在赋值上下文中,允许对 byte 执行原始收缩转换。 , charshort如果以下全部为真:
  • 该值是编译时常量表达式(包括文字)的结果。
  • 表达式的类型是 byte , short , charint .
  • 被分配的常量值在“目标”类型的域中是可表示的(没有损失)。

  • 请注意,这仅适用于赋值语句,或更技术性地适用于赋值上下文。因此:
    Byte b4 = new Byte(21);  // incorrect
    
    给出编译错误。

    1 - 例如,Eclipse IDE 有一个选项,允许您忽略编译错误并无论如何运行代码。如果选择此项,IDE 的编译器将创建一个 .class带有错误的方法如果被调用将抛出未经检查的异常的文件。异常消息将提及编译错误消息。

    关于java - "possible lossy conversion"是什么意思,我该如何解决?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51632152/

    有关java - "possible lossy conversion"是什么意思,我该如何解决?的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

    2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    4. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    5. ruby-on-rails - rails : "missing partial" when calling 'render' in RSpec test - 2

      我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou

    6. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

      我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

    7. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

      给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

    8. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

      我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

    9. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

      我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

    10. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    随机推荐