কন্ট্রোল স্ট্রাকচার
এই অধ্যায়ে আমরা এলিক্সিরে ব্যবহৃত কন্ট্রোল স্ট্রাকচার নিয়ে কথা বলব।
if ও unless
if/2
বহুল ব্যবহৃত একটি কন্ট্রোল স্ট্রাকচার যা প্রায় সমস্ত ল্যাঙ্গুয়েজেই রয়েছে। কিছু কিছু ল্যাঙ্গুয়েজ যেমন রুবী ও পার্লে এর উল্টো তথা unless/2
এর ব্যবস্থা রয়েছে। এলিক্সিরে if/2
ও unless/2
অন্যান্য ল্যাঙ্গুয়েজের মতই মূলত কাজ করে কিন্তু এরা ল্যাঙ্গুয়েজের কোন গঠন নয়, বরং ম্যাক্রো। এরা কিভাবে কাজ করে তা জানতে ভিজিট করুন কার্নেল মডিউল পেইজটিতে।
জেনে রাখা ভাল যে এলিক্সিরে “ফলসি” ভ্যালু মাত্র দুইটি- nil
এবং false
। এই ফলসি ভ্যালুর উপর নির্ভর করে কন্ট্রোল স্ট্রাকচারের পাস অথবা ফেইল করা।
iex> if String.valid?("Hello") do
...> "Valid string!"
...> else
...> "Invalid string."
...> end
"Valid string!"
iex> if "a string value" do
...> "Truthy"
...> end
"Truthy"
unless/2
আর if/2
এর কার্যপ্রণালি একই, তবে unless/2
শুধু “ফলসি” ভ্যালুকেই গ্রুহণ করে।
iex> unless is_integer("hello") do
...> "Not an Int"
...> end
"Not an Int"
case
একাধিক প্যাটার্নের সাথে ম্যাচ করতে হলে আমরা case
ব্যবহার করব।
iex> case {:ok, "Hello World"} do
...> {:ok, result} -> result
...> {:error} -> "Uh oh!"
...> _ -> "Catch all"
...> end
"Hello World"
শেষের _
ভেরিয়েবল গুরুত্বপূর্ণ এই স্ট্রাকচারে। অন্য কেউ না ম্যাচ করতে পারলে এর আওতাধীন লজিক কাজ করে। একে ছাড়া এরর রেইজড হবে যদি উপরের কেউ ম্যাচ করতে না পারে।
iex> case :even do
...> :odd -> "Odd"
...> end
** (CaseClauseError) no case clause matching: :even
iex> case :even do
...> :odd -> "Odd"
...> _ -> "Not Odd"
...> end
"Not Odd"
_
কে আপনি else
হিসেবে চিন্তা করতে পারেন। অর্থাৎ অন্য সব কন্ডিশানকে যারা ফেইল করে, তাদের এরা ম্যাচ করে।
case
যেহেতু প্যাটার্ন ম্যাচ করে কাজেই প্যাটার্নের সমস্ত নিয়ম এর উপর প্রযোজ্য। যেমন কোন ভেরিয়েবলের সাথে ম্যাচ করতে চাইলে ^
অর্থাৎ পিন অপারেটর ব্যবহার করতে হয়।
iex> pie = 3.14
3.14
iex> case "cherry pie" do
...> ^pie -> "Not so tasty"
...> pie -> "I bet #{pie} is tasty"
...> end
"I bet cherry pie is tasty"
case
এর আরেকটি ফিচার হল গার্ডের ব্যবহার।
এই উদাহরণটি সরাসরি এলিক্সিরের অফিসিয়াল ডকুমেন্টেশান থেকে নেয়া Getting Started গাইড থেকে।
iex> case {1, 2, 3} do
...> {1, x, 3} when x > 0 ->
...> "Will match"
...> _ ->
...> "Won't match"
...> end
"Will match"
আরও জানতে হলে অফিসিয়াল ডকুমেন্টেশানের Expressions allowed in guard clauses চ্যাপ্টারটি দেখুন।
cond
যখন আমরা একাধিক কন্ডিশানের সাথে আমাদের ম্যাচিং করতে হবে তখন cond
ব্যবহার করব যা অন্যান্য ল্যাঙ্গুয়েজের else if
, elsif
, elif
ইত্যাদির মত করে কাজ করে।
এই উদাহরণটি সরাসরি এলিক্সিরের অফিসিয়াল ডকুমেন্টেশান থেকে নেয়া Getting Started গাইড থেকে।
iex> cond do
...> 2 + 2 == 5 ->
...> "This will not be true"
...> 2 * 2 == 3 ->
...> "Nor this"
...> 1 + 1 == 2 ->
...> "But this will"
...> end
"But this will"
case
এর মত cond
ও এরর রেইজ করবে যদি কোন ম্যাচ পাওয়া না যায়। case
এর _
এর সমকক্ষ হিসেবে আমরা true
ব্যবহার করতে পারি যা ফলব্যাক হিসেবে কাজ করবে যখন কোন কন্ডিশান না মিলে।
iex> cond do
...> 7 + 1 == 0 -> "Incorrect"
...> true -> "Catch all"
...> end
"Catch all"
with
কখনো কখনো আমরা এমন পরিস্থিতিতে পরি যখন case
স্টেটমেন্ট এর ক্লোজগুলি সুন্দর মত পাইপ করা যায় না এবং নেস্টেড হয়ে যায়। with
এক্সপ্রেশানের আবির্ভাব হয়েছে এই ধরণের অবস্থা হ্যান্ডল করার জন্য। এটি হল with
কী-ওয়ার্ড, সংশ্লিষ্ট জেনারেটরসমূহ এবং একটি এক্সপ্রেশানের সমন্বয়।
জেনারেটর নিয়ে আমরা লিস্ট কম্প্রিহেনশান অধ্যায়ে কথা বলব। আপাতত এতটুকু জেনে রাখি যে এরা প্যাটার্ন ম্যাচিং দিয়ে <-
এর ডান হাতের এক্সপ্রেশানকে কম্পেয়ার করে বাম হাতের এক্সপ্রেশানের সাথে।
with
এর একটি সহজ উদাহরণ দিয়ে শুরু করা যাক-
iex> user = %{first: "Sean", last: "Callan"}
%{first: "Sean", last: "Callan"}
iex> with {:ok, first} <- Map.fetch(user, :first),
...> {:ok, last} <- Map.fetch(user, :last),
...> do: last <> ", " <> first
"Callan, Sean"
কোন এক্সপ্রেশান ম্যাচ করতে সক্ষম না হলে ম্যাচ না হওয়া ভ্যালু রিটার্ন করা হয়।
iex> user = %{first: "doomspork"}
%{first: "doomspork"}
iex> with {:ok, first} <- Map.fetch(user, :first),
...> {:ok, last} <- Map.fetch(user, :last),
...> do: last <> ", " <> first
:error
with
ছাড়া ব্যবহৃত একটি উদাহরণ দিয়ে দেখা যাক কিভাবে with
আমাদের উপকারে আসে-
case Repo.insert(changeset) do
{:ok, user} ->
case Guardian.encode_and_sign(user, :token, claims) do
{:ok, jwt, full_claims} ->
important_stuff(jwt, full_claims)
error ->
error
end
error ->
error
end
এবার রিফ্যাক্টর করে with
কে নিয়ে আসা যাক-
with {:ok, user} <- Repo.insert(changeset),
{:ok, jwt, full_claims} <- Guardian.encode_and_sign(user, :token, claims),
do: important_stuff(jwt, full_claims)
উপরিউক্ত কোডটি যেমন ছোট তেমনি বোধগম্য।
এলিক্সির ১.৩ থেকে with
স্টেটমেন্টে else
কে আনা হয়েছে।
import Integer
m = %{a: 1, c: 3}
a =
with {:ok, res} <- Map.fetch(m, :a),
true <- is_even(res) do
IO.puts("Divided by 2 it is #{div(res, 2)}")
else
:error -> IO.puts("We don't have this item in map")
_ -> IO.puts("It's not odd")
end
এটি আমাদের case
এর মত প্যাটার্ন ম্যাচিং কার্যপ্রণালী প্রদান করে যা গ্রহণ করে প্রথম সেই ভ্যালু যা ম্যাচড হয়নি।
Caught a mistake or want to contribute to the lesson? Edit this lesson on GitHub!