停止編寫 API函數(shù)原因示例分析
正文
RESTFUL API 通常提供在不同實體上執(zhí)行增刪改查(CRUD)操作的一組接口。我們通常在我們的前端項目中為這些每一個接口提供一個函數(shù),這些函數(shù)的功能非常的相似,只是為了服務(wù)于不用的實體。舉個例子,假設(shè)我們有這些函數(shù)。
// api/users.js // 創(chuàng)建 export function createUser(userFormValues) { return fetch('users', { method: 'POST', body: userFormValues }); } // 查詢 export function getListOfUsers(keyword) { return fetch(`/users?keyword=${keyword}`); } export function getUser(id) { return fetch(`/users/${id}`); } // 更新 export updateUser(id, userFormValues) { return fetch(`/users/${is}`, { method: 'PUT', body: userFormValues }); } // 刪除 export function removeUser(id) { return fetch(`/users/${id}`, { method: 'DELETE' }); }
類似的功能可能存在于其他實體,例如:城市、產(chǎn)品、類別...但是我們可以用一個簡單的函數(shù)調(diào)用來代替這些函數(shù):
// apis/users.js export const users = crudBuilder('/users'); // apis/cities.js export const cities = crudBuilder('/regions/cities');
然后像這樣去使用:
users.create(values); users.show(1); users.list('john'); users.update(values); users.remove(1);
你可能會問為什么?有一些很好的理由:
- 減少了代碼行數(shù):你編寫的代碼,和當你離開公司時其他人維護的代碼
- 強制執(zhí)行 API 函數(shù)的命名約定,這可以增加代碼的可讀性和可維護性。例如你已經(jīng)見過的函數(shù)名稱:
getListOfUsers
,getCities
,getAllProducts
,productIndex
,fetchCategories
等, 他們都在做相同的事情,那就是“獲取實體列表”。使用這種方法,你將始終擁有entityName.list()
函數(shù),并且團隊中的每個人都知道這一點。
所以,讓我們創(chuàng)建crudBuilder()
函數(shù),然后再添加一些糖。
一個非常簡單的 CRUD 構(gòu)造器
對于上邊的簡單示例,crudBuilder()
函數(shù)將非常簡單:
export function crudBuilder(baseRoute) { function list(keyword) { return fetch(`${baseRoute}?keyword=${keyword}`); } function show(id) { return fetch(`${baseRoute}/${id}`); } function create(formValues) { return fetch(baseRoute, { method: 'POST', body: formValues }); } function update(id, formValues) { return fetch(`${baseRoute}/${id}`, { method: 'PUT', body: formValues }); } function remove(id) { return fetch(`${baseRoute}/${id}`, { method: 'DELETE' }); } return { list, show, create, update, remove }; }
假設(shè)約定 API 路徑并且給相應(yīng)實體提供一個路徑前綴,他將返回該實體上調(diào)用 CRUD 操作所需的所有方法。
但老實說,我們知道現(xiàn)實世界的應(yīng)用程序并不會那么簡單。在將這種方法應(yīng)用于我們的項目時,有很多事情需要考慮:
- 過濾:列表 API 通常會提供許多過濾器參數(shù)
- 分頁:列表 API 總是分頁的
- 轉(zhuǎn)換:API 返回的值在實際使用之前可能需要進行一些轉(zhuǎn)換
- 準備:
formValues
對象在發(fā)送給 API 之前需要做一些準備工作 - 自定義接口:更新特定項的接口不總是
${baseRoute}/${id}
因此,我們需要可以處理更多復(fù)雜場景的 CRUD 構(gòu)造器。
高級 CRUD 構(gòu)造器
讓我們通過上述方法來構(gòu)建一些日常中我們真正使用的東西。
過濾
首先,我們應(yīng)該在 list
輸出函數(shù)中處理更加復(fù)雜的過濾。每個實體列表可能有不同的過濾器并且用戶可能應(yīng)用了其中的一些過濾器。因此,我們不能對應(yīng)用過濾器的形狀和值有任何假設(shè),但是我們可以假設(shè)任何列表過濾都可以產(chǎn)生一個對象,該對象為不同的過濾器名稱指定了一些值。例如,我們可以過濾一些用戶:
const filters = { keyword: 'john', createdAt: new Date('2020-02-10') };
另一方面,我們不知道這些過濾器應(yīng)該如何傳遞給 API,但是我們可以假設(shè)(跟 API 提供方進行約定)每一個過濾器在列表 API 中都有一個相應(yīng)的參數(shù),可以以'key=value'
URL 查詢參數(shù)的形式被傳遞。
因此我們需要知道如何將應(yīng)用的過濾器轉(zhuǎn)換成相對應(yīng)的 API 參數(shù)來創(chuàng)建我們的 list
函數(shù)。這可以通過將 transformFilters
參數(shù)傳遞給 crudBuilder()
來完成。舉一個用戶的例子:
function transformUserFilters(filters) { const params = []; if (filters.keyword) { params.push(`keyword=${filters.keyword}`); } if (filters.createdAt) { params.push(`create_at=${dateUtility.format(filters.createdAt)}`); } return params; }
現(xiàn)在我們可以使用這個參數(shù)來創(chuàng)建 list
函數(shù)了。
export function crudBuilder(baseRoute, transformFilters) { function list(filters) { let params = transformFilters(filters)?.join('&'); if (params) { params += '?'; } return fetch(`${baseRoute}${params}`); } }
轉(zhuǎn)換和分頁
從 API 接收的數(shù)據(jù)可能需要進行一些轉(zhuǎn)換才能在我們的應(yīng)用程序中使用。例如,我們可能需要將 snake_case
轉(zhuǎn)換成駝峰命名或?qū)⒁恍┤掌谧址D(zhuǎn)換成用戶時區(qū)。
此外,我們還需要處理分頁。
我們假設(shè)來自 API 的分頁數(shù)據(jù)都按照如下格式(與 API 提供者約定):
{ data: [], // 實體對象列表 pagination: {...} // 分頁信息 }
因此,我們需要知道如何轉(zhuǎn)換單個實體對象。然后我們可以遍歷列表對象來轉(zhuǎn)換他們。為此,我們需要一個 transformEntity
函數(shù)作為 crudBuilder
的參數(shù)。
export function crudBuilder(baseRoute, transformFilters, transformEntity, ) { function list(filters) { const params = transformFilters(filters)?.join('&'); return fetch(`${baseRoute}?${params}`) .then((res) => res.json()) .then((res) => ({ data: res.data.map((entity) => transformEntity(entity)), pagination: res.pagination })); } }
list()
函數(shù)我們就完成了。
準備
對于 create
和 update
函數(shù),我們需要將 formValues
轉(zhuǎn)換成 API 需要的格式。例如,假設(shè)我們在表單中有一個 City
的城市選擇對象。但是 create
API 只需要 city_id
。因此,我們需要一個執(zhí)行以下操作的函數(shù):
const prepareValue = formValue => ({city_id: formValues.city.id});
這個函數(shù)會根據(jù)用例返回普通對象或者 FormData
,并且可以將數(shù)據(jù)傳遞給 API:
export function crudBuilder(baseRoute, transformFilters, transformEntity, prepareFormValues) { function create(formValues) { return fetch(baseRoute, { method: 'POST', body: prepareFormValues(formValues) }); } }
自定義接口
在一些少數(shù)情況下,對實體執(zhí)行某些操作的 API 接口不遵循相同的約定。例如,我們不能使用 /users/${id}
來編輯用戶,而是使用 /edit-user/${id}
。對于這些情況,我們應(yīng)該指定一個自定義路徑。
在這里我們允許覆蓋 crud builder 中使用的任何路徑。注意,展示、更新、移除操作的路徑可能取決于具體實體對象的信息,因此我們必須使用函數(shù)并傳遞實體對象來獲取路徑。
我們需要在對象中獲取這些自定義路徑,如果沒有指定,就退回到默認路徑。像這樣:
const paths = { list: 'list-of-users', show: (userId) => `users/with/id/${userId}`, create: 'users/new', update: (user) => `users/update/${user.id}`, remove: (user) => `delete-user/${user.id}` };
最終的 BRUD 構(gòu)造器
這是創(chuàng)建 CRUD 函數(shù)的最終代碼。
export function crudBuilder(baseRoute, transformFilters, transformEntity, prepareFormValues, paths) { function list (filters) { const path = paths.list || baseRoute; let params = transformFilters(filters)?.join('&'); if (params) { params += '?'; } return fetch(`${path}${params}`) .then((res) => res.json()) .then(() => ({ data: res.data.map(entity => transformEntity(entity)), pagination: res.pagination })); } function show(id) { const path = paths.show?.(id) || `${baseRoute}/${id}`; return fetch(path) .then((res) => res.json()) .then((res => transformEntity(res))); } function create(formValues) { const path = paths.create || baseRoute; return fetch(path, { method: 'POST', body: prepareFormValues(formValues) }); } function update(id, formValues) { const path = paths.update?.(id) || `${baseRoute}/${id}`; return fetch(path, { method: 'PUT', body: formValues }); } function remove(id) { const path = paths.remove?.(id) || `${baseRoute}/${id}`; return fetch(path, { method: 'DELETE' }); } return { list, show, create, update, remove } }
Saeed Mosavat: Stop writing API functions
以上就是停止編寫 API函數(shù)原因示例分析的詳細內(nèi)容,更多關(guān)于停止編寫 API 函數(shù)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
js實現(xiàn)保存文本框內(nèi)容為本地文件兼容IE,chrome,火狐瀏覽器
本文實現(xiàn)了利用JS保存頁面中文本框內(nèi)容到本地,并另存為指定文件擴展名與編碼類型,兼容IE,chrome,火狐等瀏覽器2018-02-02JavaScript自動化測試添加頁面DOM元素唯一ID方案示例
這篇文章主要為大家介紹了JavaScript自動化測試添加頁面DOM元素唯一ID方案示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09JS屬性scrollTop?clientHeight?scrollHeight理解學習
這篇文章主要為大家介紹了JS屬性scrollTop?clientHeight?scrollHeight理解學習,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-07-07