成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

一個(gè)有趣的算法問(wèn)題:如何定義一個(gè)分?jǐn)?shù)類(lèi)

BearyChat / 1802人閱讀

摘要:一個(gè)來(lái)自于程序設(shè)計(jì)的經(jīng)典問(wèn)題。注意事項(xiàng)負(fù)數(shù)問(wèn)題。和上一點(diǎn)是一樣的問(wèn)題,要確定方式是屬于具體的對(duì)象,還是屬于一個(gè)類(lèi)。

一個(gè)來(lái)自于C++程序設(shè)計(jì)的經(jīng)典問(wèn)題。如何定義一個(gè)分?jǐn)?shù)類(lèi),實(shí)現(xiàn)分?jǐn)?shù)的約分化簡(jiǎn),分?jǐn)?shù)之間的加法、減法、乘法、除法四則運(yùn)算?

1.初見(jiàn)

剛看到這道題的時(shí)候,第一感覺(jué)是挺簡(jiǎn)單的啊,就是基本的面向?qū)ο螅x對(duì)應(yīng)的加減乘除類(lèi)就可以了啊,然而到了實(shí)現(xiàn)的時(shí)候才發(fā)現(xiàn)許多問(wèn)題是說(shuō)起來(lái)容易做起來(lái)難,在實(shí)現(xiàn)的過(guò)程中,發(fā)現(xiàn)了許多的注意點(diǎn),以及算法。

最終得出結(jié)論:這個(gè)問(wèn)題著實(shí)是考察程序員基本功的一道好題。

2.整體思路

分?jǐn)?shù)類(lèi)設(shè)計(jì)的總體思路如下:

首先是分?jǐn)?shù)的表示,這就需要利用兩個(gè)變量保存分?jǐn)?shù)的分子和分母;

其次是約分和通分,由于分?jǐn)?shù)四則運(yùn)算中需要借助約分和通分來(lái)實(shí)現(xiàn),因此必須先考慮實(shí)現(xiàn)這兩個(gè)算法。

加法和乘法的實(shí)現(xiàn)。利用約分和通分就可以輕松實(shí)現(xiàn)。

減法和除法的實(shí)現(xiàn)。是加法和乘法的逆運(yùn)算。和直接轉(zhuǎn)化為加法和乘法。

3.注意事項(xiàng)

負(fù)數(shù)問(wèn)題。這個(gè)問(wèn)題十分重要,在我們的算法中,都規(guī)定分母為正數(shù),如果出現(xiàn)了分母為負(fù)數(shù)的情況,就分子分母同時(shí)乘以-1,把負(fù)數(shù)運(yùn)算放在分子上。

函數(shù)的副作用(side effect)。盡量不要在方法中改變?cè)瓉?lái)分?jǐn)?shù)的值,否則會(huì)產(chǎn)生副作用,導(dǎo)致后面的運(yùn)算出錯(cuò),在代碼中會(huì)說(shuō)明。

靜態(tài)方法和動(dòng)態(tài)方法。和上一點(diǎn)是一樣的問(wèn)題,要確定方式是屬于具體的對(duì)象,還是屬于一個(gè)類(lèi)。

除法運(yùn)算中,除數(shù)不能等于0

4.代碼實(shí)現(xiàn) 4.1 屬性和構(gòu)造方法

構(gòu)造方法中有三個(gè)細(xì)節(jié):一是使用了參數(shù)默認(rèn)值,默認(rèn)不寫(xiě)參數(shù)時(shí),讓分子分母都等于1;二是在構(gòu)造方法中進(jìn)行了分母合法性的驗(yàn)證,分母等于0時(shí)直接返回錯(cuò)誤信息;三是對(duì)負(fù)值進(jìn)行了處理,使分?jǐn)?shù)的分母永遠(yuǎn)為正數(shù),方便后續(xù)運(yùn)算。

/**
* 分?jǐn)?shù)運(yùn)算類(lèi)
*/
class Fraction 
{
    //定義分子和分母
    public $fenzi;
    public $fenmu;
    
    //構(gòu)造函數(shù)
    function __construct($fenzi = 1,$fenmu = 1)
    {
        if ($fenmu == 0) {
            return "分母不能為0";
        }
        if ($fenmu < 0) {
            $fenmu = -$fenmu;
            $fenzi = -$fenzi;
        }
        $this->fenzi = $fenzi;
        $this->fenmu = $fenmu;
    }
}
4.2 最大公約數(shù)和最小公倍數(shù)

為了后續(xù)的約分和通分,必須先求出最大公約數(shù)和最小公倍數(shù)。求最大公約數(shù)采用輾轉(zhuǎn)相除法,而最小公倍數(shù)由以下公式可求:

最小公倍數(shù) = (數(shù)A * 數(shù)B)/ 最大公約數(shù)

//求最大公約數(shù)用于約分
private static function _getmax($a, $b)
{
    if($a < 0) $a = -$a;
    if($b < 0) $b = -$b;
    $tmp = $a % $b;
    while($tmp != 0) {
        $a = $b;
        $b = $tmp;
        $tmp = $a % $b;
    }
    return $b;
}

//求最小公倍數(shù)用于通分
private static function _getmin($a, $b)
{
    if($a < 0) $a = -$a;
    if($b < 0) $b = -$b;   
    $max = self::_getmax($a,$b);
    $min = intval(($a * $b) / $max);
    return $min;
}

/**
 * 約分運(yùn)算,基本算法為分子分母同時(shí)除以最大公約數(shù);
 * @return void 將對(duì)象的分子分母約分為最簡(jiǎn)形式    
 */
public function reduction()
{
    $max = $this->_getmax($this->fenzi,$this->fenmu);
    $this->fenzi = intval($this->fenzi / $max);
    $this->fenmu = intval($this->fenmu / $max);
}

這兩個(gè)方法全部都定義為靜態(tài)私有方法,只在類(lèi)內(nèi)調(diào)用且不需實(shí)例化。求最大公約數(shù)和最小公倍數(shù)的算法其實(shí)還有很多種,@燼醬采用了另外一種方法,C++代碼如下:

/**
 * 求最大公約數(shù)并進(jìn)行約分
 * @return void
 */
int reduction()
{
    int i,comdiv,small,max;

    if(above1;i--)
    {
        if(small%i==0 &max%i==0 )
            break;
    }

    comdiv=i; //最大公約數(shù)
    if(i!=0)
    {
        above/=i;
        below/=i;
    }
    return 0;
}

這種方法的本質(zhì)就窮盡法,核心思想在于for循環(huán)當(dāng)中。同樣的,也可用此法求最小公倍數(shù)。

4.3 分?jǐn)?shù)加減

分?jǐn)?shù)加法的算法如下:

/**
 * 加法運(yùn)算,寫(xiě)成靜態(tài)方法,需要傳遞兩個(gè)分?jǐn)?shù)對(duì)象實(shí)例。加法的基本步驟為:
 * 1. 求兩個(gè)分母的最小公倍數(shù);
 * 2. 利用最小公倍數(shù)進(jìn)行通分,此時(shí)分母就是最小公倍數(shù),第一個(gè)分?jǐn)?shù)的分子等于原來(lái)的分子*(最小公倍數(shù)/原來(lái)的分母),第二個(gè)分?jǐn)?shù)的分子同理;
 * 3. 分母不變,分子相加;
 * 4. 對(duì)結(jié)果進(jìn)行約分;
 * 
 * @param  fraction $fra1 分?jǐn)?shù)相加的加數(shù)1
 * @param  fraction $fra2 分?jǐn)?shù)相加的加數(shù)2  
 * @return fraction  $fra 分?jǐn)?shù)相加的計(jì)算結(jié)果
 */
public static function add($fra1, $fra2)
{
    $fra = new Fraction();
    $min = self::_getmin($fra1->fenmu,$fra2->fenmu);
    $fenzi_left = $fra1->fenzi * ($min / $fra1->fenmu);
    $fenzi_right = $fra2->fenzi * ($min / $fra2->fenmu);

    $fra->fenmu = $min;
    $fra->fenzi = $fenzi_left + $fenzi_right;
    $fra->reduction();
    return $fra;
}

/**
 * 減法運(yùn)算,加法的逆運(yùn)算,只需要將參數(shù)$fra2的分子取反,將減法運(yùn)算化為加法運(yùn)算
 * 
 * @param  fraction $fra1 分?jǐn)?shù)相減的被減數(shù)
 * @param  fraction $fra2 分?jǐn)?shù)相減的減數(shù)
 * @return fraction $fra 分?jǐn)?shù)相減的計(jì)算結(jié)果
 */
public static function minus($fra1, $fra2)
{
    $fra_t = new Fraction(-$fra2->fenzi,$fra2->fenmu);
    return self::add($fra1,$fra_t);
}

在上述算法中,定義了兩個(gè)靜態(tài)方法,每個(gè)方法需要傳入兩個(gè)分?jǐn)?shù)對(duì)象,之后就可以按上面的算法步驟進(jìn)行加法和減法運(yùn)算了。其中減法運(yùn)算只需要轉(zhuǎn)換為加法即可。需要注意的是,在減法運(yùn)算中,存在兩種可能的寫(xiě)法:

【寫(xiě)法1】

$fra2->fenzi = -$fra2->fenzi;

【寫(xiě)法2】

$fra_t = new Fraction(-$fra2->fenzi,$fra2->fenmu);

其中,第一種寫(xiě)法直接改變了減數(shù)分子的值,這里對(duì)減法本身的結(jié)果不會(huì)造成影響,表面上看是成立的,但其實(shí)這種寫(xiě)法產(chǎn)生了副作用,在計(jì)算乘法時(shí),fra2就已經(jīng)不是最初的分?jǐn)?shù)值了,因此我們需要new一個(gè)新的對(duì)象,如寫(xiě)法2所示,這樣就不會(huì)產(chǎn)生副作用改變分?jǐn)?shù)2的值。

4.4 分?jǐn)?shù)乘除

分?jǐn)?shù)乘數(shù)就比較簡(jiǎn)單了,如下所示:

/**
 * 乘法運(yùn)算,分子相乘,分母相乘之后再約分
 * 
 * @param  fraction $fra1 分?jǐn)?shù)相乘的乘數(shù)1
 * @param  fraction $fra2 分?jǐn)?shù)相乘的乘數(shù)2
 * @return fraction $fra 分?jǐn)?shù)相乘的計(jì)算結(jié)果
 */
public static function multiply($fra1,$fra2)
{
    $fra = new Fraction();

    $fenzi = $fra1->fenzi * $fra2->fenzi;
    $fenmu = $fra1->fenmu * $fra2->fenmu;

    $fra->fenzi = $fenzi;
    $fra->fenmu = $fenmu;
    $fra->reduction();
    return $fra;
}

/**
 * 除法運(yùn)算,乘法運(yùn)算的逆運(yùn)算,只需要將參數(shù)$fra2的分子分母調(diào)換,將除法運(yùn)算化為乘法運(yùn)算
 * 
 * @param  fraction $fra1 分?jǐn)?shù)相除的被除數(shù)
 * @param  fraction $fra2 分?jǐn)?shù)相除的除數(shù)
 * @return fraction       分?jǐn)?shù)相除的計(jì)算結(jié)果
 */
public static function divide($fra1,$fra2)
{
    $fra_t = new Fraction($fra2->fenmu,$fra2->fenzi);
    $fra = self::multiply($fra1,$fra_t);
    return $fra;
}
4.5 分?jǐn)?shù)的表示

最后,我們需要寫(xiě)一個(gè)方法,把分?jǐn)?shù)以a/b的形式打印出來(lái)。

public function display()
{
    printf("%d/%d
",$this->fenzi,$this->fenmu);
}

這樣我們的分?jǐn)?shù)類(lèi)的定義完了。

5. 結(jié)果展示

下面展示運(yùn)行結(jié)果,先寫(xiě)一個(gè)調(diào)用:

$fra1 = new Fraction(3,4);
$fra1->display();

$fra2 = new Fraction(12,20);
$fra2->reduction();
$fra2->display();

$fra3 = Fraction::add($fra1,$fra2);
$fra3->display();

$fra4 = Fraction::minus($fra1,$fra2);
$fra4->display();

$fra5 = Fraction::multiply($fra1,$fra2);
$fra5->display();

$fra6 = Fraction::divide($fra1,$fra2);
$fra6->display();

如上,fra1期望直接打印出3/4, fra2對(duì)12/20先約分再輸出,期望是3/5,fra3是計(jì)算fra1和fra2的加法,fra4為減法,fra5為乘法,fra6為除法。結(jié)果如下所示:

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/30348.html

相關(guān)文章

  • 「正經(jīng)字幕」太無(wú)聊?「神經(jīng)玩笑機(jī)」就可以生成逗你笑趣味字幕

    摘要:最后,我們顯示了若干張圖像中所生成的趣味字幕。圖所提出的有趣字幕生成的體系結(jié)構(gòu)。我們將所提出的方法稱(chēng)為神經(jīng)玩笑機(jī)器,它是與預(yù)訓(xùn)練模型相結(jié)合的。用戶(hù)對(duì)已發(fā)布的字幕的趣味性進(jìn)行評(píng)估,并為字幕指定一至三顆星。 可以毫不夸張地說(shuō),笑是一種特殊的高階功能,且只有人類(lèi)才擁有。那么,是什么引起人類(lèi)的笑聲表達(dá)呢?最近,日本東京電機(jī)大學(xué)(Tokyo Denki University)和日本國(guó)家先進(jìn)工業(yè)科學(xué)和技...

    lastSeries 評(píng)論0 收藏0
  • 六種GAN評(píng)估指標(biāo)綜合評(píng)估實(shí)驗(yàn),邁向定量評(píng)估GAN重要一步

    摘要:本文討論了多個(gè)評(píng)估指標(biāo),并從多個(gè)方面對(duì)評(píng)估指標(biāo)進(jìn)行了實(shí)驗(yàn)評(píng)估,包括距離分類(lèi)器。鑒于定性評(píng)估的內(nèi)在缺陷,恰當(dāng)?shù)亩吭u(píng)估指標(biāo)對(duì)于的發(fā)展和更好模型的設(shè)計(jì)至關(guān)重要。鑒于評(píng)估非常有難度,評(píng)估評(píng)估指標(biāo)則更加困難。 作者:Qiantong Xu、Gao Huang、Yang Yuan、Chuan Guo、Yu Sun、Felix Wu、Kilian Weinberger生成對(duì)抗網(wǎng)絡(luò)的評(píng)估目前仍以定性評(píng)估和...

    zorro 評(píng)論0 收藏0
  • Minimax 和 Alpha-beta 剪枝算法簡(jiǎn)介,以及以此實(shí)現(xiàn)井字棋游戲(Tic-tac-t

    摘要:我們?cè)谇拔闹锌紤]的那張圖就來(lái)自這篇文章,之后我們會(huì)用剪枝算法來(lái)改進(jìn)之前的解決方案。剪枝算法的實(shí)現(xiàn)接下來(lái)討論如何修改前面實(shí)現(xiàn)的算法,使其變?yōu)榧糁λ惴ā,F(xiàn)在我們已經(jīng)有了現(xiàn)成的和剪枝算法,只要加上一點(diǎn)兒細(xì)節(jié)就能完成這個(gè)游戲了。 前段時(shí)間用 React 寫(xiě)了個(gè)2048 游戲來(lái)練練手,準(zhǔn)備用來(lái)回顧下 React 相關(guān)的各種技術(shù),以及試驗(yàn)一下新技術(shù)。在寫(xiě)這個(gè)2048的過(guò)程中,我考慮是否可以在其中加...

    wemall 評(píng)論0 收藏0
  • Minimax 和 Alpha-beta 剪枝算法簡(jiǎn)介,以及以此實(shí)現(xiàn)井字棋游戲(Tic-tac-t

    摘要:我們?cè)谇拔闹锌紤]的那張圖就來(lái)自這篇文章,之后我們會(huì)用剪枝算法來(lái)改進(jìn)之前的解決方案。剪枝算法的實(shí)現(xiàn)接下來(lái)討論如何修改前面實(shí)現(xiàn)的算法,使其變?yōu)榧糁λ惴ā,F(xiàn)在我們已經(jīng)有了現(xiàn)成的和剪枝算法,只要加上一點(diǎn)兒細(xì)節(jié)就能完成這個(gè)游戲了。 前段時(shí)間用 React 寫(xiě)了個(gè)2048 游戲來(lái)練練手,準(zhǔn)備用來(lái)回顧下 React 相關(guān)的各種技術(shù),以及試驗(yàn)一下新技術(shù)。在寫(xiě)這個(gè)2048的過(guò)程中,我考慮是否可以在其中加...

    Eirunye 評(píng)論0 收藏0
  • 推薦系統(tǒng)02--協(xié)同過(guò)濾

    摘要:如果做推薦系統(tǒng)不知道基于物品的協(xié)同過(guò)濾,那等同于做程序員不懂得冒泡排序?;谖锲返陌素曰谖锲返膮f(xié)同過(guò)濾算法誕生于年,是由亞馬遜首先提出的,并在年由其發(fā)明者發(fā)表了相應(yīng)的論文。 不管你有沒(méi)有剁過(guò)手,你對(duì)看了這個(gè)商品的還看了這樣的推薦形式一定不陌生。無(wú)論是貓還是狗,或者是其他電商網(wǎng)站,這樣的推薦產(chǎn)品可以說(shuō)是推薦系統(tǒng)的標(biāo)配了。 類(lèi)似的還有,如點(diǎn)評(píng)標(biāo)記類(lèi)網(wǎng)站的喜歡了這部電影的還喜歡了,社交媒...

    jaysun 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<