পূর্ববর্তী অংশ: ক্লিন কোড: টাইপস্ক্রিপ্টে কাজ এবং পদ্ধতি [পর্ব 1] ক্লিন কোড: টাইপস্ক্রিপ্টে নামকরণ এবং কোড রচনা [পর্ব 2] ক্লিন কোড: টাইপস্ক্রিপ্টে ক্লাস এবং অবজেক্ট [পর্ব 3] আমরা আমাদের কোডকে পরিষ্কার, নমনীয় এবং রক্ষণাবেক্ষণযোগ্য করার পন্থা বিবেচনা করে চলেছি। এবং এখন, এর পরিষ্কার স্থাপত্য সমাধান তদন্ত শুরু করা যাক. সলিড নীতিগুলি রবার্ট সি. মার্টিন দ্বারা প্রবর্তিত হয়েছিল এবং অবজেক্ট-ওরিয়েন্টেড প্রোগ্রামিং এবং সফ্টওয়্যার ডিজাইনের সর্বোত্তম অনুশীলন হিসাবে ব্যাপকভাবে বিবেচিত হয়। এই নিবন্ধে, আমরা তিনটি প্রথম সলিড নীতির দিকে নজর দেব: একক দায়িত্ব নীতি, খোলা/বন্ধ নীতি এবং ব্যবহারিক উদাহরণগুলিতে লিসকভ প্রতিস্থাপন নীতি। সুচিপত্র: একক দায়িত্ব নীতি খোলা/বন্ধ নীতি লিসকভ প্রতিস্থাপন নীতি 1. একক দায়িত্ব নীতি (SRP) একটি ক্লাস পরিবর্তন করার শুধুমাত্র একটি কারণ থাকা উচিত অন্য কথায়, একটি শ্রেণীর একটি একক দায়িত্ব বা কাজ থাকা উচিত। এটি গুরুত্বপূর্ণ কারণ একটি একক দায়িত্ব সহ একটি ক্লাস বোঝা, পরিবর্তন করা এবং বজায় রাখা সহজ। কোডের একটি ক্ষেত্রের পরিবর্তনগুলি পুরো সিস্টেমে ঢেউ খেলানো হবে না, বাগ প্রবর্তনের ঝুঁকি হ্রাস করবে। আসুন একটি বাস্তব উদাহরণ এবং একক দায়িত্ব নীতি অনুসরণ করার উপায়গুলি একবার দেখে নেওয়া যাক: // Bad class UserSettingsService { constructor(user: IUser) { this.user = user; } changeSettings(settings: IUserSettings): void { if (this.isUserValidated()) { // ... } } getUserInfo(): Promise<IUserSettings> { // ... } async isUserValidated(): Promise<boolean> { const userInfo = await this.getUserInfo(); // ... } } এই উদাহরণে, আমাদের শ্রেণী বিভিন্ন দিকে ক্রিয়া সম্পাদন করে: এটি প্রসঙ্গ সেট আপ করে, এটি পরিবর্তন করে এবং এটি বৈধ করে। SRP অনুসরণ করার জন্য, আমাদের এই বিভিন্ন দায়িত্ব ভাগ করতে হবে। // Better class UserAuth { constructor(user: IUser) { this.user = user; } getUserInfo(): Promise<IUserSettings> { // ... } async isUserValidated(): boolean { const userInfo = await this.getUserInfo(); // ... } } class UserSettings { constructor(user: IUser) { this.user = user; this.auth = new UserAuth(user); } changeSettings(settings: IUserSettings): void { if (this.auth.isUserValidated()) { // ... } } } এটা কেন গুরুত্বপূর্ণ? SRP-এর লক্ষ্য হল কোড মডুলারিটি উন্নত করা, আন্তঃ-নির্ভরতা থেকে উদ্ভূত চ্যালেঞ্জগুলি কমিয়ে আনা। কোডগুলিকে ফাংশনে সংগঠিত করে, আলাদা করা ক্লাস, এবং মডুলারিটি প্রচার করে, এটি আরও পুনঃব্যবহারযোগ্য হয়ে ওঠে, সময় সাশ্রয় করে যা অন্যথায় বিদ্যমান কার্যকারিতা পুনরায় কোড করার জন্য ব্যয় হতে পারে। 2. খোলা/বন্ধ নীতি (OCP) সফ্টওয়্যার সত্তা (ক্লাস, মডিউল, ফাংশন) এক্সটেনশনের জন্য খোলা থাকা উচিত কিন্তু পরিবর্তনের জন্য বন্ধ করা উচিত আপনি বিদ্যমান কোড পরিবর্তন না করে নতুন কার্যকারিতা যোগ করতে সক্ষম হওয়া উচিত। এটি গুরুত্বপূর্ণ কারণ এই নীতি মেনে চলার মাধ্যমে, আপনি বিদ্যমান কোডের স্থায়িত্বকে ঝুঁকি না নিয়ে আপনার সিস্টেমে নতুন বৈশিষ্ট্য বা উপাদান প্রবর্তন করতে পারেন। এটি কোড পুনঃব্যবহারযোগ্যতা প্রচার করে এবং কার্যকারিতা প্রসারিত করার সময় ব্যাপক পরিবর্তনের প্রয়োজনীয়তা হ্রাস করে। // Bad class Product { id: number; name: string[]; price: number; protected constructor(id: number, name: string[], price: number) { this.id = id; this.name = name; this.price = price; } } class Ananas extends Product { constructor(id: number, name: string[], price: number) { super(id, name, price); } } class Banana extends Product { constructor(id: number, name: string[], price: string) { super(id, name, price); } } class HttpRequestCost { constructor(product: Product) { this.product = product; } getDeliveryCost(): number { if (product instanceOf Ananas) { return requestAnanas(url).then(...); } if (product instanceOf Banana) { return requestBanana(url).then(...); } } } function requestAnanas(url: string): Promise<ICost> { // logic for ananas } function requestBanana(url: string): Promise<ICost> { // logic for bananas } এই উদাহরণে, সমস্যাটি হল ক্লাসে, যেটি পদ্ধতিতে, বিভিন্ন ধরনের পণ্যের গণনার শর্ত ধারণ করে এবং আমরা প্রতিটি ধরনের পণ্যের জন্য পৃথক পদ্ধতি ব্যবহার করি। সুতরাং, যদি আমাদের একটি নতুন ধরনের পণ্য যোগ করার প্রয়োজন হয়, তাহলে আমাদের ক্লাসটি সংশোধন করা উচিত এবং এটি নিরাপদ নয়; আমরা অপ্রত্যাশিত ফলাফল পেতে পারি। HttpRequestCost getDeliveryCost HttpRequestCost এটি এড়াতে, আমাদের উপলব্ধি ছাড়াই শ্রেণিতে একটি বিমূর্ত পদ্ধতির তৈরি করা উচিত। বিশেষ উপলব্ধি ক্লাসগুলি উত্তরাধিকারসূত্রে প্রাপ্ত হবে: আনানস এবং কলা। তারা নিজেদের জন্য অনুরোধ বুঝতে হবে. Product request ক্লাস ইন্টারফেস অনুসরণ করে প্যারামিটার নেবে, এবং যখন আমরা এ নির্দিষ্ট নির্ভরতা পাস করি, তখন এটি ইতিমধ্যে নিজের জন্য পদ্ধতি উপলব্ধি করবে। HttpRequestCost Product product HttpRequestCost request // Better abstract class Product { id: number; name: string[]; price: string; constructor(id: number, name: string[], price: string) { this.id = id; this.name = name; this.price = price; } abstract request(url: string): void; } class Ananas extends Product { constructor(id: number, name: string[], price: string) { super(id, name, price); } request(url: string): void { // logic for ananas } } class Banana extends Product { constructor(id: number, name: string[], price: string) { super(id, name, price); } request(url: string): void { // logic for bananas } } class HttpRequestCost { constructor(product: Product) { this.product = product; } request(): Promise<void> { return this.product.request(url).then(...); } } এটা কেন গুরুত্বপূর্ণ? এই নীতি অনুসরণ করে, আপনি কোড কাপলিং কমাবেন এবং সিস্টেমটিকে অপ্রত্যাশিত আচরণ থেকে বাঁচাবেন। 3. লিসকভ প্রতিস্থাপন নীতি (LSP) একটি সুপারক্লাসের অবজেক্টগুলিকে প্রয়োগ না করেই এর সাবক্লাসের বস্তুর সাথে প্রতিস্থাপনযোগ্য হওয়া উচিত। এই নীতিটি বোঝার জন্য, আসুন এই উদাহরণটি একবার দেখে নেওয়া যাক: // Bad class Worker { work(): void {/../} access(): void { console.log('Have an access to closed perimeter'); } } class Programmer extends Worker { createDatabase(): void {/../} } class Seller extends Worker { sale(): void {/../} } class Designer extends Worker { access(): void { throwError('No access'); } } এই উদাহরণে, আমাদের শ্রেণীর সাথে একটি সমস্যা আছে। , এবং সবাই কর্মী, এবং তারা পিতামাতা শ্রেণির থেকে উত্তরাধিকার সূত্রে প্রাপ্ত। কিন্তু একই সময়ে, ডিজাইনারদের বদ্ধ পরিধিতে অ্যাক্সেস নেই কারণ তারা ঠিকাদার, কর্মচারী নয়। এবং আমরা মেথড ওভাররাইড করেছি এবং লিসকভ সাবস্টিটিউশন প্রিন্সিপল ভঙ্গ করেছি। Contractor Designer Programmer Seller Worker access এই নীতিটি আমাদের বলে যে আমরা যদি সুপারক্লাস এর সাবক্লাস দিয়ে প্রতিস্থাপন করি, উদাহরণস্বরূপ ক্লাস, কার্যকারিতা ভাঙা উচিত নয়। কিন্তু যদি আমরা এটি করি, ক্লাসের কার্যকারিতা ভেঙে যাবে - পদ্ধতিতে ক্লাস থেকে অপ্রত্যাশিত উপলব্ধি হবে। Worker Designer Programmer access Designer লিসকভ সাবস্টিটিউশন নীতি অনুসরণ করে, আমাদের উপলব্ধিগুলিকে সাবক্লাসে পুনর্লিখন করা উচিত নয় তবে বিমূর্ততার নতুন স্তর তৈরি করতে হবে যেখানে আমরা প্রতিটি ধরণের বিমূর্ততার জন্য নির্দিষ্ট উপলব্ধিকে সংজ্ঞায়িত করি। আসুন এটি সংশোধন করা যাক: // Better class Worker { work(): void {/../} } class Employee extends Worker { access(): void { console.log('Have an access to closed perimeter'); } } class Contractor extends Worker { addNewContract(): void {/../} } class Programmer extends Employee { createDatabase(): void {/../} } class Saler extends Employee { sale(): void {/../} } class Designer extends Contractor { makeDesign(): void {/../} } আমরা এবং বিমূর্তকরণের নতুন স্তর তৈরি করেছি এবং শ্রেণিতে পদ্ধতি স্থানান্তর করেছি এবং নির্দিষ্ট উপলব্ধি সংজ্ঞায়িত করেছি। আমরা যদি শ্রেণিকে সাবক্লাস দিয়ে প্রতিস্থাপন করি, তাহলে কার্যকারিতা নষ্ট হবে না। Employee Contractor Employee access Worker Contractor Worker এটা কেন গুরুত্বপূর্ণ? আপনি যদি LSP মেনে চলেন, আপনি যে কোনো সাবক্লাস ইনস্ট্যান্স প্রতিস্থাপন করতে পারেন যেখানে একটি বেস ক্লাস ইনস্ট্যান্স প্রত্যাশিত, এবং প্রোগ্রামটি এখনও উদ্দেশ্য অনুযায়ী কাজ করা উচিত। এটি কোড পুনঃব্যবহার, মডুলারিটি প্রচার করে এবং আপনার কোডকে পরিবর্তনের জন্য আরও স্থিতিস্থাপক করে তোলে। পরের প্রবন্ধে, আমরা ইন্টারফেস সেগ্রিগেশন এবং ডিপেনডেন্সি ইনভার্সন সলিড নীতিগুলির দিকে নজর দেব।