将十进制转换为双精度


672

我想使用Track-Bar来更改Form的不透明度。

这是我的代码:

decimal trans = trackBar1.Value / 5000;
this.Opacity = trans;

当我构建应用程序时,它给出以下错误:

无法将类型隐式转换decimaldouble

我已经尝试过使用transdouble但随后Control却无法正常工作。该代码在过去的VB.NET项目中运行良好。


11
另外,十进制不能代表Double那样宽的值。小数最多只能达到+/- 7.9228162514264337593543950950335E + 28; 而Double最高可达+/- 1.79769313486232E + 308
TraumaPony


8
有人应将此标为重复。
伊万

8
@Ivan:这是迄今为止提出的第四个问题……
Nikolas

1
@Nikolas:的确如此。今天追溯到这里。

Answers:


447

double不需要像这样的显式强制转换:

double trans = (double) trackBar1.Value / 5000.0;

将常量标识为5000.0(或as 5000d)就足够了:

double trans = trackBar1.Value / 5000.0;
double trans = trackBar1.Value / 5000d;

123

通用问题“十进制与双精度?”的更通用答案:十进制用于货币计算以保持精度,精度用于科学计算,不受小差异的影响。由于Double是CPU固有的类型(内部表示形式存储在基数2中),因此使用Double进行的计算要比Decimal(内部表示为基数10)更好。


83

您的代码在VB.NET中可以正常工作,因为它隐式地执行任何强制转换,而C#同时具有隐式和显式强制转换。

在C#中,由于精度下降,从十进制到双精度的转换是明确的。例如,1.1不能精确地表示为双精度数,而可以精确地表示为十进制数(请参阅“ 浮点数-比您想象的更不准确 ”,以了解原因)。

在VB中,编译器为您添加了转换:

decimal trans = trackBar1.Value / 5000m;
this.Opacity = (double) trans;

(double)必须在C#中明确声明,但是可以由VB的“更宽容”的编译器隐含


80

为什么要除以5000?只需将TrackBar的最小值和最大值设置为0到100之间,然后将不透明度百分比的值除以100。下面的最少20个示例可防止表单变得完全不可见:

private void Form1_Load(object sender, System.EventArgs e)
{
    TrackBar1.Minimum = 20;
    TrackBar1.Maximum = 100;

    TrackBar1.LargeChange = 10;
    TrackBar1.SmallChange = 1;
    TrackBar1.TickFrequency = 5;
}

private void TrackBar1_Scroll(object sender, System.EventArgs e)
{
    this.Opacity = TrackBar1.Value / 100;
}

5
这难道不能解决问题吗?5000OP可能会出现问题,而不是出现问题100
jww

62

你有两个问题。首先,Opacity需要一个双精度值,而不是十进制值。编译器告诉您,虽然在十进制和双精度之间进行了转换,但这是您需要指定的显式转换,以使其起作用。第二个是TrackBar.Value整数值,将int除以int都会得到int,无论您将变量分配给哪种类型。在这种情况下,存在从int到十进制或double的隐式转换-因为执行转换时不会损失精度-因此编译器不会抱怨,但是您得到的值始终为0,大概是因为trackBar.Value始终小于5000。解决方案是将代码更改为使用double(Opacity的本机类型),并通过将常量显式设置为double来进行浮点算术-这将促进算术-或强制转换trackBar.Value为double ,它们会做同样的事情-或两者都做。哦,除非您在其他地方使用中间变量,否则您不需要中间变量。我的猜测是,无论如何编译器都会对其进行优化。

trackBar.Opacity = (double)trackBar.Value / 5000.0;

58

我认为,希望尽可能明确。这增加了代码的清晰度,并帮助可能最终阅读该代码的其他程序员。

除了(或代替).0在数字后面加上,您还可以使用decimal.ToDouble()

这里有些例子:

// Example 1
double transperancy = trackBar1.Value/5000;
this.Opacity = decimal.ToDouble(transperancy);

// Example 2 - with inline temp
this.Opacity = decimal.ToDouble(trackBar1.Value/5000);


50

不透明度属性为双类型:

double trans = trackBar1.Value / 5000.0;
this.Opacity = trans;

或者简单地:

this.Opacity = trackBar1.Value / 5000.0;

要么:

this.Opacity = trackBar1.Value / 5000d;

请注意,我正在使用5000.0(或5000d)强制执行双除法,因为trackBar1.Value它是整数,它将执行整数除法,结果将是整数。



47

假设您使用的是WinForms,Form.Opacity类型为double,那么您应该使用:

double trans = trackBar1.Value / 5000.0;
this.Opacity = trans;

除非您在其他地方需要该值,否则编写起来会更简单:

this.Opacity = trackBar1.Value / 5000.0;

当您将代码更改为简单的两倍时,该控件不起作用的原因是:

double trans = trackbar1.Value / 5000;

将其解释5000为整数,并且由于trackbar1.Value也是整数,您的trans值始终为零。通过将数字加一个显式的浮点值,.0编译器现在可以将其解释为双精度型并执行适当的计算。



41

因为Opacity是double值,所以从一开始就只使用double而不是强制转换,但是在除法时一定要使用double,这样就不会损失任何精度

Opacity = trackBar1.Value / 5000.0;


0

尝试这个:

Opacity = decimal.ToDouble(trackBar1.Value / 5000.0);```
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.