惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

SecWiki News
SecWiki News
I
InfoQ
The Cloudflare Blog
人人都是产品经理
人人都是产品经理
博客园 - Franky
T
Tailwind CSS Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
量子位
博客园_首页
罗磊的独立博客
V
V2EX
李成银的技术随笔
大猫的无限游戏
大猫的无限游戏
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
T
True Tiger Recordings
Vercel News
Vercel News
Cyberwarzone
Cyberwarzone
Cisco Talos Blog
Cisco Talos Blog
F
Fox-IT International blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
M
Microsoft Research Blog - Microsoft Research
Know Your Adversary
Know Your Adversary
爱范儿
爱范儿
The Register - Security
The Register - Security
G
Google Developers Blog
The Hacker News
The Hacker News
Malwarebytes
Malwarebytes
S
Securelist
博客园 - 三生石上(FineUI控件)
Jina AI
Jina AI
T
Threat Research - Cisco Blogs
T
The Exploit Database - CXSecurity.com
S
SegmentFault 最新的问题
博客园 - 叶小钗
F
Fortinet All Blogs
Apple Machine Learning Research
Apple Machine Learning Research
宝玉的分享
宝玉的分享
博客园 - 聂微东
T
Threatpost
博客园 - 【当耐特】
D
Docker
P
Privacy & Cybersecurity Law Blog
www.infosecurity-magazine.com
www.infosecurity-magazine.com
G
GRAHAM CLULEY
V
Visual Studio Blog
C
Cisco Blogs
IT之家
IT之家
S
Security Archives - TechRepublic
Latest news
Latest news
阮一峰的网络日志
阮一峰的网络日志

Nic Lin's Blog

謝明真 - 高效領導力的課後筆記 NFT 開發實戰!基礎智能合約入門 (3) NFT 開發實戰!基礎智能合約入門 (2) NFT 開發實戰!基礎智能合約入門 (1) 如何自我檢測 log4j CVE 漏洞 Rails 如何在資料寫入時記錄來源 IP 位置 如何經營工程師 Youtube 頻道 - Part 8 營收篇 如何經營工程師 Youtube 頻道 - Part 7 酸民文化篇 如何經營工程師 Youtube 頻道 - Part 6 演算法趨勢篇 如何經營工程師 Youtube 頻道 - Part 5 設備器材篇 如何經營工程師 Youtube 頻道 - Part 4 後製剪輯篇 如何經營工程師 Youtube 頻道 - Part 3 文案企劃篇 如何經營工程師 Youtube 頻道 - Part 2 設備器材篇 如何經營工程師 Youtube 頻道 - Part 1 制訂頻道方向篇 如何經營工程師 Youtube 頻道 - Part 0 Rails 中避免 race condition 的最佳實踐(二) Rails 中避免 race condition 的最佳實踐(一) 10 分鐘整合 google sheet 做自動化開發功能週報 經營 Side Project 300 天所帶來的收穫及挑戰 我的 Youtube 影片製作流程 API 設計時必須注意的 HTTP header 底線問題 如何提升你的程式可讀性之實務技巧(三) 如何提升你的程式可讀性之實務技巧(二) 如何提升你的程式可讀性之實務技巧(一) Ruby 中使用 freeze 優化效能的時機 避免 React 中的 useEffect 無限 render 在 Rails 內輕量使用 Vue Component 的最佳實踐 身為工程師的你在武漢疫情時能做些什麼 如何在區域網路用 Docker 架設有 SSL 的 Gitlab 從被問到問人,那些我常問的面試問題 [Rails] 如何漂亮寫出可維護的 query (Maintainable Rails Query) 在已知長度情況下優化 slice 的性能 [ReactNative] 如何在 iOS APP 上主動要求用戶評分 Rails 的 scope 為什麼用 lambda? Proc 與 lambda 不同之處 淺談 Active Record 的 Lazy load 特性 Rails 專案搭配 Github Actions 進行 RSpec 自動化測試 JavaScript 中 require, import 的差別及效能 React 效能優化基本招 ES6 箭頭函式 (Arrow functions) 2 個月擁有 6000 用戶 Side project 這樣做(三) 2 個月擁有 6000 用戶 Side project 這樣做(二) 2 個月擁有 6000 用戶 Side project 這樣做(一) 如何讓自己成為失敗的軟體工程師 如何在 Jenkins 上用 Docker 跑 Rails + Rspec 做 CI 如何用 Rack::Attack 阻擋 DDOS / 惡意流量 用 OpenSSL 自簽開發用 HTTPS SSL 憑證 以 OOP 的角度提升 Ruby code 質量 不停機 migration 避免鎖表的幾種操作 為機器加上登入訊息,在 ubuntu 設置登入歡迎詞 Ruby Memoization 性能優化之記憶化 淺談 SSH agent forwarding 和 proxy command 的安全風險與應用 [Rails] Service / Library / Concern 的差異 Ruby conf Taiwan 2019 參與筆記 避免過度的 Defensive Programming 防禦性程式設計 Rails 要用 Time.zone.now 還是 Time.now Rails i18n 小技巧總匯 1:1 攪亂器,如何用 Ruby 做可逆推序號 Rails 中的欄位及方法命名原則 [Rails] 用 puma-dev 作為本地開發伺服器 (支援 https 自簽憑證) 我的中高階 Rails 工作面試心得分享 讀書心得 - 「一流的人如何保持顛峰」 讀書心得 - 「窮查理的普通常識」 將 Rails 專案從手動部屬遷移使用 Capistrano 自動化部屬 工程師提昇自己的教學和簡報技術的方法 [筆記] Rails 3.2 升級 Rails 6.beta 經驗分享 Grape on Rails 實戰 101 Class method 氾濫帶來什麼問題 Rspec 中 let / let!(驚嘆號) / Instance variables / subject 的用法與差異 RDBMS 課程心得與筆記 常用的 Rails 開發規範 Rest-Client 如何做 Basic Authentication 驗證 用 ssh config 管理多台機器 [Rails] 實做批次操作的小技巧 [Rails] 何為 tld_lebgth? 遵循 Semantic Versioning 軟體開發語意化版本管理 請直接在 MySQL 裡面直接用 utf8mb4 取代 utf8 如何解決在 awesome print 中遇到 ActionController::Parameters unable to convert unpermitted 如何在 Mac 上升級 PostgreSQL 並遷移資料 如何解決 Mysql2::Error: Incorrect string value 讀書心得 - 「信任因子:信任如何影響大腦運作、激勵員工、達到組織目標」 我是如何寫部落格筆記的 讀書心得 - 「先問,為什麼?:顛覆慣性思考的黃金圈理論,啟動你的感召領導力」 [Rails] 解決 Reset Password 帶來的 token 洩漏問題 我的軟體工程師生涯:如何挑選適合你的公司 Rails 中的 delegate 用法 淺述 SSR SPA 優缺點 Rails 非同步工作請用 Global ID [React] Class Component 傳遞 props 的 2 種方式 好用的隱私權政策 URL 自動生成 Rails 5.1 之後的 tag helper Rails 5.2 Encrypted Credentials 最近面試被給的建議和書單 一般架構需要用到 K8S 嗎 透過 commit SHA 找 github Pull request 從零搭建,如何讓 Rails 跑在 Kubernetes(k8s)(二) 從零搭建,如何讓 Rails 跑在 Kubernetes(k8s)(一) if/unless 寫作不要用多重否定句啊 Load balance 負載平衡設計 ES6: export default 和 export 的差別 搞懂 React 中的 state 和 props
盡可能的減少使用具感染性的 Try 或是 lonely/safe navigation operator
2018-11-25 · via Nic Lin's Blog

在 Rails application 中,我們可以用 Object#try 來避免 NoMethodError 拋出,而當 recevier 發現該 method 不存在時,會直接回傳 nil,可以避免更冗長的判斷、額外的錯誤處理,聽起來確實更好了,同時,我認為是製造更多的問題

# 原本的寫法

if user && user.eamil
  # dosomething
end

# 用了 Try 以後

if user.try(:email)
# dosomething
end

看起來不錯吧?好像更簡潔了。

但這種情況還算可以接受,如果濫用呢?看看以下兩個例子

# 將每個有可能回傳 nil 的 method 串接起來
payment.client.try(:addresses).try(:first).try(:country).try(:name)
# 一個 Class 用 try 應付多種情況
class GoodService
  def call(object)
    object.save!
    object.try(:send_success_notification, "saved from GoodService")
  end
end

這樣 Try 到底會產生什麼問題呢?

程式碼意義非常不明確try 的意思就是

這裡可能會是 NoMethodError, 但我不告訴你是 NoMethodError 的時候該怎麼辦,反正一律給你 nil

payment.client.try(:address) 來說,如果這是合法的邏輯,有些 payment 就是沒有 client

  • 那是不是可以針對這些特別的 payment 進行額外處理?
  • 如果是 polymorphic relationship 那情況有可能更糟,也許是其他的 model 有實作而這裡沒有?
  • 或是更嚴重的數據遺失問題,程式的 bug 導致數據儲存不正確,又或是被黑客刪掉資料?

在這個例子裡面其實有很多隱患,光看這段 code 也無法馬上知道用 try 的意圖是什麼,也會造成日後 debug 的困難

處理 exception 其實有更好的作法。

遵循 Law of Demeter 最小知識原則

假設 A 要問 B 一個問題,但是 B 得問 C 才知道答案。

那麼 A 不需要知道 B 還要去問 C,對 A 來說,只要問 B 就能知道答案了。

# 範例

A.askB #=> Answer

# 違反此原則的範例:

A.askB.askC #=> Answer

這個問題不在於程式要 . 多長,而在於和 Object 之間的耦合程度,所以說如果是做一些轉換和操作並沒有違反原則

# 這個例子並沒有違反原則,因為只是做轉換和操作而已

input.to_s.strip.split(" ").map(&:capitalize).join(" ")

遵循最小知識原則,可以避免緊耦合狀況發生。

幾個改進的方法,而你應該認真處理錯誤

就上述例子 payment.client.try(:address) 來說,一般改進的作法會是

邏輯上不應該有 nil 狀況的寫法(如果真的遇到 nil, 一定就是問題)

class Payment
  def client_address
    client.address
  end
end

直接在 method 裡面處理 nil 狀況(意義明白,好讀)

class Payment
  def client_address
    return nil if client.nil?

    client.address
  end
end

Rails 的話可以直接用 delegate 做,而 allow_nil 用來確定你的邏輯是否可以接受 nil

class Payment
  delegate :address, to: :client, prefix: true, allow_nil: true
end

或是直接在一開始拉資料時就確保資料範圍正確性,避免不必要的錯誤發生,因為有可能某些 payment 就是沒有 client,那麼避免這樣的錯誤,可以直接在拉資料的時候,確保找出來的資料都是有 client 的 payment

這樣一來,其他的 developer 也能夠更清楚為什麼要這樣寫

Payment.with_completed_transactions.find_each do |payment|
  do_something_with_address(payment.client_address)
end

還有一種狀況是,確保資料型態正確時,也不應該用 Try,看看下面這樣的寫法,我們可以猜測會有一個 String 傳進來,但也有可能會被亂傳其他參數進來,我們將永遠不會知道。

params[:name].try(:upcase)

但我認為更好的作法,是在程式碼裡面告訴其他的 developer, 這裡有可能會有 nil,如果有不是 nil 的狀況,那就應該要有 Error exception。

return if params[:name].nil?

params[:name].to_s.upcase

做多型態的 service 時,也請不要用 Try,完全看不懂啊!(所以什麼 Object 會發通知,什麼不會?)

class GoodService
  def call(object)
    object.save!
    object.try(:send_success_notification, "saved from GoodService")
  end
end

這裡建議兩種更好的改進作法

直接做兩個 service,把責任區分清楚。

class GoodServiceA
  def call(object)
    object.save!
  end
end

class GoodServiceB
  def call(object)
    object.save!
    object.send_success_notification("saved from GoodService")
  end
end

或是把 send_success_notification 單獨拉出來做


class GoodService
  def call(object)
    object.save!
    object.send_success_notification("saved from GoodService")
  end
end

def send_success_notification(string)
  # dosomething
end

如果有興趣的話,還有 Null Object Pattern 可以參考,這邊就不多說明了。

小結

  • Rails 裡面有 try()
  • Ruby 裡面則是有 &. (Version 2.3.0 之後,稱為 lonely/safe navigation operator)

&. 其實比 try() 好一些,在某些情況下還是會拋出 NoMethodError,但其實都還是很曖昧不明的寫法,在讀 code 時也會很困擾,總結來講,個人習慣並不偏好這種寫法。

參考