摘要:至于一對(duì)一和一對(duì)多這兩種關(guān)系,可以觸類旁通。注意文件的下載或來源,請(qǐng)自行獲取。接下來,我們使用取得文章的標(biāo)簽,這個(gè)就是我們聲明多對(duì)多關(guān)系的方法。
免費(fèi)視頻教程地址 https://laravist.com/series/laravel-5-basic原文來自 https://laravist.com/article/18
Laravist是我剛剛上線的Laravel社區(qū),有任何與Laravel相關(guān)的問題可以到這里來問我,我會(huì)盡力去幫大家解決問題,后期會(huì)嘗試錄制一些視頻教程,形式大概是這樣的
https://laravist.com/lesson/1
前奏在開始正文之前,我們首先來說說在實(shí)際的開發(fā)中,經(jīng)常會(huì)接觸到幾種常見的對(duì)應(yīng)關(guān)系模式:
One-To-One //一對(duì)一 One-To-Many //一對(duì)多 Many-To-Many //多對(duì)多
不知道你對(duì)這些概念是一種什么樣的感受,如果是不太理解的。你可以將這些概念應(yīng)用到生活中,理解起來就很簡(jiǎn)單了,就舉一個(gè)與我們?cè)诰W(wǎng)上經(jīng)常見到的例子:
User-To-Profile // One-To-One User-To-Articles // One-To-Many Article-To-Comments // One-To-Many Articles-To-Tags // Many-To-Many
翻譯過來就是:
一個(gè)用戶對(duì)應(yīng)一個(gè)用戶檔案
一個(gè)用戶可以發(fā)表多篇文章
一篇文章可以有多個(gè)評(píng)論
而文章和標(biāo)簽確實(shí)多對(duì)多的關(guān)系,一篇文章可以有多個(gè)標(biāo)簽;一個(gè)標(biāo)簽可以屬于多篇文章
在這些關(guān)系模型中,最難實(shí)現(xiàn)的就是Many-To-Many這種多對(duì)多的關(guān)系,但是我們這個(gè)簡(jiǎn)單地博客并沒有用戶管理,也就是并沒有開放讓用戶注冊(cè),所以我們?cè)谶@里還是要挑戰(zhàn)一下難度,實(shí)現(xiàn)Articles-To-Tags這種Many-To-Many關(guān)系,借助Laravel的強(qiáng)大的Eloquent,實(shí)現(xiàn)這個(gè)功能還是比較順心的。至于一對(duì)一和一對(duì)多這兩種關(guān)系,可以觸類旁通。
創(chuàng)建tags表要實(shí)現(xiàn)Articles-To-Tags這種Many-To-Many關(guān)系,我們需要tags表和Tag模型,所以我們分別來創(chuàng)建之。
php artisan make:migration create_tags_table --create=tags
打開生成的migration文件,為up()方法增加一行代碼:
public function up() { Schema::create("tags", function (Blueprint $table) { $table->increments("id"); $table->string("name"); $table->timestamps(); }); }
這里我們?cè)黾恿?b>$table->string("name");這一行,這個(gè)字段表示為tags table添加一個(gè)name字段,代表標(biāo)簽的名字。
接下來,我們?yōu)?b>tags表創(chuàng)建一個(gè)Tag模型:
php artisan make:model Tag
生成了Tag模型之后,我們先不用去管Tag.php文件,因?yàn)槲覀冞€需要一張關(guān)系表article_tag,這個(gè)表只存tag_id和article_id,所以我們來創(chuàng)建之:
php artisan make:migration create_article_tag_table --create=article_tag
打開migration文件來為之加上tag_id和article_id這兩個(gè)字段:
public function up() { Schema::create("article_tag", function (Blueprint $table) { $table->increments("id"); $table->integer("article_id")->unsigned()->index(); $table->foreign("article_id")->references("id")->on("articles")->onDelete("cascade"); $table->integer("tag_id")->unsigned()->index(); $table->foreign("tag_id")->references("id")->on("tags")->onDelete("cascade"); $table->timestamps(); }); }
這里貌似就添加tag_id和article_id這兩個(gè)字段,但是用了很多行代碼,我們只要是理解下面這個(gè):
$table->foreign("article_id")->references("id")->on("articles")->onDelete("cascade");
foreign():外鍵
references():參照字段
on():參照表
onDelete():刪除時(shí)的執(zhí)行動(dòng)作
這里是跟著刪除,比如刪除了某篇文章,我們將article_tag中包含article_id一樣的記錄也刪除
最后,執(zhí)行migration生成article_tag表:
php artisan migrate
OK,生成這兩個(gè)表之后,我們就可以正式開始我們的工作了。
聲明Eloquent的關(guān)系Articles和Tags是多對(duì)多的關(guān)系,所以我們需要在Article.php中聲明下面的關(guān)系:
public function tags() { return $this->belongsToMany("AppTag"); }
在Tag.php,也同樣:
public function articles() { return $this->belongsToMany("AppArticle"); }
我們使用$this->belongsToMany()來表明Eloquent的關(guān)系,這里需要注意的是如果你的外鍵并不是article_id和tag_id,你需要在第三個(gè)參數(shù)進(jìn)行設(shè)置,寫成類似下面這樣:
public function articles() { return $this->belongsToMany("AppArticle","conversation_id"); }
OK,這樣,我們的多對(duì)多關(guān)系就聲明完畢了。
使用Select2在開始之前,我們使用tinker生成幾個(gè)tag,過程就不演示了,最后是這樣的:
然后,為了更好地用戶體驗(yàn),我們引入Select2,這個(gè)對(duì)選擇多個(gè)選項(xiàng)的時(shí)候表現(xiàn)得異常完美。
Select2 用法:https://select2.github.io/examples.html
我們?cè)赼pp.blade.php引入Select2的css文件和js文件:
在
標(biāo)簽內(nèi),我們還引入了jquery,因?yàn)閟elect2依賴于jquery,所以。注意文件的下載或來源,請(qǐng)自行獲取。引入之后,我們就可以在文件創(chuàng)建的頁面依舊使用我們的Form來生成我們的選擇框了,來到articles/create.blade.php文件,在published_at下面添加一個(gè)輸入表單:
{!! Form::label("tag_list","選擇標(biāo)簽") !!} {!! Form::select("tag_list[]",$tags,null,["class"=>"form-control js-example-basic-multiple","multiple"=>"multiple"]) !!}
這里需要注意的是tag_list[],如果我們只是使用tag_list,就只能選到一個(gè)標(biāo)簽,如果我們需要選擇多個(gè),我們需要已數(shù)組的形式來儲(chǔ)存我們的標(biāo)簽,還有一個(gè)就是指定一下"multiple"=>"multiple",就是開啟支持多選模式。然后$tags就是我們需要從數(shù)據(jù)庫獲tags表取到得數(shù)據(jù),所以自然而然,我們到ArticleController中的create()方法中,稍微修改一下代碼:
public function create() { $tags = Tag::lists("name", "id"); //為了在界面中顯示標(biāo)簽name,id為了在保存文章的時(shí)候使用。 return view("articles.create",compact("tags")); }
這里我們使用lists()方法將Tag中(對(duì)于tags數(shù)據(jù)表)name和id以一個(gè)Eluqoent的方式返回,你可以使用dd($tags),來看看。恩,這個(gè)時(shí)候來看看我們的create頁面:
這時(shí)候我們發(fā)現(xiàn),樣式并沒有Select2那么好看,那是因?yàn)槲覀冞€沒有初始化Select2,所以我們?cè)赾reate.blade.php寫幾行簡(jiǎn)單地js代碼:
在@endsection緊接著的上一行加上上面的代碼,這里我們使用jquery的選擇器,然后調(diào)用select2();來初始化我們的選擇框,再來看看效果:
很完美,我們將整個(gè)UI完善得還不錯(cuò),我們用dd();來看看我們表單提交過來的是什么,在ArticleController中的store()方法中添加一行代碼:
dd($request->all());
我們來看看效果:
我們看到得tag_list是一個(gè)數(shù)組,里面的值并不是我們選擇的標(biāo)簽的name,而是標(biāo)簽的id,這樣我們就可以使用laravel提供的attach()來添加我們的標(biāo)簽了,這個(gè)attach()接受一個(gè)id的數(shù)組,這里正好!,所以我們來稍微來修改一下store()方法:
public function store(RequestsStoreArticleRequest $request) { $input = $request->all(); $input["intro"] = mb_substr($request->get("content"),0,64); $article = Article::create($input); $article->tags()->attach($request->input("tag_list")); return redirect("/"); }
我們這里首先將Article::create($input)賦予$article變量(Eloquent對(duì)象),然后使用$article->tags()->attach()來添加標(biāo)簽,并將我們的標(biāo)簽數(shù)組傳給attach()方法,我們來看看有沒有成功:
這里的文章是發(fā)表成功了,我們?cè)賮砜纯次覀兊臉?biāo)簽是否添加成功,來看看我們的article_tag表:
是添加了三個(gè)標(biāo)簽,但是我們發(fā)現(xiàn)這個(gè)created_at和updated_at貌似有點(diǎn)問題,我們來修復(fù)一下,在Article.php中的tags()方法中:
public function tags() { return $this->belongsToMany("AppTag")->withTimestamps(); }
我們?cè)诤竺嬷苯邮褂?b>withTimestamps()來同步我們的時(shí)間,我們?cè)賮碓囈辉嚕?/p>
再來看看我們的數(shù)據(jù)庫:
看到最后的兩個(gè)記錄,很完美。
在視圖中顯示我們的tags我們既然有了標(biāo)簽,我們?yōu)槭裁床粊韺⑺故境鰜砟???b>articles/index.blade.php中,我們來將文件的標(biāo)簽輸出一下:
{{ $article->title }}
我們?cè)?b>
接下來,我們使用$article->tags取得文章的標(biāo)簽,這個(gè)tags就是我們聲明多對(duì)多關(guān)系的tags()方法。我們來看看效果:
我們發(fā)現(xiàn)我們的多少分鐘之前都是英文,那是因?yàn)槲覀儧]有設(shè)置Carbon,我們來修復(fù)一下,在app/Providers/AppServiceProvider.php中的boot()方法添加下面這一行:
CarbonCarbon::setLocale("zh");
然后刷新,見證一下奇跡吧:
總結(jié)到這里我們利用laravel提供的attach()方法將基本的多對(duì)多關(guān)系實(shí)現(xiàn)了,并且還稍微美化了一下輸出,將published_at字段完美呈現(xiàn)。接下來我打算說一說怎么實(shí)現(xiàn)修改文章了,這是一個(gè)必走的流程嘛。所以。。。
最后,Happy Hacking
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/21125.html
摘要:來看看具體的步驟注冊(cè)路由在中,注冊(cè)我們的編輯頁面的路由這個(gè)路由接受一個(gè)參數(shù),意為文章的,我們會(huì)需要根據(jù)這個(gè)來查詢我們要修改的文章。然后渲染視圖,并將查詢到的和傳給視圖。下面我打算再開一個(gè)系列說說的新特性 原文來自https://laravist.com/article/20 免費(fèi)視頻教程地址 https://laravist.com/series/laravel-5-basic La...
摘要:原文發(fā)表在我的個(gè)人網(wǎng)站系列入門教程四最適合中國(guó)人的教程本教程示例代碼見大家在任何地方卡住,最快捷的解決方式就是去看我的示例代碼。 原文發(fā)表在我的個(gè)人網(wǎng)站:Laravel 5 系列入門教程(四)【最適合中國(guó)人的 Laravel 教程】 本教程示例代碼見:https://github.com/johnlui/Learn-Laravel-5 大家在任何地方卡住,最快捷...
摘要:原文發(fā)表在我的個(gè)人網(wǎng)站系列入門教程一最適合中國(guó)人的教程本教程示例代碼見大家在任何地方卡住,最快捷的解決方式就是去看我的示例代碼。在此我推薦一個(gè)全量中國(guó)鏡像。 原文發(fā)表在我的個(gè)人網(wǎng)站:Laravel 5 系列入門教程(一)【最適合中國(guó)人的 Laravel 教程】 本教程示例代碼見:https://github.com/johnlui/Learn-Laravel-5 大...
摘要:本文來自原文鏈接歡迎作客我們的學(xué)習(xí)群最簡(jiǎn)單的,里得到的是一組數(shù)據(jù),得到的是一個(gè)數(shù)據(jù)。 本文來自pilishen.com----原文鏈接; 歡迎作客我們的php&Laravel學(xué)習(xí)群:109256050 最簡(jiǎn)單的,laravel里get()得到的是一組數(shù)據(jù),first()得到的是一個(gè)model數(shù)據(jù)。 從形式上,laravel里每一個(gè)model數(shù)據(jù)(record),在取出的時(shí)候都是用的PH...
摘要:原文來自免費(fèi)視頻教程地址上一篇寫了一些的基本用法和給視圖傳遞變量的幾種方式,這一節(jié)我們來說說跟數(shù)據(jù)庫打交道的數(shù)據(jù)庫配置和強(qiáng)大的。 原文來自: https://jellybool.com/post/programming-with-laravel-5-database-and-eloquent-model 免費(fèi)視頻教程地址 https://laravist.com/series/lara...
閱讀 2225·2021-09-07 09:58
閱讀 3402·2019-08-30 14:07
閱讀 1310·2019-08-29 12:32
閱讀 677·2019-08-29 11:06
閱讀 3703·2019-08-26 18:18
閱讀 3742·2019-08-26 17:35
閱讀 1391·2019-08-26 11:35
閱讀 619·2019-08-26 11:35