L v SOLIDe
Milan Herda, 05 / 2023
SOLID princípy v JavaScripte
O čom budeme hovoriť
- Čo je SOLID
- Liskov Substitution Principle
- Výhody Liskov Substitution Principle
- Ako rozpoznať porušenia princípu
- Ako refaktorovať, aby sme dodržali princíp
SOLID
- Single Responsibility Principle
- Open-Closed Principle
- Liskov Substitution Principle
- Interface Segregation Principle
- Dependency Inversion Principle
Skratka predstavujúca 5 základných princípov dobrého softvérového návrhu
SOLID
Nie je iba pre OOP
Princípy sú všeobecne použiteľné
Liskov Substitution Principle
Liskovej princíp zameniteľnosti
Odvodená trieda musí byť náhradou základnej triedy
Odvodená trieda musí byť náhradou základnej triedy
Odvodená trieda musí byť náhradou základnej triedy
Odvodená trieda musí byť náhradou svojho interfacu
Objekt musí byť náhradou svojho interfacu
Implementácia musí byť náhradou svojho typového predpisu
Byť dobrou náhradou svojej základnej triedy/interfacu
znamená:
- poskytovať implementáciu pre všetky predpísané metódy
- mať rovnaké typy návratových hodnôt
- nezvoľňovať typ návratovej hodnoty
- nesprísňovať požiadavky na argumenty
- nesprísňovať kontrakt
- neobchádzať kontrakt v kóde
Cieľom LSP je kód, kde môžeme
vymeniť implementáciu rodiča
za implementáciu potomka
a kód bude ďalej fungovať.
Výhody dodržiavania
LSP
- časti kódu sú navzájom bezpečne vymeniteľné
- eliminácia chýb spôsobených nedodržaním kontraktu
Symptómy porušenia princípu
- nie sú poriadne implementované všetky metódy
- potomok má inú návratovú hodnotu ako rodič
- prísnejšie požiadavky na argumenty
- prísnejší kontrakt
- obchádzanie kontraktu v kóde
- nevieme napísať taký kód potomka, ktorý by vyhovel požiadavke rodiča
Veľkú časť porušení dnes vieme odchytiť statickou analýzou
- chýbajúce metódy
- sprísnené parametre metód
- nesprávne návratové hodnoty
Pozrieme sa preto na menej triviálne situácie
Ako refaktorovať
Podľa individuálnej situácie
Často je problém v nesprávne definovanej základnej triede/interface
// entity/user.ts
export interface User {
getId: () => number;
getName: () => string;
getType: () => UserType;
getFavouriteTvShow: () => string;
getAdminRights: () => AdminRight[];
}
const users: User[] = [
{
getId: () => 11,
getName: () => 'Ferko Mrkvička',
getType: () => UserType.User,
getFavouriteTvShow: () => 'Babylon 5',
//getAdminRights: () => ???,
},
// ...
];
const admins: User[] = [
{
getId: () => 101,
getName: () => 'Neo',
getType: () => UserType.Admin,
//getFavouriteTvShow: () => ???,
getAdminRights: () => [
AdminRight.BanUser,
AdminRight.EditUser,
AdminRight.ViewUser
],
},
// ...
];
Doplňte kód pre chýbajúce metódy
Refactoring time!
Stiahnite si zdrojáky
Problém je, že objekty majú nastaveného nesprávneho rodiča
Prečo má interface User metódy getFavouriteShow a getAdminRights, keď ich nevedia implementovať všetci potomkovia?
- rozdelíme interface
- každý typ používateľa bude implementovať iný interface
- upravíme typehinty v kóde
export interface BaseUser {
getId: () => number;
getName: () => string;
getType: () => UserType;
}
export interface User extends BaseUser {
getFavouriteTvShow: () => string;
getType: () => UserType.User;
}
export interface AdminUser extends BaseUser {
getAdminRights: () => AdminRight[];
getType: () => UserType.Admin;
}
src/entity/user.ts
import { UserType, AdminRight, AdminUser } from '@local/entity/user';
export function getAdmins() {
const admins: AdminUser[] = [
{
getId: () => 101,
getName: () => 'Neo',
getType: () => UserType.Admin,
getAdminRights: () => [
AdminRight.BanUser,
AdminRight.EditUser,
AdminRight.ViewUser
],
},
{
getId: () => 102,
getName: () => 'Mr. Anderson',
getType: () => UserType.Admin,
getAdminRights: () => [AdminRight.ViewUser],
},
];
return admins;
}
src/services/users/adminRepository.ts
import { AdminUser } from '@local/entity/user';
function ListOfAdminUsers({ users }: { users: AdminUser[] }) {
// ...
}
function AdminRow({ user }: { user: AdminUser }) {
// ...
}
src/components/ListOfAdminUsers.tsx
Hotovo
Príklad 2
export interface Validator {
getRules: () => any;
validate: (input: any) => boolean;
}
const StrictValidator: Validator = {
getRules() {
return [
ValidationRule.Required,
ValidationRule.NotEmpty,
ValidationRule.IsNumeric
];
},
validate(input: any) {
// ...
},
};
const BenevolentValidator: Validator = {
getRules() {
return null;
},
validate(input: any) {
// ...
},
};
function ListOfValidation({ validators }: { validators: Validator[] }) {
return (
<>
<h2 className="title is-2">Validation Rules</h2>
<table className="table is-striped">
<thead>
<th>Validator</th>
<th>Rules</th>
</thead>
<tbody>
{validators.map((validator) => {
return (
<tr key={`${validator.getName()}`}>
<td>{validator.getName()}</td>
<td>{validator.getRules().join(', ')}</td>
</tr>
);
})}
</tbody>
</table>
</>
);
}
Problém:
LSP je síce formálne dodržané, ale...
- implementácie metódy getRules sú zásadne odlišné
- komponent ListOfValidation tvrdí, že očakáva typ Validator, ale v skutočnosti predpokladá subtyp
- problém máme v rodičovi, ktorý nie je nadefinovaný tak, aby jeho potomkovia boli poriadnymi potomkami
Skúste kód upraviť tak, aby rodičovský typ presnejšie definoval návratový typ pre getRules
export interface Validator {
getRules: () => ValidationRule[];
validate: (input: any) => boolean;
}
const StrictValidator: Validator = {
getRules() {
return [
ValidationRule.Required,
ValidationRule.NotEmpty,
ValidationRule.IsNumeric
];
},
validate(input: any) {
// ...
},
};
const BenevolentValidator: Validator = {
getRules() {
return [];
},
validate(input: any) {
// ...
},
};
Hotovo
Príklad 3
// types.ts
import { UserBase } from '@local/entity/user';
export type AdminRightsFormatter = (user: UserBase) => string;
// index.ts
import { UserType } from '@local/entity/user';
import { AdminRightsFormatter } from './types';
export const rightsFormatter: AdminRightsFormatter = function (user) {
if (user.getType() !== UserType.Admin) {
throw new Error(`User ${user.getId()} is not an admin`);
}
return user.getAdminRights().join(', ');
};
src/services/users/adminRights/*
LSP (a dedičnosť) vraví, že namiesto inštancie rodičovského typu viem použiť inštanciu potomka a kód pobeží ďalej.
export const rightsFormatter: AdminRightsFormatter = function (user) {
if (user.getType() !== UserType.Admin) {
throw new Error(`User ${user.getId()} is not an admin`);
}
return user.getAdminRights().join(', ');
};
Vyhodenie výnimky toto pravidlo porušuje, lebo rodičovský typ o tom nehovorí.
Nie je to ani súčasťou neformálneho kontraktu (napr. v komentári k typu)
Odstráňte vyhadzovanie výnimky
import { UserType } from '@local/entity/user';
import { AdminRightsFormatter } from './types';
export const rightsFormatter: AdminRightsFormatter = function (user) {
if (user.getType() !== UserType.Admin) {
return '';
}
return user.getAdminRights().join(', ');
};
Hotovo
Naozaj?
Je v poriadku, keď metóda dostane argument typu, s ktorým neráta?
Ako vieme, že sme pre tento typ zadefinovali správne správanie?
Čo ak by mohlo byť nesprávnych typov viac?
A prečo nám rodič prikazuje robiť s typom, o ktorom by mohol vedieť, že nie je správny?
Opäť máme problém v rodičovi!
Rodičovský typ by nemal definovať príliš široké požiadavky na potomkov.
Mal by potomkom ponúknuť jasné hranice, ale zároveň implementačnú voľnosť.
Nemal by nám prikazovať pracovať s typom, o ktorom môže predpokladať, že nie je správny.
// types.ts
import { UserBase } from '@local/entity/user';
export type AdminRightsFormatter = (user: UserBase) => string;
src/services/users/adminRights/types.ts
- Použitie UserBase nie je správne
- Rodič by o tom mohol vedieť, pretože jeho meno aj umiestnenie hovorí o admin používateľoch
- Preto vymeníme UserBase za správny typ
// types.ts
import { AdminUser } from '@local/entity/user';
export type AdminRightsFormatter = (user: AdminUser) => string;
// index.ts
import { AdminRightsFormatter } from './types';
export const rightsFormatter: AdminRightsFormatter = function (user) {
return user.getAdminRights().join(', ');
};
src/services/users/adminRights/*
Hotovo
Čo by sme robili v prípade, keby AdminUser neexistoval a mali by sme iba typ User so začiatku školenia?
Vytvorili by sme ho
Príklad 4
export type GetFansFunc = () => User[];
export const getFansOfVariousShows: GetFansFunc = function () {
return [
createFan(101, 'Susan', 'Star Trek'),
createFan(102, 'Monica', 'Star Wars'),
createFan(103, 'Peter', 'StarGate - SG1'),
createFan(104, 'John', 'Battlestar Galactica'),
];
};
function createFan(id: number, name: string, tvShow: string) {
let favouriteTvShow = tvShow;
return {
getId: () => id,
getName: () => name,
getType: () => UserType.User,
getFavouriteTvShow: () => favouriteTvShow,
setFavouriteTvShow: (show: string) => {
favouriteTvShow = show;
},
};
}
src/services/fans/fansRepository.ts
function watchFewEpisodesOfB5(fan: User) {
const tvShow = new TvShow('Babylon 5');
tvShow.addFan(fan);
}
const fans = getFansOfVariousShows();
watchFewEpisodesOfB5(fans[0]);
src/components/ListOfFans.tsx
class TvShow {
name: string;
constructor(name: string) {
this.name = name;
}
getName() {
return this.name;
}
addFan(user: User) {
user.setFavouriteTvShow(this.name);
}
}
export default TvShow;
src/services/tvShows/index.ts
class TvShow {
name: string;
constructor(name: string) {
this.name = name;
}
getName() {
return this.name;
}
addFan(user: User) {
// @ts-ignore
user.setFavouriteTvShow(this.name);
}
}
export default TvShow;
src/services/tvShows/index.ts
Programovať by sme mali len voči deklarovaným rozhraniam
Ak nám rozhranie nevyhovuje, tak ho vymeníme
Typ User pre metódu addFan nie je dostatočný, vymeníme ho
export interface Fan extends User {
setFavouriteTvShow: (show: string) => void;
}
export type GetFansFunc = () => Fan[];
export const getFansOfVariousShows: GetFansFunc = function () {
return [
createFan(101, 'Susan', 'Star Trek'),
createFan(102, 'Monica', 'Star Wars'),
createFan(103, 'Peter', 'StarGate - SG1'),
createFan(104, 'John', 'Battlestar Galactica'),
];
};
function createFan(id: number, name: string, tvShow: string) {
let favouriteTvShow = tvShow;
return {
getId: () => id,
getName: () => name,
getType: () => UserType.User,
getFavouriteTvShow: () => favouriteTvShow,
setFavouriteTvShow: (show: string) => {
favouriteTvShow = show;
},
};
}
src/services/fans/fansRepository.ts
function watchFewEpisodesOfB5(fan: Fan) {
const tvShow = new TvShow('Babylon 5');
tvShow.addFan(fan);
}
const fans = getFansOfVariousShows();
watchFewEpisodesOfB5(fans[0]);
src/components/ListOfFans.tsx
class TvShow {
name: string;
constructor(name: string) {
this.name = name;
}
getName() {
return this.name;
}
addFan(user: Fan) {
user.setFavouriteTvShow(this.name);
}
}
export default TvShow;
src/services/tvShows/index.ts
Hotovo
Opakovanie
Odvodená trieda musí byť náhradou základnej triedy
Odvodená trieda musí byť náhradou svojho interfacu
Objekt musí byť náhradou svojho interfacu
Implementácia musí byť náhradou svojho typového predpisu
Byť dobrou náhradou svojej základnej triedy/interfacu
znamená:
- poskytovať implementáciu pre všetky predpísané metódy
- mať rovnaké typy návratových hodnôt
- nezvoľňovať typ návratovej hodnoty
- nesprísňovať požiadavky na argumenty
- nesprísňovať kontrakt
- neobchádzať kontrakt v kóde
Výhody dodržiavania
LSP
- časti kódu sú navzájom bezpečne vymeniteľné
- eliminácia chýb spôsobených nedodržaním kontraktu
Ďakujem za pozornosť
L v SOLIDe (JavaScript)
By Milan Herda
L v SOLIDe (JavaScript)
- 245