博客
required : 必须值验证属性||CRequiredValidator 的别名, 确保了特性不为空. [['字段名1','字段名2'],required] //字段1 2 必填 [['字段名'],required,'requiredValue'=>'必填值','message'=>'提示信息']; email : 邮箱验证||CEmailValidator 的别名,确保了特性的值是一个有效的电邮地址. ['email', 'email']; match : 正则验证||CRegularExpressionValidator 的别名, 确保了特性匹配一个正则表达式. [['字段名'],'match','pattern'=>'正则表达式','message'=>'提示信息']; [['字段名'],'match','not'=>ture,'pattern'=>'正则表达式','message'=>'提示信息']; /*正则取反*/ url : 网址||CUrlValidator 的别名, 确保了特性是一个有效的路径. ['website', 'url', 'defaultScheme' => 'http']; captcha(验证码)||CCaptchaValidator 的别名,确保了特性的值等于 CAPTCHA 显示出来的验证码. ['verificationCode', 'captcha']; safe : 安全 ['description', 'safe']; compare :(比较) CCompareValidator 的别名, 确保了特性的值等于另一个特性或常量. ['repassword', 'compare', 'compareAttribute' => 'password','message'=>'两次输入的密码不一致!'], //compareValue:比较常量值 operator:比较操作符 ['age', 'compare', 'compareValue' => 30, 'operator' => '>=']; default : 默认值||CDefaultValueValidator 的别名, 为特性指派了一个默认值. ['age', 'default', 'value' => null]; exist : 存在||CExistValidator 的别名, 确保属性值存在于指定的数据表字段中. ['字段名', 'exist']; file : 文件||CFileValidator 的别名, 确保了特性包含了一个上传文件的名称. ['primaryImage', 'file', 'extensions' => ['png', 'jpg', 'gif'], 'maxSize' => 1024*1024*1024] filter : 滤镜||CFilterValidator 的别名, 使用一个filter转换属性. //'skipOnArray' => true 非必填 [['username', 'email'], 'filter', 'filter' => 'trim', 'skipOnArray' => true]; in : 范围||CRangeValidator 的别名, 确保了特性出现在一个预订的值列表里. ['level', 'in', 'range' => [1, 2, 3]]; unique : 唯一性||CUniqueValidator 的别名, 确保了特性在数据表字段中是唯一的. ['字段名', 'unique'] 补充:联合唯一索引rule规则 [ ['app_id', 'group_id'], 'unique', 'targetAttribute' => ['app_id', 'group_id'], 'message' => 'app_id和group_id已经被占用!' ], integer : 整数 ['age', 'integer']; number : 数字 ['salary', 'number']; double : 双精度浮点型 ['salary', 'double']; date : (日期) [['from', 'to'], 'date']; string : 字符串 ['username', 'string', 'length' => [4, 24]]; boolean : 是否为一个布尔值||CBooleanValidator 的别名 ['字段名', 'boolean', 'trueValue' => true, 'falseValue' => false, 'strict' => true]; image :是否为有效的图片文件 [ 'primaryImage', 'image', 'extensions' => 'png, jpg', 'minWidth' => 100, 'maxWidth' => 1000, 'minHeight' => 100, 'maxHeight' => 1000 ] each:遍历,ids 和 product_ids 是数字的集合 [['ids', 'product_ids'], 'each', 'rule' => ['integer']], 自定义rules: ['password', 'validatePassword'], /** * Validates the password. * This method serves as the inline validation for password. * * @param string $attribute the attribute currently being validated * @param array $params the additional name-value pairs given in the rule */ public function validatePassword($attribute, $params) { if (!$this->hasErrors()) { $user = $this->getUser(); if (!$user || !$user->validatePassword($this->password)) { $this->addError($attribute, '账号或者密码错误!'); } } }
29天前 喜欢(3) 浏览(185) 评论(2)
博客
基础 Gridview 插件被应用于数据呈现,它提供了很多功能,如浏览、排序、分页和数据过滤。 下面是一个简单的 Gridview 应用实例: <?= GridView::widget([ 'dataProvider' => $dataProvider, 'columns' => [ 'id', 'name', 'created_at:datetime', // ... ], ]) ?> 用法示例 1.在例头添加排序,在cloumns中加入以下代码 ['class' => 'yii\grid\SerialColumn'] 2.列表勾选框,在cloumns中加入以下代码 ['class' => 'yii\grid\CheckboxColumn'], 3.列数据快速格式化:时间戳转化为时间格式显示 'created_at:datetime', 4.显示关联表数据:前提是在model中有关联关系,例如:getAuthor() 'author.name', //获取关联表author的name的值 5.列表中显示图片:显示一张50*100的图片,label_img为图片地址 'label_img'=>[ 'label' => '标签图', 'format' => [ 'image', [ 'height' =>50, 'width' => 100 ] ], 'value' => function($model){ return $model->label_img; } ], 6.显示状态,且带过滤 [ 'attribute' => 'is_valid', 'label' => '发布状态', 'value' => function($model) { return $model->is_valid == 0 ? '未发布' : '发布'; }, 'filter' => [ 0 => '未发布', 1 => '发布' ] ], 7.显示带html标签的例值:正常情况下是过滤html标签的 [ 'attribute' => 'content', 'format' => 'raw', 'value' => function ($model) { return $model->content; }, ], 8.自定义按钮:{view} {update} {delete} 为默认,可以不填显示默认,也可以覆盖重新定义 [ 'class' => 'yii\grid\ActionColumn', 'template' => '{test} {view} {update} {delete}', 'header' => '操作', 'buttons' => [ 'test' => function ($url, $model, $key) { return Html::a('测试按钮', $url, ['data-method' => 'post','data-pjax'=>'0'] ); }, 'delete'=> function ($url, $model, $key){ return Html::a('删除', ['delete', 'id'=>$model->id],[ 'data-method'=>'post', //POST传值 'data-confirm' => '确定删除该项?', //添加确认框 ] ) ; } ], ], 9.修改列表顶部分页信息 //{begin}:当前列的第一个元素序号 //{end}:当前页的最后一个元素序号 //{count}:当前页的元素总数 //{totalCount}:所有元素总数 //{page}:当前页 //{pageCount}:总页数 <?= GridView::widget([ 'dataProvider' => $dataProvider, 'summary' => '第{begin}-{end}页,共计{totalCount}篇文章', ... 以上为gridview的一些基本操作,当然还有很多其他情景,但要活学活用,举一反三
1月前 喜欢(3) 浏览(233) 评论(2)
讨论
在提问的时候,适当使用一些小技巧,会让老师快速get到你的问题在哪里,这样会使问题更快被解答喔~ 提问技巧 标题:简洁、易懂,完整表达主题;避免无意义、模糊、口语化标题。 提问内容: 客观描述问题:中性客观描述问题,保证问题的可读性,避免过于简单或模糊; 提供编程环境:问题描述注明编程环境信息,包括操作系统、语言、使用的软件版本等; 输出报错信息:过程中出现异常的话,需提供重现异常的步骤; 输出错误信息:编码过程中的技术问题出现错误应该输出日志或控制台的全部信息,以便更好排查错误; 注明课程时间:针对课程的问题,请在问题描述中注明出现时间,便于及时解答; 排版规范合理:注意文本,代码,标点、语法使用规范,排版合理。 提倡有偿问答:对于有能力的学者,可以提供有偿提问,勾引大神们快速响应(提问者自愿为主,答题者不可强求) 注意: 不要在提问内容框里长篇粘入代码,只需粘出关键代码并加以描述即可 标题不等于提问内容,标题应简明扼要,提问内容应详细描述,二者尽量不要重复。 提问尽量以 PHP/Yii2.0 的内容为主。 千万不要贴一段业务报错代码,问为什么这里报错了,代码调试可以帮您解决大部分的问题。 学员和老师要互相尊重,尊重是有效解决问题的前提,对于人身攻击的问题管理员是有权隐藏的哦~ 说完了如何提问,那么大!神!般!的你在问答区看到了其他小伙伴提出的问题,想要回答时需要注意些什么呢?^_^ 回答建议: 回答保证内容的准确性,不模糊,能实际解决提问者疑问; 回答保证内容通俗易懂,语句通顺、有条理,排版利于阅读; 回答保证内容全面,避免过于简单,可对回答做详细阐释、扩展和引导; 回答保证内容与问题相关,避免所问非所答和过多的重复内容; 添加图片或代码应附上说明,代码用对应代码块包裹并合理排版; 核心思想:要持着“授人以鱼不如授人以渔”的心来回答提问者的问题~
1月前 喜欢(3) 浏览(256) 评论(4)
Wiki
查询一个作者(Author)写的所有书籍(Book),其中Author 和 Book 是对应的 Model 作者模型:Author Class Author extend \yii\db\ActiveRecord { ··· public function getBook() { return $this->hasMany(Book::className(), ['author_id'=>'id']); } ··· } 书籍模型:Book Class Author extend \yii\db\ActiveRecord { ··· } 查询一个id=1的作者的信息及其所写的所有书籍: $authorInfo = Author::find()->where(['id'=>1])->with('book')->one(); hasOne 是1对1,hasMany 是 1对多,写法基本一致;
24天前 喜欢(2) 浏览(144) 评论(0)
博客
打开配置文件将下面代码添加到 components => [...]中(例:高级版默认配置在/common/config/main-local.php) 'mailer' => [ 'class' => 'yii\swiftmailer\Mailer', 'viewPath' => '@common/mail', 'useFileTransport' => false, //这里一定要改成false,不然邮件不会发送 'transport' => [ 'class' => 'Swift_SmtpTransport', 'host' => 'smtp.163.com', 'username' => 'xianan_huang@163.com', 'password' => '*********', //如果是163邮箱,此处要填授权码 'port' => '25', 'encryption' => 'tls', ], ], 在控制器中调用: $mail = \Yii::$app->mailer->compose() ->setFrom(['xianan_huang@163.com' => 'Yii 中文网']) ->setTo('391430388@qq.com') ->setSubject('邮件发送配置') //->setTextBody('Yii中文网教程真好 www.yii-china.com') //发布纯文字文本 ->setHtmlBody("<br>Yii中文网教程真好!www.yii-china.com") //发布可以带html标签的文本 ->send(); if($mail) echo 'success'; else echo 'fail'; 注意:很多报错原因都是因为163邮箱的smtp没有开,进入邮箱设置一下 默认不开启smtp要绑定手机之后才能开启
29天前 喜欢(2) 浏览(171) 评论(3)
博客
如果做API的话,如何使别人再调用你的接口时能够有一个统一标准的json或者jsonp格式,然而 json响应的格式和内容,每个人的约定都是有差异的,所以我们必须再数据出去之前做一定的处理。 1.首先我们需要初始化去调用beforeSend,因为我们需要对beforesend做一些处理,以下是init初始化处理代码: /** * (non-PHPdoc) * @see \yii\base\Object::init() */ public function init() { parent::init(); //绑定beforeSend事件,更改数据输出格式 Yii::$app->getResponse()->on(Response::EVENT_BEFORE_SEND, [$this, 'beforeSend']); } 2.然后我们就需要对beforesend进行处理,处理点有下面几个重点: 1>更改数据输出格式 2>默认情况下输出Json数据 3>如果客户端请求时有传递$_GET['callback']参数,输出Jsonp格式 4>请求正确时数据为 {"success":true,"data":{...}} 5>请求错误时数据为 {"success":false,"data":{"name":"Not Found","message":"页面未找到。","code":0,"status":404}} 6>具体代码如下: /** * 更改数据输出格式 * 默认情况下输出Json数据 * 如果客户端请求时有传递$_GET['callback']参数,输入Jsonp格式 * 请求正确时数据为 {"success":true,"data":{...}} * 请求错误时数据为 {"success":false,"data":{"name":"Not Found","message":"页面未找到。","code":0,"status":404}} * @param \yii\base\Event $event */ public function beforeSend($event) { /* @var $response \yii\web\Response */ $response = $event->sender; $isSuccessful = $response->isSuccessful; if ($response->statusCode>=400) { //异常处理 if (true && $exception = Yii::$app->getErrorHandler()->exception) { $response->data = $this->convertExceptionToArray($exception); } //Model出错了 if ($response->statusCode==422) { $messages=[]; foreach ($response->data as $v) { $messages[] = $v['message']; } //请求错误时数据为 {"success":false,"data":{"name":"Not Found","message":"页面未找到。","code":0,"status":404}} $response->data = [ 'name'=> 'valide error', 'message'=> implode(" ", $messages), 'info'=>$response->data ]; } $response->statusCode = 200; } elseif ($response->statusCode>=300) { $response->statusCode = 200; $response->data = $this->convertExceptionToArray(new ForbiddenHttpException(Yii::t('yii', 'Login Required'))); } //请求正确时数据为 {"success":true,"data":{...}} $response->data = [ 'success' => $isSuccessful, 'data' => $response->data, ]; $response->format = Response::FORMAT_JSON; \Yii::$app->getResponse()->getHeaders()->set('Access-Control-Allow-Origin', '*'); \Yii::$app->getResponse()->getHeaders()->set('Access-Control-Allow-Credentials', 'true'); //jsonp 格式输出 if (isset($_GET['callback'])) { $response->format = Response::FORMAT_JSONP; $response->data = [ 'callback' => $_GET['callback'], 'data'=>$response->data, ]; } } 3.针对请求可能会发生一些异常,同样我们也需要对异常进行一些标准化处理,将异常转换为array输出,具体代码如下: /** * 将异常转换为array输出 * @see \yii\web\ErrorHandle * @param \Exception $exception * @return multitype:string NULL Ambigous <string, \yii\base\string> \yii\web\integer \yii\db\array multitype:string NULL Ambigous <string, \yii\base\string> \yii\web\integer \yii\db\array */ protected function convertExceptionToArray($exception) { if (!YII_DEBUG && !$exception instanceof UserException && !$exception instanceof HttpException) { $exception = new HttpException(500, Yii::t('yii', 'An internal server error occurred.')); } $array = [ 'name' => ($exception instanceof Exception || $exception instanceof ErrorException) ? $exception->getName() : 'Exception', 'message' => $exception->getMessage(), 'code' => $exception->getCode(), ]; if ($exception instanceof HttpException) { $array['status'] = $exception->statusCode; } if (YII_DEBUG) { $array['type'] = get_class($exception); if (!$exception instanceof UserException) { $array['file'] = $exception->getFile(); $array['line'] = $exception->getLine(); $array['stack-trace'] = explode("\n", $exception->getTraceAsString()); if ($exception instanceof \yii\db\Exception) { $array['error-info'] = $exception->errorInfo; } } } if (($prev = $exception->getPrevious()) !== null) { $array['previous'] = $this->convertExceptionToArray($prev); } return $array; } 好了,这样我们就有了标准同一个的api接口返回数据格式了,在调用接口的人员也不用为了格式不统一感到烦恼
3天前 喜欢(1) 浏览(30) 评论(0)
置顶 博客
概念 把一个大型的单个应用程序和服务拆分为数个甚至数十个的支持微服务,它可扩展单个组件而不是整个的应用程序堆栈,从而满足服务等级协议。 传统的开发模式就是把所有功能都放在一个包里,基本不存在依赖,这样的优势在于开发简单,集中式管理,功能都在本地,不存在分布式的管理和调度消耗。但缺点也很明显:效率低,开发都在同一个项目改代码,相互等待,冲突不断。稳定性差,一个微小的问题,都可能导致整个应用挂掉。另外在资源利用上表现出明显的劣势,比如电商双11大促场景,下单压力非常大,评价的压力相对较少,那么我们希望临时增配应对双11的大流程,只能全部增配,而不能定点只对订单服务增配。所以微服务的架构开始慢慢流行并应用于大型的网站平台。 那么引入今天的主题,Yii 如何做微服务?Yii 可以轻松使用,而不需要基本和高级模板中包含的功能。换句话说,Yii 已经是一个微框架。不需要由模板提供的目录结构与 Yii 一起工作。 安装 Yii 为您的项目创建一个目录并将工作目录更改为该路径。示例中使用的命令是基于 Unix 的,但在 Windows 中也存在类似的命令。 mkdir micro-app cd micro-app Note:需要一些 Composer 的知识才能继续。如果您还不知道如何使用 composer,请花些时间阅读 Composer 指南。 使用您最喜爱的编辑器在 micro-app 目录下创建 composer.json 文件并添加以下内容: { "require": { "yiisoft/yii2": "~2.0.0" }, "repositories": [ { "type": "composer", "url": "https://asset-packagist.org" } ] } 保存文件并运行 composer install 命令。这将安装框架及其所有依赖项。 创建项目结构 安装框架之后,需要为此应用程序创建一个入口点。入口点是您尝试打开应用程序时将执行的第一个文件。出于安全原因,建议将入口点文件放在一个单独的目录中,并将其设置为Web根目录。 创建一个 web 目录并将 index.php 放入其中,内容如下: <?php // comment out the following two lines when deployed to production defined('YII_DEBUG') or define('YII_DEBUG', true); defined('YII_ENV') or define('YII_ENV', 'dev'); require(__DIR__ . '/../vendor/autoload.php'); require(__DIR__ . '/../vendor/yiisoft/yii2/Yii.php'); $config = require __DIR__ . '/../config.php'; (new yii\web\Application($config))->run(); 还要创建一个名为 config.php 的文件,它将包含所有的应用程序配置: <?php return [ 'id' => 'micro-app', //设置`micro-app`的根目录 'basePath' => __DIR__, // 控制器所在目录。 'controllerNamespace' => 'micro\controllers', // 设置命名空间为 micro 'aliases' => [ '@micro' => __DIR__, ], //默认访问地址 'defaultRoute' => 'home/index', 'components' => [ //请求配置 'request' => [ 'cookieValidationKey' => 'test&123456', 'parsers' => [ 'application/json' => 'yii\web\JsonParser', ] ], //Url 美化 'urlManager' => [ 'enablePrettyUrl' => true, 'showScriptName' => false, 'enableStrictParsing' => false, 'rules' => [ '<controller:\w+>/<action:\w+>/<id:\w+>' => '<controller>/<action>', ], ], //数据库配置 'db' => [ 'class' => 'yii\db\Connection', 'dsn' => 'mysql:host=localhost;dbname=micro', 'username' => 'root', 'password' => '数据库密码', 'charset' => 'utf8', ], ], ]; Info:尽管配置可以保存在 index.php 文件中,建议单独使用它。 这样它也可以用于控制台应用程序,如下所示。 您的项目现在已经准备进行编码了。尽管由您决定项目目录结构,只要您遵守命名空间即可。 创建第一个控制器 在创建控制器之前,创建一个 controllers/base 目录并创建一个基础控制器 BaseController。 <?php namespace micro\controllers\base; use yii\web\Controller; class BaseController extends Controller { //关闭 csrf 验证 public $enableCsrfValidation = false; } 然后在 controller 文件夹下面 新建一个 SiteController.php,这是默认的 控制器将处理没有路径信息的请求。 <?php namespace micro\controllers; use yii\web\Controller; class HomeController extends BaseController { public function actionIndex() { return '欢迎来到 Yii2.0 微服务!'; } } 如果您想为此控制器使用不同的名称,则可以配置 yii\base\Application::$defaultRoute 进行更改。 例如,对于 HomeController 将会是 'defaultRoute' => 'home/index'。 在这一点上,项目结构应该如下所示: micro-app/ ├── composer.json ├── config.php ├── web/ └── index.php └── controllers/ └── base └── BaseController.php └── HomeController.php └── vendor 如果您尚未设置 Web 服务器,则可能需要查看Web服务器配置文件示例。 另一种选择是使用 yii serve 命令,它将使用 PHP 内置 web 服务器。 您可以通过以下方式从 micro-app / 目录运行它: vendor/bin/yii serve --docroot=./web 在浏览器中打开应用程序URL现在应该打印出“欢迎来到 Yii2.0 微服务!”,它已经在 HomeController::actionIndex() 中返回。 Info:在我们的示例中,我们已将默认应用程序名称空间 app 更改为 micro, 以表明您不受此名称的限制(如果您是这样认为), 然后调整 controllers namespace 并设置正确的别名。
精华贴
10天前 喜欢(1) 浏览(200) 评论(0)
置顶 讨论
说明 课程简介 素材及源码 纠错 课程简介 本课程主要内容是如何通过使用 yii2 框架,快速打造个人博客系统。 博客系统主要分两个部分: 后台管理系统(backend) 相关内容:yii2-admin扩展, RBAC基于角色的权限管理,基于Markdown的富文本编辑器,上传图片到七牛,Gii生成代码,自定义Gii模版 前台内容展示(frontend) 相关内容:挂件的应用(widget),博客首页界面排版,文章内容呈现,博客时间史 素材及源码 为了方便大家学习 yii2 相关的知识并应用于实践中,本次教程会提供学习素材。什么是素材呢?就是基于 hyii2 后台管理系统添加了学习教程时所需要的样式文件,js文件,界面排版等。让大家免去在学习后端框架的过程中,去写前端的页面和js。更加精确的掌握 yii2 框架的实战应用。通过教程的学习可以开发出相应的一套完整的博客系统。详情查看 应用搭建 一章 为了保证课程的高品质,我们需要对素材进行收费,购买链接。购买后请联系 391430388 获取素材和源码 前台界面预览:http://hyblog.itdocs.org 管理后台地址:http://hyblog-admin.itdocs.org 账户:test 密码:123456 作者会提供完整的开发素材和成品源码,希望不要借此传播,尊重原创,万分感谢。 纠错 教程中遇到的问题,可以先自己尝试解决,培养自身的问题排查能力是一个程序员基本的要求。 本地环境问题(本站统一环境教程:PHP开发环境部署)。 代码书写错误,引起的报错。 搜索本教程错误集锦(开发中) 若仍不能解决,请联系 QQ:391430388 当然人无完人,也许教程中也存在一些不可避免的疏漏,如果发现可以联系作者,先行致谢!
精华贴
12天前 喜欢(1) 浏览(296) 评论(1)
博客
场景 现在有客户表、订单表、图书表、作者表 客户表:Customer【id customer_name】 订单表:Order【id order_name customer_id book_id】 图书表:Book【id book_name author_id】 作者表:Author【id author_name】 怎么来确定是一对多还是一对一呢?这个很简单,比如下面的Customer,一个Customer有多个Order,所以就是一对多;又比如Book,一个Book只有一个作者(这里的情况就是一本书只有一个作者),所以就是一对一。 模型定义 下面是这4个个模型的定义,只写出其中的关联 Customer class Customer extends \yii\db\ActiveRecord { //这是获取客户的订单,由上面我们知道这个是一对多的关联,一个客户有多个订单 public function getOrders() { //第一个参数为要关联的子表模型类名, //第二个参数指定 通过子表的customer_id,关联主表的id字段 return $this->hasMany(Order::className(), ['customer_id' =>'id']); } } Order class Order extends \yii\db\ActiveRecord { //获取订单所属用户 public function getCustomer() { //同样第一个参数指定关联的子表模型类名 return $this->hasOne(Customer::className(), ['id' =>'customer_id']); } //获取订单中所有图书 public function getBooks() { //同样第一个参数指定关联的子表模型类名 return $this->hasMany(Book::className(), ['id' =>'book_id']); } } Book class Book extends \yii\db\ActiveRecord { //获取图书的作者 public function getAuthor() { //同样第一个参数指定关联的子表模型类名 return $this->hasOne(Author::className(), ['id' => 'author_id']); } } Author class Autor extends \yii\db\ActiveRecord { } hasMany、hasOne使用 Yii2中的表之间的关联有2种,它们用来指定两个模型之间的关联。 一对多:hasMany 一对一:hasOne 返回结果:这两个方法的返回结果都为yii\db\ActiveQuery对象 - 第一个参数:所关联的模型的类名称。 - 第二个参数:是一个数组,其中键为所关联的模型中的属性,值为当前模型中的属性。 关联的使用 现在我们获取一个客户的所有的订单信息 // 获取一个客户信息 $customer = Customer::findOne(1); //通过在Customer中定义的关联方法(getOrders())来获取这个客户的所有的订单。 $orders = $customer->orders; 上面的两行代码会生成如下sql语句 SELECT * FROM customer WHERE id=1; SELECT * FROM order WHERE customer_id=1; 关联结果缓存 如果客户的订单改变了,我们再重新调用 $orders = $customer->orders; 再次得到订单的时候你会发现没有变化。原因是只会在第一次执行$customer->orders的时候才会去数据库里面查询,然后会把结果缓存起来,以后查询的时候都不会再执行sql。 那么如果我想再次执行sql如何做呢?可以执行 unset($customer->orders); $customer->orders; 然后就可以从数据库里面取数据了。 定义多个关联 同样,我们还可以在Customer里面定义多个关联。 如返回总数大于100的订单。 class Customer extends \yii\db\ActiveRecord { public function getBigOrders($threshold = 100) { return $this->hasMany(Order::className(), ['customer_id' => 'id']) ->where('subtotal > :threshold', [':threshold' => $threshold]) ->orderBy('id'); } } 关联的两种访问方式 如上面的,如果使用 $customer->bigOrders 将会得到大于100的所有的订单。如果要返回大于200的订单可以这样写 $orders = $customer->getBigOrders(200)->all(); 从上面可以看出访问一个关联的时候有两种方法 如果以函数的方式调用,会返回一个 ActiveQuery 对象($customer->getOrders()->all()) 如果以属性的方式调用,会直接返回模型的结果($customer->orders) with的使用看如下代码,是取一个客户的订单 // 执行sql语句: SELECT * FROM customer WHERE id=1 $customer = Customer::findOne(1); //执行sql:SELECT * FROM order WHERE customer_id=1 $orders1 = $customer->orders; //这个不会执行sql,直接使用上面的缓存结果 $orders2 = $customer->orders; 如果现在我们要取出100个用户,然后访问每个用户的订单,从上面的了解我们可能会写出如下代码 // 执行sql语句: SELECT * FROM customer LIMIT 100 $customers = Customer::find()->limit(100)->all(); foreach ($customers as $customer) { //执行sql: SELECT * FROM order WHERE customer_id=... $orders = $customer->orders; // 处理订单。。。 } 然而,如果真要这样写的话,会在foreach的每个循环里面都执行一次sql去数据库里面查询数据。因为每个$customer对象都是不一样的。 为了解决上面的问题 可以使用 yii\db\ActiveQuery::with()。 其中with的参数为关系的名称,也就在model里面定义的getOrders,getCustomer中的orders和customer // 先执行sql: SELECT * FROM customer LIMIT 100; // SELECT * FROM orders WHERE customer_id IN (1,2,...) $customers = Customer::find()->limit(100)->with('orders')->all(); foreach ($customers as $customer) { // 在这个循环的时候就不会再执行sql了 $orders = $customer->orders; // ...handle $orders... } 如果使用了select来指定返回的列,一定要确保返回的列里面包含所关联的模型的关联字段,否则将不会返回关联的表的Model $orders = Order::find()->select(['id', 'amount'])->with('customer')->all(); // $orders[0]->customer 的结果将会是null // 因为上面的select中没有返回所关联的模型(customer)中的指定的关联字段。 // 如果加上customer_id,$orders[0]->customer就可以返回正确的结果 $orders = Order::find()->select(['id', 'amount', 'customer_id'])->with('customer')->all(); 给with加过滤条件 查询一个客户大于100的订单 //首先执行sql: SELECT * FROM customer WHERE id=1 $customer = Customer::findOne(1); // 再执行查询订单的sql语句:SELECT * FROM order WHERE customer_id=1 AND subtotal>100 $orders = $customer->getOrders()->where('subtotal>100')->all(); 查询100个客户的,每个客户的总合大于100的订单 // 下面的代码会执行sql语句: // SELECT * FROM customer LIMIT 100 // SELECT * FROM order WHERE customer_id IN (1,2,...) AND subtotal>100 $customers = Customer::find()->limit(100)->with([ 'orders' => function($query) { $query->andWhere('subtotal>100'); }, ])->all(); 在这里with的参数为数组,键为关联的名称,值为回调函数。 也就是说 对 orders 这个关联返回的 ActiveQuery,再执行一次$query->andWhere(‘subtotal>100′); 使用joinWith进行表关联 我们都知道可以用join on来写多个表之间的关联。先看看yii2中joinWith的声明 joinWith( $with, $eagerLoading = true, $joinType = ‘LEFT JOIN’ ) $with 数据类型为字符串或数组, 如果为字符串,则为模型中定义的关联的名称(可以为子关联)。 如果为数组,键为model中以getXXX格式定义的关联,值为对这个关联的进一步的回调操作。 $eagerLoading 是否加载在 $with 中关联的模型的数据。 $joinType 联接类型,可用值为:LEFT JOIN、INNER JOIN,默认值为LEFT JOIN // 订单表和客户表以Left join的方式关联。 // 查找所有订单,并以客户 ID 和订单 ID 排序 $orders = Order::find()->joinWith('customer')->orderBy('customer.id, order.id')->all(); // 订单表和客户表以Inner join的方式关联 // 查找所有的订单和书 $orders = Order::find()->innerJoinWith('books')->all(); // 使用inner join 连接order中的 books关联和customer关联。 // 并对custmer关联再次进行回调过滤:找出24小时内注册客户包含书籍的订单 $orders = Order::find()->innerJoinWith([ 'books', 'customer' => function ($query) { $query->where('customer.created_at > ' . (time() - 24 * 3600)); } ])->all(); // 使用left join连接 books关联,books关联再用left join 连接 author关联 $orders = Order::find()->joinWith('books.author')->all(); 在实现上,Yii 先执行满足JOIN查询条件的SQL语句,把结果填充到主模型中, 然后再为每个关联执行一条查询语句, 并填充相应的关联模型。 // Order和books关联 inner join ,但不获取books关联对应的数据 $orders = Order::find()->innerJoinWith('books', false)->all(); On条件 在定义关联的时候还可以指定on条件 class User extends ActiveRecord { public function getBooks() { return $this->hasMany(Item::className(), ['owner_id' => 'id'])->onCondition(['category_id' => 1]); } } 在 joinWith 中使用 //先查询主模型(User)的数据 SELECT user.* FROM user LEFT JOIN item ON item.owner_id=user.id AND category_id=1 // 然后再根据关联条件查询相关模型数据 SELECT * FROM item WHERE owner_id IN (...) AND category_id=1 // 这两个在查询的过程中都使用了 on条件。 $users = User::find()->joinWith('books')->all(); 如果没有使用join操作,即使用的是with 或者 直接以属性来访问关联。这个时候on条件会作为where 条件。 SELECT * FROM user WHERE id=10 $user = User::findOne(10);
24天前 喜欢(1) 浏览(160) 评论(0)
博客
<?php /** * Created by PhpStorm. * User: 16487 * Date: 2018/3/29 * Time: 11:24 */ namespace app\models; use yii\db\ActiveRecord; /** * Class AccountAR * @package app\models * @property integer $id;//ID * @property string $account_classification;//科目类别 * @property string $account_name;//科目名称 * @property string $remarks;//科目备注 */ class AccountAR extends ActiveRecord { const TABLE_NAME = 'tb_account'; const COLUMN_NAME_id = 'id'; //id const COLUMN_NAME_account_classification = 'account_classification'; //科目编号 const COLUMN_NAME_account_name = 'account_name'; //科目名称 const COLUMN_NAME_remarks = 'remarks';//科目备注 public static function tableName() { return static::TABLE_NAME; } public function attributeLabels() { return [ 'id' => 'ID', 'account_classification' => '科目编号', 'account_name' => '科目名称', 'remarks' => '备注', ]; } //添加一些常用的查询方法 /** * @param $id integer * @return static */ public static function findOneByID($id) { return static::findOne([self::COLUMN_NAME_id => $id]); } /** * @return \yii\db\ActiveQuery */ public static function getAllActiveQuery() { return static::find(); } } 代码这样写有很多好处,便于阅读以及良好的代码提示等,比如在PHPstorm下通能很友好的智能提示AccountAR::findOneByID(1)->account_name, 以及 $accountAR=new AccountAR(); accountAR->account_name='**'; 也是能友好的输入->对属性进行了提示其中@property 注释标记了该class有哪些对象和变量,以及这个变量的类型,方法中 @param 表示参数类型,这样做对调用者也是一个良好的提示,@return是返回值的类型,调用者也能通过IDE立马知道返回了什么对象。而const COLUMN_NAME_*之类的变量的作用是对应了数据表中的列名,同时也为find数据时提供了友好的便利。比如$sccountARs=AccountAR::find()->where([AccountAR::COLUMN_NAME_account_name=>$text])->all();,这样及避免了频繁的输入字符导致的错误问题也能良好的提示使用者你的字符表中有哪些字段
28天前 喜欢(1) 浏览(150) 评论(0)
博客
有以下需求输入页数按回车跳转至某一页,效果如下 思路如下,获取到当前的url,并且获取到总页数 View: <?php $view = ''; $pageCount=$dataProvider->pagination->pageCount; if ( $pageCount> 1) { $url = $dataProvider->pagination->createUrl(0); $view = '<input id="input-gopage" style="display: inline;float: left;margin-top: 5px;width: 100px" type="number" class="btn btn-default" placeholder="共'.$pageCount.'页" data-url="' . $url . '">'; } ?> <?= GridView::widget([ 'dataProvider' => $dataProvider, 'id' => 'stock-control', 'columns' => $columns, 'formatter' => ['class' => 'yii\i18n\Formatter', 'nullDisplay' => ''], 'emptyText' => '当前没有找到库存记录', 'emptyTextOptions' => ['style' => 'min-height: 360px;color:red;font-weight:bold'], 'layout' => "<div style=\"margin:auto 2px;\" class=\"row\" id=\"stock-control-grid-view\">{items}</div>\n<div>{pager}" . $view . "</div>{summary}", 'showOnEmpty' => true, 'options' => ['class' => 'storehouse-stock-grid-view '], 'rowOptions' => ['class' => 'item-focus-base tr-item'] ]) ?> Controller: $dataProvider = new ActiveDataProvider([ 'query' => $model->getQuery()->orderBy([ 'id' => SORT_ASC,]), 'pagination' => [ 'pageSize' => 20, 'totalCount'=>$model->getQuery()->count(), ], 'key' => StockControlBean::COLUMN_NAME_sku, 'sort' => ['defaultOrder' => [StorehouseStockAR::COLUMN_NAME_id => SORT_DESC]], ]); JS: $('input#input-gopage').keyup(function (event) { if(event.keyCode==13){ var url=$(this).attr('data-url'); var url2=url.replace('page=1','page='+$(this).val()); window.location.replace(url2); } }); Controller 对 ActiveDataProvider 设置 totalCount 总数目属性,View 中 $pageCount=$dataProvider->pagination->pageCount;获取到总页数, $url = $dataProvider->pagination->createUrl(0);,构建page1 的url并且携带在input的data-url中,部分则需要监测输入回车,并且获取到url将page=1属性替换成'page='+输入的值即可, window.location.replace(url2) 实现页面跳转
29天前 喜欢(1) 浏览(152) 评论(0)
Wiki
先贴一下 Gridview 列表尾部按钮的代码: [ 'header'=>'操作', 'headerOptions' => ['width' => '8%'], 'class' => 'yii\grid\ActionColumn', 'template' => '{update} {delete}', 'buttons'=>[ 'update' => function ($url, $model, $key){ return Html::a('编辑', '#', ['class' => 'j-edit', 'data-toggle' => 'modal', 'data-target' => '#page-modal']); }, 'delete'=> function ($url, $model, $key){ return Html::a('删除', ['delete', 'id'=>$model->id],[ 'data-method'=>'post', 'data-confirm' => '确定删除该项?', ] ) ; } ], ], 添加一个自定义的按钮 [ 'header'=>'操作', 'headerOptions' => ['width' => '8%'], 'class' => 'yii\grid\ActionColumn', 'template' => '{update} {delete} {test}', //此处添加一个 {test} 'buttons'=>[ 'update' => function ($url, $model, $key){ return Html::a('编辑', '#', ['class' => 'j-edit', 'data-toggle' => 'modal', 'data-target' => '#page-modal']); }, 'delete'=> function ($url, $model, $key){ return Html::a('删除', ['delete', 'id'=>$model->id],[ 'data-method'=>'post', 'data-confirm' => '确定删除该项?', ] ) ; } //这里对应添加一个test 'test'=> function ($url, $model, $key){ return Html::a('测试', ['delete', 'id'=>$model->id]) ; } ], ],
1月前 喜欢(1) 浏览(158) 评论(0)
Wiki
在页面中添加 Modal 组件 <?php use yii\bootstrap\Modal; Modal::begin([ 'id' => 'page-modal', 'header' => '<h5>这里是标题</h5>', 'toggleButton' => ['label' => 'click me'], ]); echo '这里是模态内容...'; Modal::end(); ?> 添加出发按钮 <?= Html::a('点击按钮', '#', [ 'class' => 'btn btn-success', 'data-toggle' => 'modal', 'data-target' => '#page-modal' //此处对应Modal组件中设置的id ]) ?>
1月前 喜欢(1) 浏览(199) 评论(3)
Wiki
一般数据表结构中经常会定义 created_at 和 updated_at 两个时间字段来记录表的创建及更新时间,那么 Yii2.0 如何通过模型来自动更新时间呢? 只需要在模型中添加以下方法即可: use yii\behaviors\TimestampBehavior; use yii\db\ActiveRecord; public function behaviors() { return [ [ 'class' => TimestampBehavior::className(), 'attributes' => [ # 创建时更新 ActiveRecord::EVENT_BEFORE_INSERT => ['created_at', 'updated_at'], # 修改时更新 ActiveRecord::EVENT_BEFORE_UPDATE => ['updated_at'] ], #设置默认值 'value' => date("Y-m-d H:i:s") ] ]; }
2月前 喜欢(1) 浏览(202) 评论(0)
社区公告
[公告] Yii中文网为优化用户体验进行大版本升级,老版网站会维持一段时间,可以点击顶部"旧版"链接访问旧版网站。
沟通交流

:492175201(技术1群)

:183620600(技术2群)

:291010569(技术3群)