Laravel教程 三:视图变量传递和Blade

JellyBool

JellyBool

上一篇我们简单地说了Router,Views和Controllers的工作流程,这一次我就按照上一篇的计划,来说说下面几个内容:

  1. 向视图中传递变量

  2. Blade模板的用法

向视图中传递变量

我们在开发web应用当中,通常都不是为了写静态页面而生的,我们需要跟数据打交道,那么这个时候,问题就来了,在一个MVC的框架中,怎么将数据传给视图呢?比如我们要在 ArticleControllerindex 方法的视图输出一个 $title 的变量,在Laravel中,有下面几种常见的方法:

使用with()方法

 public function index()
    {
        $title = '文章标题1';
        return view('articles.lists')->with('title',$title);
    }

这样的 with('title',$title) 中,第一个 'title' 就是key,第二个 $title 就是值,这样我们就可以在我们的 articles/lists.blade.php 中输出这个变量了:

<body>
<h1><?php echo $title; ?></h1>

</body>

刷新我们的 blog.dev ,就可以看到类似这样的页面了:

替代文字

而在blade引擎中,我们可以这样输出变量:

<body>
<h1>{{ $title }}</h1>

</body>

其实在blade引擎中, {{ $title }} 会被解析为类似 这样的输出 <?php echo $title; ?> ,不过这里的 {{ }} 符号会将数据原样输出,比如你将 $title 写成这样:

 public function index()
    {
        $title = '<span style="color: red">文章</span>标题1';
        return view('articles.lists')->with('title',$title);
    }

这个时候你用 {{ $title }} 输出,会看到类似下面这样:

替代文字

如果你想将 $title 作为页面元素渲染输出,你需要这样写:

<h1>{!! $title !!}</h1>

替代文字

这里的 {{ }}{!! !!} 是blade的最基础的用法,这两个我们会用得特别多,后面我会详细说说blade的用法。

直接给view()传参数

使用这个方法的时候,你可以这样写:

public function index()
    {
        $title = '<span style="color: red">文章</span>标题1';
        return view('articles.lists',['title'=>$title]);
    }

刷新页面,你依然会看到一样的输出。这里需要说明一下,如果你传多个变量,比如:

 public function index()
    {
        $title = '<span style="color: red">文章</span>标题1';
        $intro = '文章一的简介';
        return view('articles.lists',[
                                        'title'=>$title,
                                        'introduction'=>$intro
                                        ]);
    }

在传递的数组中:

[
'title'=>$title,
'introduction'=>$intro
]

每一个key会在视图中作为变量,而 value 就作为变量的值。所以在视图中我们需要这样输出:

<body>
<h1>{!! $title !!}</h1>
<p>{{ $introduction }}</p>
</body>

这里应写成 {{ $introduction }} ,而不是 {{ $intro }}

使用compact

使用compact是这样写的:

 public function index()
    {
        $title = '<span style="color: red">文章</span>标题1';
        $intro = '文章一的简介';
        return view('articles.lists',compact('title','intro'));
    }

compact() 的字符串可以就是变量的名字,多个变量名用逗号隔开。这个时候注意更改视图的变量输出。

以上就是Laravel中常用的几种向视图传递变量的方法,选择一种你喜欢的方式并坚持这一种写法就可以了,我是使用第三种。

Blade的基本用法

上面的内容介绍了一点点blade的语法,这里我们再统一介绍blade,说说下面几个比较常用的:

@yield()
@extends()
@if() and @unless()
@foreach()

@yield()@extends() 通常会结合者使用,实现我们通常所说的layouts布局:就是在web开发的过程中,我们将一些公用的部分如 headerfooter 等直接放在一个视图文件中,然后在使用的使用直接继承 (使用@extends) 就可以了,比如我们在 resources/views/ 文件夹之下创建一个 app.blade.php

<!DOCTYPE html>
<html class="no-js" lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
    <title>Laravel 5 教程</title>
    <link rel='stylesheet' href="/css/all.css" type='text/css' media='all'/>
    <script type='text/javascript' src="/js/all.js"></script>
</head>
<body>
<div id="wrapper">

    @yield('content')

    <nav class="nav-container group" id="nav-footer">
        <div class="nav-wrap">
            <ul class="nav container group">
                <li class="menu-item">
                    <a href="/" rel="nofollow" target="_blank">Laravel 5 Blog</a>
                </li>
            </ul>
        </div>
    </nav>
</div>
</body>
</html>

写上这么一些内容,其中css这个 href="/css/all.css" ,需要我们手动在 public/ 文件夹之下创建css/文件夹,并创建all.css这个文件,对于js的src="/js/all.js"也是同理,这两个文件是为了后面的页面美化而做的提前准备。

注意到@yield('content')这个语法,这里就是说,这里有一个content的内容区域,如果某个页面继承了这个app.blade.php,然后那个页面就可以动态改变@yield('content')的内容了。比如我们在articles/lists.blade.php中,我们继承一下app.blade.php:

@extends('app')
@section('content')
<h1>{!! $title !!}</h1>
<p>{{ $intro }}</p>
@endsection 

这里的第一行@extends('app')就是声明这个页面继承于app.blade.php,也就是我们的articles/lists.blade.php可以使用到all.cssall.js文件,然后@section('content')就是对于app.blade.php@yield('content'),表明就是:在渲染加载articles/lists.blade.php的时候,@yield('content')这部分内容会被替换为下面的内容:


<h1>{!! $title !!}</h1>
<p>{{ $intro }}</p>

替代文字

@if()通常是用于在视图中根据某些条件来判断是否该显示某些内容,比如我们可以很“无聊”地试试这样:

public function index()
    {
        $first = 'jelly';
        $last = 'bool';
        return view('articles.lists',compact('first','last'));
    }

在views文件中,我们使用一下@if():

@extends('app')
@section('content')
 @if($first == 'jellybool')
 <h1>{{ $first }}</h1>
 @else
 <h1>{{ $last  }}</h1>
@endif
@endsection

刷新一下就可以看到页面的输出,为$last的值。

替代文字

上面的@if(),还有一个可以使用的标签就是@unless()@unless()就可以理解为 if( ! ),就是if not 这样理解就OK。

@foreach()用于循环输出变量,比如:

public function index()
    {
        $first = ['jelly','bool'];
        return view('articles.lists',compact('first'));
    }

我们传一个数组给视图,然后,我们就可以使用@foreach()循环输出了:

@extends('app')
@section('content')
@foreach( $first as $name)
    <h1> {{ $name }}</h1>
@endforeach
@endsection

刷新一下页面,就可以看到循环的结果了:

替代文字

blade的更多知识,可以参考文档:

http://laravel.com/docs/5.1/blade

下一节

貌似上面配置数据库链接之后还没有用到,不用着急,我接下来的一篇会具体说说Eloquent的用法,这个在Laravel中是很有代表性的部分,希望你可以学到一点东西。

Happy Hacking

本文由 JellyBool 创作, 转载和引用遵循 署名-非商业性使用 2.5 中国大陆 进行许可。

共有 19 条评论

TimeIsGoOn

模板里变量的写法看不清我来补充上

@extends('app')
@section('content') 
@if($first == 'jellybool') 
	<h1>{!!  $title !!}</h1> 
@else 
	<h1>today is nice</h1>
@endif
<ul>
@foreach($people as $person)
	<li>
	  <ul>
	     	@foreach( $person as $v)
	     	  <li>{$v}</li>
	     	@endforeach
	  </ul>
	</li>
@endforeach
</ul>

@endsection
JellyBool

@TimeIsGoOn 发现一旦有@这个字符在前面,代码块的Markdown解析就出问题,我有时间修改一下

woailuosj

请问视图中如何引用静态资源比较好?

ziyouwa

$title = ‘文章标题1’;

直接显示红色字体了,

{ $title } 显示文章标题1
{!! $title !!} 显示红色字体

是5.1的变化?!

dudushuang

里面不显示的空格是啥子嘛undefined

Anonymous

找了好久才找到讲的这么好的博客,别的站都在翻译官方文档,你却可以用自己的语言把体会和经验写出来,很受用。多谢分享。

Anonymous

谢谢,常来看看或者推荐给身边的人也可以。还可以到我新搭建的站点来看看:https://laravist.com/

我会在laravist上面录视频教程,虽然还是在探索当中

Anonymous

讲的很好啊,如果有对应的视频录制麻烦通知我一下,邮箱472542015@qq.com

Anonymous

感谢楼主分享如此的详细教程,[鼓掌][鼓掌][鼓掌],有没有一些开源的项目,再分享一下,我想在项目中体会,再加上楼主分享的一些小细节可能更加深刻一点

Anonymous

写的不是一般的好,条理清晰,语言明了,没有啰嗦的言语, 让人看了受益匪浅。。[赞][赞][赞][赞]

Anonymous

谢谢夸奖,视频教程可以看这里 https://laravist.com/series/laravel-5-basic

Anonymous

{ } 和 {!! !!} 的作用是一样的,对吗?

Anonymous

并不一样。前者转义,后者不转义

Anonymous

看文档看的模模糊糊的,你说的很清晰

Anonymous

感谢楼主分享 学到了很多 是我看到的最详细的教程了。

Anonymous

语言表述通俗易懂,很多知识点一点即破,谢谢楼主。向楼主学习[嘻嘻]

hehorange

模板里引入的两个文件 all.css 和 all.js 哪里有啊?

Cyclone

教主我问个问题,视图变量传递不能传递到网页的母版页里面吗?我的问题详细如下:

比如我有一个母版页app,然后有一个子网页index。我在一个路由函数里面定义量变量As和变量Bs,他们两个变量都从数据库中获取数据了,然后我写了这个函数:

return view('index',['As'=>$As, 'Bs'=>$Bs]);
但是我设置了变量Bs在index的母版页app里面显示,我心想既然index里面都有母版页app的代码那应该可以显示出来,但是运行的时候却报出错误是在app网页里面没有定义$Bs变量,无法运行。请问这种情况该要怎么办?