在設(shè)計 API 時,出于安全性等因素考慮,有時需要放棄使用自增 ID,使 ID 非連續(xù)且不可猜測。通??梢允褂?Hash id,UUID,雪花 ID 等來實現(xiàn)。
在最近的一個項目中,我嘗試使用雪花 ID。一通折騰下來發(fā)現(xiàn),逼格挺高,實現(xiàn)也挺簡單。然而當(dāng)我繼續(xù)擼起袖子與前端部分對接時,卻出現(xiàn)了 JS 精度丟失問題,因為存儲的 ID 是一個 unsigned bigint 型的值。(至于為什么會有精度丟失現(xiàn)象,這里就不具體解釋了,不清楚的可以自行搜索),本文主要介紹解決辦法。
想要解決這問題,基本原理也很簡單,就是把 ID 轉(zhuǎn)成字符串再返回給前端。
錯誤嘗試
一開始我想到的是使用 Laravel Eloquent 模型的模型訪問器。只要給需要轉(zhuǎn)換的模型加一個 getIdAttribute,將 ID 轉(zhuǎn)成字符串不就行了嘛?
如:AppModelsUser 模型里這樣寫:
/**
* @return string
*/
public function getIdAttribute()
{
return strval($this->attributes['id']);
}
但事實并非如此,屬性訪問器確實能讓 API 返回給前端的 ID 變?yōu)樽址?。但同時也會影響關(guān)聯(lián)模型插入、修改時的結(jié)果,例如,user 關(guān)聯(lián)的了 post 模型,使用 $user->posts()->saveMany(...); 這種方式保存的新的 posts 記錄,對應(yīng)的 user_id 會為空。
這也不難理解,因為模型訪問器是要參與模型相關(guān)處理的,訪問器將 ID 由數(shù)字轉(zhuǎn)為了字符串,自然會導(dǎo)致數(shù)據(jù)錯亂。
正確姿勢
冷靜下來決定先認(rèn)真思考再動手,查閱了官方文檔,才發(fā)現(xiàn) Resource 正是我想要的。Resource 只會影響返回給前端的數(shù)據(jù),我們可以通過自定義 Resource 來實現(xiàn) API 返回結(jié)果的結(jié)構(gòu)、類型轉(zhuǎn)換等功能。轉(zhuǎn)換個 ID 自然也不在話下。
為了省事,我直接修改 AppHttpResource 這個基類。只需要重載它的 toArray() 方法,在其中使用遞歸,對可能超出 JS 安全數(shù)值范圍的值進行轉(zhuǎn)換就可以了。大家也可以根據(jù)自己的實際情況,新建 Resource 類,如 UserResource 來處理。
?php
namespace App\Http\Resources;
use Illuminate\Http\Resources\Json\JsonResource;
class Resource extends JsonResource
{
/**
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request $request
*
* @return array
*/
public function toArray($request)
{
$parentReturn = parent::toArray($request);
foreach (array_keys($parentReturn) as $key) {
// 為方便演示這里把所有整型字段都轉(zhuǎn)成字符串
if (is_int($parentReturn[$key])) {
$parentReturn[$key] = strval($parentReturn[$key]);
}
// 關(guān)聯(lián)的字段,如 $user->post,相當(dāng)于遞歸處理
if (is_array($parentReturn[$key])) {
$parentReturn[$key] = new Resource($parentReturn[$key]);
}
}
return $parentReturn;
}
}
然后,在接口控制器中返回 Resource 返回數(shù)據(jù),整型字段值就會自動變?yōu)樽址恕?/p>
?php
namespace App\Http\Controllers;
use App\Http\Resources\Resource;
use App\Models\User;
use Illuminate\Http\Request;
class TestController extends Controller
{
/**
* @return \App\Http\Resources\Resource
*/
public function __invoke(Request $request)
{
$user = User::first();
return new Resource($user);
}
}
結(jié)果如下圖:

注意事項
因為這種辦法使用了遍歷,而且有遞歸處理,當(dāng)數(shù)據(jù)結(jié)構(gòu)復(fù)雜、數(shù)據(jù)量較大時可能會對性能造成一定影響。我這里算是比較偷懶取巧的寫法,如果對性能有追求,自定義 Resource 類,然后根據(jù)特定的已知的字段名來進行轉(zhuǎn)換會比較好
因為返回給前端的 ID 轉(zhuǎn)為了字符串,前端在進行比較判斷,特別是 === 判斷時要特別注意
到此這篇關(guān)于Laravel 自動轉(zhuǎn)換長整型雪花 ID 為字符串的實現(xiàn)的文章就介紹到這了,更多相關(guān)Laravel 長整型雪花ID轉(zhuǎn)換為字符串內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
您可能感興趣的文章:- laravel5.6實現(xiàn)數(shù)值轉(zhuǎn)換
- 在laravel中實現(xiàn)將查詢的對象轉(zhuǎn)換為多維數(shù)組的函數(shù)
- laravel 字段格式化 modle 字段類型轉(zhuǎn)換方法
- Laravel項目中timeAgo字段語言轉(zhuǎn)換的改善方法示例