一道有趣的函數式建模練習

来源:http://www.cnblogs.com/magicbowen/archive/2016/05/26/5529523.html
-Advertisement-
Play Games

一道有趣的函數式建模練習 領域建模在於不斷挖掘領域的本質,然後用優秀的代碼簡潔地表現出來。而函數式非常適合將領域映射到數學本質上。前一陣子學習erlang,用erlang做了一些練習,分享其中的一個。 practice 1 : count tiangle 題目: 數數如下圖形中一共包含多少三角形? ...


一道有趣的函數式建模練習

領域建模在於不斷挖掘領域的本質,然後用優秀的代碼簡潔地表現出來。而函數式非常適合將領域映射到數學本質上。前一陣子學習erlang,用erlang做了一些練習,分享其中的一個。

practice 1 : count tiangle

題目: 數數如下圖形中一共包含多少三角形?

答案是24,你數對了嗎?回憶一下你剛纔是怎麼數的?

這道題如果由人來數,每個人數法各異!但是如果要交給電腦來做,就必須將數的方法描述成電腦可以執行的形式化演算法。這種描述方法可以有非常多種,而我們希望找到一套抽象層次高的和領域非常貼切的描述,以降低後續理解、維護成本。

對於該問題,我們首先站在領域角度定義什麼是三角形?

triangle([A, B, C]) ->
    connected(A, B) andalso
    connected(B, C) andalso
    connected(C, D) andalso
    (not in_same_line(A, B, C)).

如上,我們定義了一個三角形就是三個點,兩兩相連,但是三個點不同時在一條直線上。

對於如上描述,關鍵是如何定義connectedin_same_line
對於connected,就是兩個點同時在一條直線連接上。那麼什麼是一條直線連接?
由於我們關註的是點線上上的關係,所以我們定義一條直線為在直線上所有點得集合。
例如對以上例,我們存在直線:[a, b, h],[a, c, g, i]等等。

我們把上圖中所有直線用erlang描述出來:

lines() ->
    ["abh", "acgi", "adfj", "aek", "bcde", "hgfe", "hijk"].

由於我們用單字元表示點,而字元串在erlang中實際就是list,所以我們將一條直線簡寫為線上上所有點的字元的字元串。

有了對直線的定義,接下來,一個點是否在直線上,那就是元素與集合的屬於關係;而兩個點是否相連就是集合與集合之間的包含關係。

subset([], _S) -> true;
subset([H|T], S) -> 
    lists:member(H, S) andalso subset(T, S).

belong(_, []) -> false;
belong(S, [H|T]) -> subset(S, H) orelse belong(S, T).

connected(P1, P2) -> belong([P1, P2], lines()).

in_same_line(P1, P2, P3) -> belong([P1, P2, P3], lines()).

可以看到,connected的定義是兩個點組成的集合屬於所有直線的集合的任一個的子集。而in_same_line則是三個點的集合屬於所有直線的集合的任一個的子集。
我們在這裡將該問題映射到熟悉的集合領域。

在有了對connectedin_same_line的定以後,我們就可以對triangle進行測試了!

test() ->
    true = triangle("abc"),
    false = triangle("abh").

下來我們來進行數三角形。要能夠數三角形,我們需要找到所有三個點的組合,用來驗證是否滿足triangle?將滿足triangle的進行統計,這樣我們就得到了結果!

在這裡我們已經有了所有點的集合:

Points = "abcdefghijk"

為了得到所有3個點的組合,我們實現一個演算法,對於集合L,求其N個元素的所有組合的集合。

comb(L, 1) -> [[I] || I <- L];
comb(L, N) when length(L) =:= N -> [L];
comb([H|T], N) -> 
    [[H|R] || R <- comb(T, N - 1)] ++ comb(T, N).
Points = "abcdefghijk",
TriplePoints = comb(Points, 3),

下麵我們實現一個count方法,用來數滿足要求的三角形個數:

count(Triple) -> count(Triple, 0).

count([], N) -> N;
count([H|T], N) ->
    case triangle(H) of
        true -> count(T, N + 1);
        false -> count(T, N)
    end.

最後可以調用run測試一下是不是24!

run() ->
    Points = "abcdefghijk",
    TriplePoints = comb(Points, 3),
    count(TriplePoints).

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 解釋器模式: 給定一個語言, 定義它的文法的一種表示,並定義一個解釋器,該解釋器使用該表示來解釋語言中的句子。 角色: 環境角色:定義解釋規則的全局信息。 抽象解釋器::定義了部分解釋具體實現,封裝了一些由具體解釋器實現的介面。 具體解釋器(MusicNote):實現抽象解釋器的介面,進行具體的解釋 ...
  • 1.意圖 將抽象部分與它的實現部分分離,使它們都可以獨立地變化。 2.動機 在抽象類與它的實現之間起到橋梁作用,使它們可以獨立地變化。 3.適用性 不希望在抽象和它的實現部分之間有一個固定的綁定關係。這種情況可能是因為,在程式運行時刻實現部分可以被選擇或切換。 類的抽象以及它的實現部分都應該可以通過 ...
  • 今天看到一篇講解設計模式六大原則的文章,非常深刻細緻,轉過來給大家共同學習。 作者:zhengzhb ,發佈於2012-11-2,來源:CSDN 作者:zhengzhb ,發佈於2012-11-2,來源:CSDN 設計模式六大原則(1):單一職責原則 定義:不要存在多於一個導致類變更的原因。通俗的說 ...
  • 代理模式和裝飾模式有很大的相似性,二者的類圖(幾乎)是一樣的。下麵分別講解代理模式和裝飾模式。 1、代理模式 一般著名的跑步運動員都會有自己的代理人,如果想聯繫該運動員的比賽事宜,可以直接聯繫他的代理人就可以了。類圖如下所示: IRunner介面如下: Runner類如下所示: RunnerAgen ...
  • 外觀模式: 外觀模式(Facade Pattern):外部與一個子系統的通信必須通過一個統一的外觀對象進行,為子系統中的一組介面提供一個一致的界面,外觀模式定義了一個高層介面,這個介面使得這一子系統更加容易使用。外觀模式又稱為門面模式,它是一種對象結構型模式。 目的: 1、為一個複雜子系統提供簡單的 ...
  • 裝飾模式(Decorator Pattern) : 動態地給一個對象增加一些額外的職責(Responsibility),就增加對象功能來說,裝飾模式比生成子類實現更為靈活。其別名也可以稱為包裝器(Wrapper),與適配器模式的別名相同,但它們適用於不同的場合。根據翻譯的不同,裝飾模式也有人稱之為“ ...
  • 1. 概述 將一個類的介面轉換成客戶希望的另外一個介面。Adapter模式使得原本由於介面不相容而不能一起工作的那些類可以在一起工作。 2. 解決的問題 即Adapter模式使得原本由於介面不相容而不能一起工作的那些類可以在一起工作。 3. 模式中的角色 3.1 目標介面(Target):客戶所期待 ...
  • OOA - Object-Oriented Analysis(面向對象分析) OOT - Object-Oriented Testing (面向對象測試) OOP - Object-Oriented Programming (面象對象編程) OOD - Object-Oriented Design( ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...