與 TypeScript 共舞 🐉 - 跟我想的不一樣R之真實世界篇
俗話說 OOP 學得好,Coding 沒煩惱。然後 Swift 有 POP,Ruby 有 mixin,還暫且不說 FP ㄏ
From the ancent......the Dragon 🐲......
讓我們從講古開始切入,其實 TypeScript 已經從古早時候就一直存在 (好吧說長不長說短不短的 5 年),只是近期內開始受到歡迎,吸引更多的開發者選擇使用 TypeScript,並且有穩定的商業應用表現 (有興趣的各位可以聽聽看 SoftwareEnginneringDaily 訪問 slack 使用 TypeScript 的經驗分享 - 傳送門)。
Compare with Flow!?
Facebook 毋庸置疑的贏了一場漂亮的戰爭,不管是 React 也好 babel 也行,
interface IntlExtension {
intl: InjectedIntl
}
function mountWithIntl<C extends IntlExtension>(node: React.ReactElement<C>) {
const intlProvider = new IntlProvider({ locale: 'zh-TW', messages }, {})
const { intl } = intlProvider.getChildContext()
return mount(
React.cloneElement(node, { intl }) as React.ReactElement<C>,
{
context: { intl },
childContextTypes: { intl: intlShape }
}
)
}
https://softwareengineeringdaily.com/?s=typescript
export interface TableProps<Data extends BasicDataDefine> {
classNamePrefix?: string
wrapperClassName?: string
className?: string
loading?: boolean
loader?: JSX.Element
rowSelection?: RowSelectionOption<Data>
pagination?: PaginationOption
dataSource: Data[]
columns: BasicColumnDefine[]
children?: undefined
// #TODO: filter or sorting function
onChange?: (currentPage: number, pageSize: number) => void
}
interface DefaultProps {
classNamePrefix: string
wrapperClassName: string
className: string
loading: boolean
loader: JSX.Element
}
type PropsWithDefault = TableProps & DefaultProps
TS2315: Type 'PropsWithDefault' is not generic.
deconstructor
this.selectedRecords = dataSource.filter(data =>
(data.key ? false : selectedKeys.includes(data.key))
)
TS2345: Argument of type 'string | undefined' is not assignable to parameter of type 'string'.
Type 'undefined' is not assignable to type 'string'.
this.selectedRecords = dataSource.filter(data =>
(data.key ? false : selectedKeys.includes(data.key as string))
)
const foo?: string = undefined
console.log(foo as string)
Overload!?
我們期望的 Overload in Swift
class User {
func register(name: String, gender: Gender, age: Int) {
...
}
func register(facebookID: String, gender: Gender) {
...
}
func register(referURL: URL) {
...
}
}
Protocol 版本
protocol User {
func register(name: String, gender: Gender, age: Int)
func register(facebookID: String, gender: Gender)
func register(referURL: URL)
}
哦哦哦!? TypeScript 有型態,所以把前面的邏輯慣例應該可以套用在 TypeScript 吧?
class User {
register(name: string, gender: Gender, age: number): void {
...
}
register(facebookID: string, gender: Gender): void {
...
}
register(referURL: URL): void {
...
}
}
......淦,怎麼只有最後一個 method 被叫到!!看了一下官方的文件...好吧,手動處理囉!
class User {
register(name: string, gender: Gender, age: number): void;
register(facebookID: string, gender: Gender): void;
register(referURL: URL): void;
register(nameOrFacebookIDOrReferURL: URL | string, gender?: Gender, age?: number): void {
......
}
}
雖然這樣寫真心非常醜,不過至少可以算是一種往前踏了一個閃身的改進了吧(?)
嗯,那這樣訂個 interface 如果要有 overload 大概也知道要怎麼做了。
interface User {
register(name: string, gender: Gender, age: number): void;
register(facebookID: string, gender: Gender): void;
register(referURL: URL): void;
register(nameOrFacebookIDOrReferURL: URL | string, gender?: Gender, age?: number): void;
}
WTF!? 怎麼有 4 個 method,做 Overload 處理的只有三個啊?最後一個實作細節怎麼被暴露出來了...
其實在 interface 宣告層想要有 Overload 效果的話,只要宣告需要的 method,而實際上會被呼叫到的 method 在 implement interface 的時候寫上就沒問題了。
interface User {
register(name: string, gender: Gender, age: number);
register(facebookID: string, gender: Gender);
register(referURL: URL);
}
const user: User {
...
register(nameOrFacebookIDOrReferURL: URL | string, gender?: Gender, age?: number) {
......
}
}
// Or
class Member implements User {
...
register(nameOrFacebookIDOrReferURL: URL | string, gender?: Gender, age?: number) {
......
}
}