安装yii2时出错 Invalid Argument – yii\base\InvalidArgumentException The file or directory to be published does not exist: D:\Mrli_work\apps\legacy-sale_erp/vendor\bower/jquery/dist

The “https://bower.herokuapp.com/packages/jquery” file could not be downloaded (HTTP/1.1 502 Bad Gateway)
https://bower.herokuapp.com/packages could not be fully loaded, package information was loaded from the local cache and may be out of date

 

解决方法:

你要把composer升级到最新,然后fxp那个插件设置成~1.2(也是最新的吧)

composer self-update

composer global require “fxp/composer-asset-plugin:~1.2”

ActiveRecord 查询的结果和之间执行 sql 不一样。

特意故意写错一个字段,暴露出sql,复制sql到客户端查询,查询出来的条数和ar直接查出来的不一致。
数量不一致。ar查询出来的少一点。

  • 回复于 2014-12-18 16:31 举报

    $model->find().........->all() 就一条
    $model->find().........->count() 显示有两条,实际也是两条
    $model->find().........->createCommand()->queryAll()正常两条全出来。
    你们说这时为什么?
    这个all()感觉就是有问题,我不止遇到一次了。
    每次遇到这种情况,我只能用->createCommand()->queryAll()来替代all()

  • 回复于 2014-12-19 15:37 举报

    为什么find()后面要加all()???
    不能直接findAll()吗?

  • 回复于 2014-12-20 12:33 举报

    find() 返回的是一个 queryBuilder, 可以 select,where, andWhere, group, orderBy,offset, limit 等,findAll() 可没有这些

  • 回复于 2014-12-20 12:34 举报

    findAll 也是间接的调用了 find()..all()

  • 回复于 2014-12-23 22:32 举报

    表示没遇到过这种问题,是不是你姿势不对?

  • 回复于 2014-12-24 17:39 举报

    上面我省略号里写的都是一样的。

  • 回复于 2017-06-07 11:16 举报

    知道是什么原因?我也遇到了

    1 条回复
    回复于 2017-09-12 16:27 回复

    all()方法中的populate()方法会去掉主键相同的行

  • 回复于 2017-09-12 16:27 举报

    all()方法中的populate()方法会去掉主键相同的行

Yii2实现同时搜索多个字段的方法 一个关键词在两个不同的字段以like和or查询

/*获取商家产品列表*/
    public static function getProductList($request,$seller_id)
    {
        $where=[];
        $where['seller_id']=$seller_id;
        /*配送类型*/
        if($request['shipment']&&in_array($request['shipment'],self::$shipments))
        {
            $where['fulfillmentchannel']=$request['shipment'];
        }
        /*关键词搜索*/
        $search_like=[];
        if($request['search'])
        {
            $search=trim($request['search']);
            $search_like = ['or', ['like', 'title', $search], ['like', 'asin', $search], ['like', 'sellersku', $search], ['like', 'tag', $search]];
        }
        /*tag标签*/
        if($request['tag'])
        {
            $where['tag']=$request['tag'];
        }
        /*asin类型*/
        if($request['asin_type']=='parent')
        {
            $where[]=['<>','parentasin',''];
        }
        $prodcuts=SellerProduct::find()
            ->select(['title','asin','sellersku','tag'])
            ->asArray()
            ->where($where)
            ->andWhere($search_like)
            ->limit(10)
//            ->createCommand()->getRawSql();exit;
            ->all();
//        return SellerProduct::getAsinBySellerId($seller_id);
    }

SELECT `title`, `asin`, `sellersku`, `tag` FROM `tbl_seller_product` WHERE (`seller_id`=’A20ANC9S9NO5G6′) AND ((`title` LIKE ‘%222%’) OR (`asin` LIKE ‘%222%’) OR (`sellersku` LIKE ‘%222%’) OR (`tag` LIKE ‘%222%’)) LIMIT 10

YII 开源商城

Fecshop 全称为Fancy ECommerce Shop,是基于php Yii2框架之上开发的一款优秀的开源电商系统,
Fecshop支持多语言,多货币,架构上支持pc,手机web,手机app,和erp对接等入口,您可以免费快速的定制和部署属于您的电商系统。

详细参看地址:Fecshop介绍

Fecshop 官网:http://www.fecshop.com ,您可以在这里提交bug,问题咨询等等。

Fecshop 【已完成】PC Web Demo :http://fecshop.appfront.fancyecommerce.com/

Fecshop 【已完成】Mobile Web Demo(WAP):http://fecshop.apphtml5.fancyecommerce.com

Fecshop 【已完成】Mobile VUE Demo(Appserver,前后端彻底分离模式):http://demo.fancyecommerce.com/#/

Fecshop 后台演示地址:加QQ群,在群公告里面有后台演示地址,账号密码等信息

Fecshop QQ群:186604851 ,入群验证:fecshop

Fecshop 作者QQ:2358269014

FecShop Email:2358269014@qq.com

Fecshop Github地址: https://github.com/fancyecommerce/yii2_fecshop

开源协议:Fecshop 授权协议

yii2 多条件搜索(like > < <> = ) 打印sql

如何打印sql语句

$products = TProductArt::find()
    - >select(['product_id','art_name','domain','price','final_price','discount_type','discount','art_expiry','image_large'])
    ->where($where)
    ->orderBy('score DESC')
    ->andWhere(['>', 'art_expiry', $after_2_minute])
    ->andWhere(['<', 'art_start', $current_date])
    ->andWhere(['<>', 'image_large',''])
    ->andWhere($search_like)
    ->andWhere($search_price)
    ->limit(20)
    ->createCommand()->getRawSql();


public static function searcharray($search, $domain, $group,$price)
{
    $where = array();
    if(in_array($domain, self::$domainsList)){
        $where['domain']=$domain;
    }
    $search_like=array();
    if ($search!='')
    {
        $search = trim($search);
        $search = preg_replace('!\s+!', ' ', $search);  // replaces multiple space with single space
        $search_like = ['or like', 'art_name',explode(" ", $search)];
    }
    if ($group!='')
    {
        if (in_array($group, Yii::$app->params['amazon_product_groups']))
        {
            $where['art_product_group']=$group;
        }
    }
    $search_price=array();
    if(floatval($price)>0)
    {
        $search_price=['<', 'final_price', $price];
    }
    $where['status']=0;
    $after_2_minute = date('Y-m-d H:i:s', time() + 120);
    $current_date = date('Y-m-d H:i:s', time());
    return $products = TProductArt::find()
        ->select(['product_id','art_name','domain','price','final_price','discount_type','discount','art_expiry','image_large'])
        ->where($where)
        ->orderBy('score DESC')
        ->andWhere(['>', 'art_expiry', $after_2_minute])
        ->andWhere(['<', 'art_start', $current_date])
        ->andWhere(['<>', 'image_large',''])
        ->andWhere($search_like)
        ->andWhere($search_price)
        ->limit(20)
     //   ->createCommand()->getRawSql();
        ->asarray()
        ->all()
    ;


}

YII2项目常用技能知识总结

1、不通过日志获取AR执行的原生SQL语句和打印变量数据

$query = User::find() ->select(['username'])->where(['id'=>[1,2,3,4])
// get the AR raw sql in YII2
$commandQuery = clone $query;
echo $commandQuery->createCommand()->getRawSql();$users = $query->all();
打印变量数据可以这样写:
//引用命名空间
use yii\helpers\VarDumper;
//使用
VarDumper::dump($var);
//使用2  第二个参数是数组的深度  第三个参数是是否显示代码高亮(默认不显示)
VarDumper::dump($var, 10 ,true);

2、从数据库二维数组中返回一维数组并配合rules验证规则实现分类数据过滤。

普通返回表记录的二维数组

Member::find()->select('userid')->asArray()->all();
Array
(
    [0] => Array
        (
            [userid] => 1
        )

    [1] => Array
        (
            [userid] => 2
        )

    [2] => Array
        (
            [userid] => 3
        )

)

返回字段的一维数组

Member::find()->select('userid')->asArray()->column();

或者:

\yii\helpers\ArrayHelper::getColumn(Member::find()->all(), 'userid')
Array
(
    [0] => 1
    [1] => 2
    [2] => 3
)

返回一维数组配合验证规则验证数据正确性,如分类catid正确分为只有1-4,但是在devTools打开修改catid为5,提交同样会到数据库,此时rules验证规则如下:

['catid', 'in', 'range' => category::find()->select('id')->asArray()->column()],

当然,这个也可以通过下面这样子写,一样的:

['catid', 'in', 'range' => \yii\helpers\ArrayHelper::getColumn(category::find()->all(), 'catid')],

这样就可以过滤不正确的分类数据了!

3、友好时间表示方法

之前一直使用自定义的友好时间函数。几天前发现万能的YII已经提供了友好时间访问,代码如下:

Yii::$app->formatter->asRelativeTime('1447565922'); //2小时前

4、使用不同的响应类型或者自定义响应类型

有效的格式:

  • FORMAT_RAW
  • FORMAT_HTML
  • FORMAT_JSON
  • FORMAT_JSONP
  • FORMAT_XML

JSON响应

public function actionIndex()
{
    \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
    $items = ['some', 'array', 'of', 'data' => ['associative', 'array']];
    return $items;
}

返回:

{
    "0": "some",
    "1": "array",
    "2": "of",
    "data": ["associative", "array"]
}

自定义响应格式

让我们创建一个定制的响应格式。例子做点有趣和疯狂的事我返回PHP 数组。 首先,我们需要格式化程序本身。创建 components/PhpArrayFormatter.php:

<?php
namespace app\components;
use yii\helpers\VarDumper;
use yii\web\ResponseFormatterInterface;
class PhpArrayFormatter implements ResponseFormatterInterface
{
    public function format($response)
    {
        $response->getHeaders()->set('Content-Type', 'text/php; charset=UTF-8');
        if ($response->data !== null) {
            $response->content = "<?php\nreturn " . VarDumper::export($response->data) . ";\n";
        }
    }
}

组件配置:

return [
    // ...
    'components' => [
        // ...
        'response' => [
            'formatters' => [
                'php' => 'app\components\PhpArrayFormatter',
            ],
        ],
    ],
];

现在是准备使用。在 controllers/SiteController 创建一个新的方法 actionTest:

public function actionTest()
{
    Yii::$app->response->format = 'php';
    return [
        'hello' => 'world!',
    ];
}

返回如下:

<?php
return [
    'hello' => 'world!',
];

5、AR入库前时间通过在模型重写behaviors方法实现优雅入库方式。

如下:

public function behaviors()
{
    return [
        'timestamp' => [
            'class' => TimestampBehavior::className(),
            'attributes' => [
                ActiveRecord::EVENT_BEFORE_INSERT => 'creation_time',
                ActiveRecord::EVENT_BEFORE_UPDATE => 'update_time',
            ],
            'value' => function() { return date('U'); // unix timestamp },
        ],
    ];
}

6、除配置组件记录不同级别日志外,也可以自定义在某个地方记录LOG日志

use yii\log\Logger;
\Yii::getLogger()->log('User has been created', Logger::LEVEL_INFO);

7、 ActiveForm类不让生成label标签

//方法一,通过ActiveForm类
$form->field($model, '字段名')->passwordInput(['maxlength' => true])->label(false) ?>
//方法二,通过 HTML类
Html::activeInput($type,$model,'字段名')

Yii2给必填项加星,样式如下:

div.required label:after {
    content: " *";
    color: red;
}

8、Yii2 获取接口传过来的 JSON 数据:

接收get和post的数据很容易,那么接收json数据呢?!没关系,看这里:

Yii::$app->request->rawBody;

9、座机和手机号码必须填写一个:

public function rules()
{
    return [
        [['telephone', 'mobile'], function ($attribute, $param) {//至少要一个
            if (empty($this->telephone) && empty($this->mobile)) {
                $this->addError($attribute, 'telephone/mobile至少要填一个');
            }
        }, 'skipOnEmpty' => false],
    ];
}

10、where多条件查询示例:

//and复杂示例:
$time = time();
Member::find()->where(['and', ['userid' => 1, 'company' =>'测试公司'], ['>', 'addtime', $time]]);
//SELECT * FROM `member` WHERE ((`userid`=1) AND (`company`='测试公司')) AND (`addtime` > 1447587486)
//and和or组合示例:
$query = Member::find()->where(['and', ['>','userid',2], ['or', ['company' => '深圳市新民家具有限公司'], ['address' => '深圳']]]);
//SELECT * FROM `member` WHERE (`userid` > 2) AND ((`company`='深圳市新民家具有限公司') OR (`address`='深圳'))

11、关于事务:

优雅的写法

Yii::$app->db->transaction(function() {
    $order = new Order($customer);
    $order->save();
});

这相当于下列冗长的代码:

$transaction = Yii::$app->db->beginTransaction();
try {
    $order = new Order($customer);
    $order->save();
    $transaction->commit();
} catch (\Exception $e) {
    $transaction->rollBack();
    throw $e;
}

12、rest风格API获取客户端提交的get和post的数组

// post
Yii::$app->request->bodyParams
// get
Yii::$app->request->queryParams;

13、一个控制器调用其他控制器action的方法:

方法一:

是经典的重写actions方法

  public function actions()
    {
        return [
            'error' => [
                'class' => 'yii\web\ErrorAction',
            ],
            'captcha' => [
                'class' => 'yii\captcha\CaptchaAction',
                'fixedVerifyCode' => YII_ENV_TEST ? 'testme' : null,
            ],
        ];
    }

actions继承yii\base\Actions类,并重写父类的run方法。

方法二:

site控制器如下,访问MemberController控制器下面的index方法。

class SiteController extends Controller
{
    public function actionIndex(){
     Yii::$app->runAction('member/index', ['param'=>'123']);
    }
}

MemberController控制器如下:

class MemberController extends Controller
{
    public function actionIndex($param = '456'){
     echo "second Controller".$param;
    }
}

访问:http://www.yii.dev/site/index.html

输出:second Controller123

14、点击下载,如下载安卓APK文件。

public function actionDownload(){
    return \Yii::$app->response->setDownloadHeaders("http://xxx.com/apk/com.trade.activity.3.0.8.apk");
    //return \Yii::$app->response->sendFile("./com.trade.activity.3.0.8.apk");
}

15、YII模块IP白名单设置,增加安全性

$config['modules']['gii'] = [
     'class' => 'yii\gii\Module',
     'allowedIPs' => ['127.0.0.1', '::1','10.10.1.*'], 
];
$config['modules']['debug'] = [
    'class' => 'yii\debug\Module',
    'allowedIPs' => ['127.0.0.1', '::1', '192.168.0.*', '192.168.33.1'],
];

16、防止 SQL 和 Script 注入

use yii\helpers\Html;
use yii\helpers\HtmlPurifier;
echo Html::encode($view_hello_str) //可以原样显示<script></script>代码  
echo HtmlPurifier::process($view_hello_str)  //可以过滤掉<script></script>代码

17、验证某个ID值是否存在

//之前一直用$model->findOne($id);exists()方法,资源节约,有没有?!

public function validateAttribute($model, $attribute)
{
   $value = $model->$attribute;
   if (!Status::find()->where(['id' => $value])->exists()) {
       $model->addError($attribute, $this->message);
   }
}

18、批量查询

如查询并循环10000条数据。一次性拿1万条内存会有压力,通过批量查询,每次拿1000条,那么内存始终只有1000条的占有量。

foreach(Member::find()->batch(1000) as $value){
    //do something
    //print_r(count($value));
}

19、关于CSRF验证

方法一:关闭Csrf,除非必要,否则不推荐

public function init(){
    $this->enableCsrfValidation = false;
}

方法二:普通提交,form表单中加入隐藏域

<input name="_csrf" type="hidden" id="_csrf" value="<?= Yii::$app->request->csrfToken ?>">

方法三:ajax异步提交,加入_csrf字段

var csrfToken = $('meta[name="csrf-token"]').attr("content");
$.ajax({
  type: 'POST',
  url: url,
  data: {_csrf:csrfToken},
  success: success,
  dataType: dataType
});

20、YII命令行生成数据库文件

自动列出可用的migrate文件

php yii migrate

从vendor/callmez/wechat/migrations目录下生成数据表

php yii migrate --migrationPath=@callmez/wechat/migrations

从当前应用/migrations/db1下初始化数据到db1表

php yii migrate --migrationPath=@app/migrations/db1 --db=db1

21.关联查询

//客户表Model:CustomerModel 
//订单表Model:OrdersModel
//国家表Model:CountrysModel
//首先要建立表与表之间的关系 
//在CustomerModel中添加与订单的关系
      
Class CustomerModel extends \yii\db\ActiveRecord
{
    ...
    
    public function getOrders()
    {
        //客户和订单是一对多的关系所以用hasMany
        //此处OrdersModel在CustomerModel顶部别忘了加对应的命名空间
        //id对应的是OrdersModel的id字段,order_id对应CustomerModel的order_id字段
        return $this->hasMany(OrdersModel::className(), ['id'=>'order_id']);
    }
     
    public function getCountry()
    {
        //客户和国家是一对一的关系所以用hasOne
        return $this->hasOne(CountrysModel::className(), ['id'=>'Country_id']);
    }
    ....
}
      
// 查询客户与他们的订单和国家
CustomerModel::find()->with('orders', 'country')->all();

// 查询客户与他们的订单和订单的发货地址
CustomerModel::find()->with('orders.address')->all();

// 查询客户与他们的国家和状态为1的订单
CustomerModel::find()->with([
    'orders' => function ($query) {
        $query->andWhere('status = 1');
        },
        'country',
])->all();

22、yii2中关闭debug后return $this->redirect($url);不能跳转,服务器报500错误。

问题分析:

1.必须 return 才能让$this->redirect($url);立马跳转, 而不执行后续代码;

2.redirect() 中指定了响应的 http status code,默认是302;

3.当执行$this->redirect($url)时,不管是否在后面加return false 、return true都没有用,还是继续执行完代码。使用header(“Location:$url”);exit;可以解决此问题,但是,这不是yii2的逻辑,并不完美。

解决办法:

【本文由php_sir的博客 http://blog.sina.com.cn/phpsir原创,未经授权禁止转载】

1.在正常情况下,使用 return $this->redirect($url);

2.在解决方案1不生效时,用$this->redirect($url);Yii::$app->response->send();

3.在解决方案2不生效时,用$this->redirect($url);Yii::$app->end();

总结:

用Yii::$app->end();、Yii::$app->response->send();不管在actionXXX还是init方法都能终止代码,而return只能在action终止代码,是因为在init()里仅仅是代码的执行,return只是代码返回。

https://www.cnblogs.com/sandea/p/5714830.html

yii2 安装使用redis 通过composer 安装Predis及使用教程

yii 安装Predis扩展程序

参考地址

https://packagist.org/packages/predis/predis

composer 执行以下命令

composer require predis/predis

php 代码如下

<?php
namespace frontend\controllers;

use Yii;
use yii\web\Controller;
/*redis 需要使用的命名空间*/
use Predis\Autoloader;
use Predis\Client;

class RedisController extends Controller
{

    public function actionIndex()
    {
//        Autoloader::register();
        $client =new Client([
            'scheme' => 'tcp',
            'host'   => '127.0.0.1',
            'port'   => 6379,
        ]);
        $client->set('foo', 'bar5');
        echo $value = $client->get('foo');
    }
    
}

https://github.com/nrk/predis

 


//使用autoload加载相关库,这边重点就是为了require $file;
spl_autoload_register ( function ( $class ) {
$file = __DIR__ . '/lib/Predis/' . $class . '.php' ;
if ( file_exists ( $file ) ) {
require $file ;
return true ;
}
} ) ;

&nbsp;

//配置连接的IP、端口、以及相应的数据库
$server = array (
'host' => '127.0.0.1' ,
'port' => 6379 ,
'database' => 15
) ;
$redis = new Client ( $server ) ;

&nbsp;

//普通set/get操作
$redis -> set ( 'library' , 'predis' ) ;
$retval = $redis -> get ( 'library' ) ;
echo $retval ; //显示 'predis'

//setex set一个存储时效
$redis -> setex ( 'str' , 10 , 'bar' ) ; //表示存储有效期为10秒

//setnx/msetnx相当于add操作,不会覆盖已有值
$redis -> setnx ( 'foo' , 12 ) ; //true
$redis -> setnx ( 'foo' , 34 ) ; //false

//getset操作,set的变种,结果返回替换前的值
$redis -> getset ( 'foo' , 56 ) ; //返回34

// incrby/incr/decrby/decr 对值的递增和递减
$redis -> incr ( 'foo' ) ; //foo为57
$redis -> incrby ( 'foo' , 2 ) ; //foo为59

//exists检测是否存在某值
$redis -> exists ( 'foo' ) ; //true

//del 删除
$redis -> del ( 'foo' ) ; //true

//type 类型检测,字符串返回string,列表返回 list,set表返回set/zset,hash表返回hash
$redis -> type ( 'foo' ) ; //不存在,返回none
$redis -> set ( 'str' , 'test' ) ;
$redis -> type ( 'str' ) ; //字符串,返回string

//append 连接到已存在字符串
$redis -> append ( 'str' , '_123' ) ; //返回累加后的字符串长度8,此进str为 'test_123'

//setrange 部分替换操作
$redis -> setrange ( 'str' , 0 , 'abc' ) ; //返回3,参数2为0时等同于set操作
$redis -> setrange ( 'str' , 2 , 'cd' ) ; //返回4,表示从第2个字符后替换,这时'str'为'abcd'

//substr 部分获取操作
$redis -> substr ( 'str' , 0 , 2 ) ; //表示从第0个起,取到第2个字符,共3个,返回'abc'

//strlen 获取字符串长度
$redis -> strlen ( 'str' ) ; //返回4

//setbit/getbit 位存储和获取
$redis -> setbit ( 'binary' , 31 , 1 ) ; //表示在第31位存入1,这边可能会有大小端问题?不过没关系,getbit 应该不会有问题
$redis -> getbit ( 'binary' , 31 ) ; //返回1

//keys 模糊查找功能,支持*号以及?号(匹配一个字符)
$redis -> set ( 'foo1' , 123 ) ;
$redis -> set ( 'foo2' , 456 ) ;
$redis -> keys ( 'foo*' ) ; //返回foo1和foo2的array
$redis -> keys ( 'f?o?' ) ; //同上

//randomkey 随机返回一个key
$redis -> randomkey ( ) ; //可能是返回 'foo1'或者是'foo2'及其它任何一存在redis的key

//rename/renamenx 对key进行改名,所不同的是renamenx不允许改成已存在的key
$redis -> rename ( 'str' , 'str2' ) ; //把原先命名为'str'的key改成了'str2'

//expire 设置key-value的时效性,ttl 获取剩余有效期,persist 重新设置为永久存储
$redis -> expire ( 'foo' , 1 ) ; //设置有效期为1秒
$redis -> ttl ( 'foo' ) ; //返回有效期值1s
$redis -> expire ( 'foo' ) ; //取消expire行为

//dbsize 返回redis当前数据库的记录总数
$redis -> dbsize ( ) ;

/*
队列操作
*/

//rpush/rpushx 有序列表操作,从队列后插入元素
//lpush/lpushx 和rpush/rpushx的区别是插入到队列的头部,同上,'x'含义是只对已存在的key进行操作
$redis -> rpush ( 'fooList' , 'bar1' ) ; //返回一个列表的长度1
$redis -> lpush ( 'fooList' , 'bar0' ) ; //返回一个列表的长度2
$redis -> rpushx ( 'fooList' , 'bar2' ) ; //返回3,rpushx只对已存在的队列做添加,否则返回0
//llen返回当前列表长度
$redis -> llen ( 'fooList' ) ; //3

//lrange 返回队列中一个区间的元素
$redis -> lrange ( 'fooList' , 0 , 1 ) ; //返回数组包含第0个至第1个共2个元素
$redis -> lrange ( 'fooList' , 0 ,- 1 ) ; //返回第0个至倒数第一个,相当于返回所有元素,注意redis中很多时候会用到负数,下同

//lindex 返回指定顺序位置的list元素
$redis -> lindex ( 'fooList' , 1 ) ; //返回'bar1'

//lset 修改队列中指定位置的value
$redis -> lset ( 'fooList' , 1 , '123' ) ; //修改位置1的元素,返回true

//lrem 删除队列中左起指定数量的字符
$redis -> lrem ( 'fooList' , 1 , '_' ) ; //删除队列中左起(右起使用-1)1个字符'_'(若有)

//lpop/rpop 类似栈结构地弹出(并删除)最左或最右的一个元素
$redis -> lpop ( 'fooList' ) ; //'bar0'
$redis -> rpop ( 'fooList' ) ; //'bar2'

//ltrim 队列修改,保留左边起若干元素,其余删除
$redis -> ltrim ( 'fooList' , 0 , 1 ) ; //保留左边起第0个至第1个元素

//rpoplpush 从一个队列中pop出元素并push到另一个队列
$redis -> rpush ( 'list1' , 'ab0' ) ;
$redis -> rpush ( 'list1' , 'ab1' ) ;
$redis -> rpush ( 'list2' , 'ab2' ) ;
$redis -> rpush ( 'list2' , 'ab3' ) ;
$redis -> rpoplpush ( 'list1' , 'list2' ) ; //结果list1 =>array('ab0'),list2 =>array('ab1','ab2','ab3')
$redis -> rpoplpush ( 'list2' , 'list2' ) ; //也适用于同一个队列,把最后一个元素移到头部list2 =>array('ab3','ab1','ab2')

//linsert 在队列的中间指定元素前或后插入元素
$redis -> linsert ( 'list2' , 'before' , 'ab1' , '123' ) ; //表示在元素'ab1'之前插入'123'
$redis -> linsert ( 'list2' , 'after' , 'ab1' , '456' ) ; //表示在元素'ab1'之后插入'456'

&nbsp;

//blpop/brpop 阻塞并等待一个列队不为空时,再pop出最左或最右的一个元素(这个功能在php以外可以说非常好用)
//brpoplpush 同样是阻塞并等待操作,结果同rpoplpush一样
$redis -> blpop ( 'list3' , 10 ) ; //如果list3为空则一直等待,直到不为空时将第一元素弹出,10秒后超时

&nbsp;

/**
set表操作
*/

//sadd 增加元素,返回true,重复返回false
$redis -> sadd ( 'set1' , 'ab' ) ;
$redis -> sadd ( 'set1' , 'cd' ) ;
$redis -> sadd ( 'set1' , 'ef' ) ;

//srem 移除指定元素
$redis -> srem ( 'set1' , 'cd' ) ; //删除'cd'元素

//spop 弹出首元素
$redis -> spop ( 'set1' ) ;

//smove 移动当前set表的指定元素到另一个set表
$redis -> sadd ( 'set2' , '123' ) ;
$redis -> smove ( 'set1' , 'set2' , 'ab' ) ; //移动'set1'中的'ab'到'set2',返回true or false

//scard 返回当前set表元素个数
$redis -> scard ( 'set2' ) ; //2

//sismember 判断元素是否属于当前表
$redis -> sismember ( 'set2' , '123' ) ; //true or false

//smembers 返回当前表的所有元素
$redis -> smembers ( 'set2' ) ; //array('123','ab');

//sinter/sunion/sdiff 返回两个表中元素的交集/并集/补集
$redis -> sadd ( 'set1' , 'ab' ) ;
$redis -> sinter ( 'set2' , 'set1' ) ; //返回array('ab')

//sinterstore/sunionstore/sdiffstore 将两个表交集/并集/补集元素copy到第三个表中
$redis -> set ( 'foo' , 0 ) ;
$redis -> sinterstore ( 'foo' , 'set1' ) ; //这边等同于将'set1'的内容copy到'foo'中,并将'foo'转为set表
$redis -> sinterstore ( 'foo' , array ( 'set1' , 'set2' ) ) ; //将'set1'和'set2'中相同的元素copy到'foo'表中,覆盖'foo'原有内容

//srandmember 返回表中一个随机元素
$redis -> srandmember ( 'set1' ) ;

&nbsp;

/**
有序set表操作
*/

//sadd 增加元素,并设置序号,返回true,重复返回false
$redis -> zadd ( 'zset1' , 1 , 'ab' ) ;
$redis -> zadd ( 'zset1' , 2 , 'cd' ) ;
$redis -> zadd ( 'zset1' , 3 , 'ef' ) ;

//zincrby 对指定元素索引值的增减,改变元素排列次序
$redis -> zincrby ( 'zset1' , 10 , 'ab' ) ; //返回11

//zrem 移除指定元素
$redis -> zrem ( 'zset1' , 'ef' ) ; //true or false

//zrange 按位置次序返回表中指定区间的元素
$redis -> zrange ( 'zset1' , 0 , 1 ) ; //返回位置0和1之间(两个)的元素
$redis -> zrange ( 'zset1' , 0 ,- 1 ) ; //返回位置0和倒数第一个元素之间的元素(相当于所有元素)

//zrevrange 同上,返回表中指定区间的元素,按次序倒排
$redis -> zrevrange ( 'zset1' , 0 ,- 1 ) ; //元素顺序和zrange相反

//zrangebyscore/zrevrangebyscore 按顺序/降序返回表中指定索引区间的元素
$redis -> zadd ( 'zset1' , 3 , 'ef' ) ;
$redis -> zadd ( 'zset1' , 5 , 'gh' ) ;
$redis -> zrangebyscore ( 'zset1' , 2 , 9 ) ; //返回索引值2-9之间的元素 array('ef','gh')
//参数形式
$redis -> zrangebyscore ( 'zset1' , 2 , 9 , 'withscores' ) ; //返回索引值2-9之间的元素并包含索引值 array(array('ef',3),array('gh',5))
$redis -> zrangebyscore ( 'zset1' , 2 , 9 , array ( 'withscores' => true , 'limit' => array ( 1 , 2 ) ) ) ; //返回索引值2-9之间的元素,'withscores' =>true表示包含索引值; 'limit'=>array(1, 2),表示最多返回2条,结果为array(array('ef',3),array('gh',5))

//zunionstore/zinterstore 将多个表的并集/交集存入另一个表中
$redis -> zunionstore ( 'zset3' , array ( 'zset1' , 'zset2' , 'zset0' ) ) ; //将'zset1','zset2','zset0'的并集存入'zset3'
//其它参数
$redis -> zunionstore ( 'zset3' , array ( 'zset1' , 'zset2' ) , array ( 'weights' => array ( 5 , 0 ) ) ) ; //weights参数表示权重,其中表示并集后值大于5的元素排在前,大于0的排在后
$redis -> zunionstore ( 'zset3' , array ( 'zset1' , 'zset2' ) , array ( 'aggregate' => 'max' ) ) ; //'aggregate' => 'max'或'min'表示并集后相同的元素是取大值或是取小值

//zcount 统计一个索引区间的元素个数
$redis -> zcount ( 'zset1' , 3 , 5 ) ; //2
$redis -> zcount ( 'zset1' , '(3' , 5 ) ) ; //'(3'表示索引值在3-5之间但不含3,同理也可以使用'(5'表示上限为5但不含5

//zcard 统计元素个数
$redis -> zcard ( 'zset1' ) ; //4

//zscore 查询元素的索引
$redis -> zscore ( 'zset1' , 'ef' ) ; //3

//zremrangebyscore 删除一个索引区间的元素
$redis -> zremrangebyscore ( 'zset1' , 0 , 2 ) ; //删除索引在0-2之间的元素('ab','cd'),返回删除元素个数2

//zrank/zrevrank 返回元素所在表顺序/降序的位置(不是索引)
$redis -> zrank ( 'zset1' , 'ef' ) ; //返回0,因为它是第一个元素;zrevrank则返回1(最后一个)

//zremrangebyrank 删除表中指定位置区间的元素
$redis -> zremrangebyrank ( 'zset1' , 0 , 10 ) ; //删除位置为0-10的元素,返回删除的元素个数2

&nbsp;

/**
hash表操作
*/

//hset/hget 存取hash表的数据
$redis -> hset ( 'hash1' , 'key1' , 'v1' ) ; //将key为'key1' value为'v1'的元素存入hash1表
$redis -> hset ( 'hash1' , 'key2' , 'v2' ) ;
$redis -> hget ( 'hash1' , 'key1' ) ; //取出表'hash1'中的key 'key1'的值,返回'v1'

//hexists 返回hash表中的指定key是否存在
$redis -> hexists ( 'hash1' , 'key1' ) ; //true or false

//hdel 删除hash表中指定key的元素
$redis -> hdel ( 'hash1' , 'key2' ) ; //true or false

//hlen 返回hash表元素个数
$redis -> hlen ( 'hash1' ) ; //1

//hsetnx 增加一个元素,但不能重复
$redis -> hsetnx ( 'hash1' , 'key1' , 'v2' ) ; //false
$redis -> hsetnx ( 'hash1' , 'key2' , 'v2' ) ; //true

//hmset/hmget 存取多个元素到hash表
$redis -> hmset ( 'hash1' , array ( 'key3' => 'v3' , 'key4' => 'v4' ) ) ;
$redis -> hmget ( 'hash1' , array ( 'key3' , 'key4' ) ) ; //返回相应的值 array('v3','v4')

//hincrby 对指定key进行累加
$redis -> hincrby ( 'hash1' , 'key5' , 3 ) ; //返回3
$redis -> hincrby ( 'hash1' , 'key5' , 10 ) ; //返回13

//hkeys 返回hash表中的所有key
$redis -> hkeys ( 'hash1' ) ; //返回array('key1','key2','key3','key4','key5')

//hvals 返回hash表中的所有value
$redis -> hvals ( 'hash1' ) ; //返回array('v1','v2','v3','v4',13)

//hgetall 返回整个hash表元素
$redis -> hgetall ( 'hash1' ) ; //返回array('key1'=>'v1','key2'=>'v2','key3'=>'v3','key4'=>'v4','key5'=>13)

&nbsp;

/**
排序操作
*/

//sort 排序
$redis -> rpush ( 'tab' , 3 ) ;
$redis -> rpush ( 'tab' , 2 ) ;
$redis -> rpush ( 'tab' , 17 ) ;
$redis -> sort ( 'tab' ) ; //返回array(2,3,17)
//使用参数,可组合使用 array('sort' => 'desc','limit' => array(1, 2))
$redis -> sort ( 'tab' , array ( 'sort' => 'desc' ) ) ; //降序排列,返回array(17,3,2)
$redis -> sort ( 'tab' , array ( 'limit' => array ( 1 , 2 ) ) ) ; //返回顺序位置中1的元素2个(这里的2是指个数,而不是位置),返回array(3,17)
$redis -> sort ( 'tab' , array ( 'limit' => array ( 'alpha' => true ) ) ) ; //按首字符排序返回array(17,2,3),因为17的首字符是'1'所以排首位置
$redis -> sort ( 'tab' , array ( 'limit' => array ( 'store' => 'ordered' ) ) ) ; //表示永久性排序,返回元素个数
$redis -> sort ( 'tab' , array ( 'limit' => array ( 'get' => 'pre_*' ) ) ) ; //使用了通配符'*'过滤元素,表示只返回以'pre_'开头的元素

&nbsp;

/**
redis管理操作
*/

//select 指定要操作的数据库
$redis -> select ( 'mydb' ) ; //指定为mydb,不存在则创建

//flushdb 清空当前库
$redis -> flushdb ( ) ;

//move 移动当库的元素到其它库
$redis -> set ( 'foo' , 'bar' ) ;
$redis -> move ( 'foo' , 'mydb2' ) ; //若'mydb2'库存在

//info 显示服务当状态信息
$redis -> info ( ) ;

//slaveof 配置从服务器
$redis -> slaveof ( '127.0.0.1' , 80 ) ; //配置127.0.0.1端口80的服务器为从服务器
$redis -> slaveof ( ) ; //清除从服务器

//同步保存服务器数据到磁盘
$redis -> save ( ) ;
//异步保存服务器数据到磁盘
$redis -> bgsave ( ) ;
//??
$redis -> bgrewriteaof ( ) ;
//返回最后更新磁盘的时间
$redis -> lastsave ( ) ;

&nbsp;

//set/get多个key-value
$mkv = array (
'usr:0001' => 'First user' ,
'usr:0002' => 'Second user' ,
'usr:0003' => 'Third user'
) ;
$redis -> mset ( $mkv ) ; //存储多个key对应的value
$retval = $redis -> mget ( array_keys ( $mkv ) ) ; //获取多个key对应的value
print_r ( $retval ) ;

&nbsp;

//批量操作
$replies = $redis -> pipeline ( function ( $pipe ) {
$pipe -> ping ( ) ;
$pipe -> flushdb ( ) ;
$pipe -> incrby ( 'counter' , 10 ) ; //增量操作
$pipe -> incrby ( 'counter' , 30 ) ;
$pipe -> exists ( 'counter' ) ;
$pipe -> get ( 'counter' ) ;
$pipe -> mget ( 'does_not_exist' , 'counter' ) ;
} ) ;
print_r ( $replies ) ;

&nbsp;

&nbsp;

//CAS,事务性操作

function zpop ( $client , $zsetKey ) {
$element = null ;
$options = array (
'cas' => true , // Initialize with support for CAS operations
'watch' => $zsetKey , // Key that needs to be WATCHed to detect changes
'retry' => 3 , // Number of retries on aborted transactions, after
// which the client bails out with an exception.
) ;

$txReply = $client -> multiExec ( $options , function ( $tx )
use ( $zsetKey , & $element ) {
@ list ( $element ) = $tx -> zrange ( $zsetKey , 0 , 0 ) ;
if ( isset ( $element ) ) {
$tx -> multi ( ) ; // With CAS, MULTI *must* be explicitly invoked.
$tx -> zrem ( $zsetKey , $element ) ;
}
} ) ;
return $element ;
}
$zpopped = zpop ( $redis , 'zset' ) ;
echo isset ( $zpopped ) ? "ZPOPed $zpopped" : "Nothing to ZPOP!" , "\n" ;

&nbsp;

//对存取的key加前缀,如: 'nrk:'
$redis -> getProfile ( ) -> setPreprocessor ( new KeyPrefixPreprocessor ( 'nrk:' ) ) ;

&nbsp;

&nbsp;

&nbsp;

//分布式存储的一些方法
$multiple_servers = array (
array (
'host' => '127.0.0.1' ,
'port' => 6379 ,
'database' => 15 ,
'alias' => 'first' ,
) ,
array (
'host' => '127.0.0.1' ,
'port' => 6380 ,
'database' => 15 ,
'alias' => 'second' ,
) ,
) ;

&nbsp;

use Predis\Distribution\IDistributionStrategy ;

class NaiveDistributionStrategy implements IDistributionStrategy {
private $_nodes , $_nodesCount ;

public function __constructor ( ) {
$this ->_nodes = array ( ) ;
$this ->_nodesCount = 0 ;
}

public function add ( $node , $weight = null ) {
$this ->_nodes [ ] = $node ;
$this ->_nodesCount ++;
}

public function remove ( $node ) {
$this ->_nodes = array_filter ( $this ->_nodes , function ( $n ) use ( $node ) {
return $n !== $node ;
} ) ;
$this ->_nodesCount = count ( $this ->_nodes ) ;
}

public function get ( $key ) {
$count = $this ->_nodesCount ;
if ( $count === 0 ) {
throw new RuntimeException ( 'No connections' ) ;
}
return $this ->_nodes [ $count > 1 ? abs ( crc32 ( $key ) % $count ) : 0 ] ;
}

public function generateKey ( $value ) {
return crc32 ( $value ) ;
}
}

//配置键分布策略
$options = array (
'key_distribution' => new NaiveDistributionStrategy ( ) ,
) ;

$redis = new Predis\Client ( $multiple_servers , $options ) ;

for ( $i = 0 ; $i set ( "key:$i" , str_pad ( $i , 4 , '0' , 0 ) ) ;
$redis -> get ( "key:$i" ) ;
}

$server1 = $redis -> getClientFor ( 'first' ) -> info ( ) ;
$server2 = $redis -> getClientFor ( 'second' ) -> info ( ) ;

printf ( "Server '%s' has %d keys while server '%s' has %d keys.\n" ,
'first' , $server1 [ 'db15' ] [ 'keys' ] , 'second' , $server2 [ 'db15' ] [ 'keys' ]
) ;

yii框架post提交遇到400 (Unable to verify your data submission. )

第一种解决办法是关闭Csrf

public function init(){
    $this->enableCsrfValidation = false;
}

第二种解决办法是在form表单中加入隐藏域

<input name="_csrf" type="hidden" id="_csrf" value="<?= Yii::$app->request->csrfToken ?>">

第三种解决办法是在AJAX中加入_csrf字段

<input name="_csrf" type="hidden" id="_csrf" value="<?= Yii::$app->request->csrfToken ?>">
var csrfToken = $('#_csrf').val();
$.ajax({
  type: 'POST',
  url: url,
  data: {_csrf:csrfToken},
  success: success,
  dataType: dataType
});

yii中常用路径

调用YII框架中jquery:Yii::app()->clientScript->registerCoreScript(‘jquery’);

framework/web/js/source的js,其中registerCoreScriptkey调用的文件在framework/web/js/packages.php列表中可以查看

在view中得到当前controller的ID方法:Yii::app()->getController()->id;

在view中得到当前action的ID方法:Yii::app()->getController()->getAction()->id;

yii获取ip地址:Yii::app()->request->userHostAddress;

yii判断提交方式:Yii::app()->request->isPostRequest

得到当前域名: Yii::app()->request->hostInfo

得到proteced目录的物理路径:YII::app()->basePath;

获得上一页的url以返回:Yii::app()->request->urlReferrer;

得到当前url :Yii::app()->request->url;

得到当前home url :Yii::app()->homeUrl

得到当前return url :Yii::app()->user->returnUrl

项目路径:dirname(Yii::app()->BasePath)

一:Yii framework 已经定义的命名空间常量

system: 指向Yii框架目录; Yii\framework

zii: 指向zii library 目录; Yii\framework\zii

application : 指向应用程序基本目录; protected\

webroot: 指向包含里入口脚本 文件的目录; .\

ext : 指向包含所有第三方扩展的目录; \protected\extensions

用法:Yii::getPathOfAlias(‘webroot’)

**二:

{full URL}:取得当前的完整路径**

Yii::getFrameworkPath() :YII framework路径

** 三:

插入meta信息**

 

复制代码

Yii::app()->clientScript->registerMetaTag(‘keywords’,'关键字’); Yii::app()->clientScript->registerMetaTag(‘description’,'一些描述’); Yii::app()->clientScript->registerMetaTag(‘author’,'作者’);

示例:

 

表示为:

Yii::app()->clientScript->registerLinkTag(‘alternate’,'application/rss+xml’,$this->createUrl(‘/feed’));

 

复制代码

在控制器添加CSS文件或JavaScript文件

Yii::app()->clientScript->registerCssFile(Yii::app()->baseUrl.’/css/my.css’); Yii::app()->clientScript->registerScriptFile(Yii::app()->baseUrl.’/css/my.js’);

在view中得到当前controller的ID方法

Yii::app()->getController()->id;

在view中得到当前action的ID方法:

Yii::app()->getController()->getAction()->id;

Yii获取ip地址

Yii::app()->request->userHostAddress;

Yii判断提交方式

Yii::app()->request->isPostRequest

得到当前域名:

Yii::app()->request->hostInfo

得到proteced目录的物理路径

Yii::app()->basePath;

获得上一页的url以返回

Yii::app()->request->urlReferrer;

得到当前url

Yii::app()->request->url;

得到当前home url

Yii::app()->homeUrl

得到当前return url

Yii::app()->user->returnUrl

项目路径

dirname(Yii::app()->BasePath)
调用YII框架中jquery:Yii::app()->clientScript->registerCoreScript('jquery');

framework/web/js/source的js,其中registerCoreScriptkey调用的文件在framework/web/js/packages.php列表中可以查看

在view中得到当前controller的ID方法:Yii::app()->getController()->id;

在view中得到当前action的ID方法:Yii::app()->getController()->getAction()->id;

yii获取ip地址:Yii::app()->request->userHostAddress;

yii判断提交方式:Yii::app()->request->isPostRequest

得到当前域名: Yii::app()->request->hostInfo

得到proteced目录的物理路径:YII::app()->basePath;

获得上一页的url以返回:Yii::app()->request->urlReferrer;

得到当前url :Yii::app()->request->url;

得到当前home url :Yii::app()->homeUrl

得到当前return url :Yii::app()->user->returnUrl

项目路径:dirname(Yii::app()->BasePath)

四:Yii framework 已经定义的命名空间常量
system: 指向Yii框架目录; Yii\framework
zii: 指向zii library 目录; Yii\framework\zii
application : 指向应用程序基本目录; protected\
webroot: 指向包含里入口脚本 文件的目录; .\
ext : 指向包含所有第三方扩展的目录; \protected\extensions

用法:Yii::getPathOfAlias('webroot')
五:
{full URL}:取得当前的完整路径

Yii::getFrameworkPath() :YII framework路径

六:
插入meta信息

复制代码
Yii::app()->clientScript->registerMetaTag('keywords','关键字'); Yii::app()->clientScript->registerMetaTag('description','一些描述'); Yii::app()->clientScript->registerMetaTag('author','作者');

示例:
<link rel="alternate" type="application/rss+xml" href="http://www.dreamdu.com/feed/" />

表示为:
Yii::app()->clientScript->registerLinkTag('alternate','application/rss+xml',$this->createUrl('/feed'));

复制代码

在控制器添加CSS文件或JavaScript文件
Yii::app()->clientScript->registerCssFile(Yii::app()->baseUrl.'/css/my.css'); Yii::app()->clientScript->registerScriptFile(Yii::app()->baseUrl.'/css/my.js');

在view中得到当前controller的ID方法
Yii::app()->getController()->id;

在view中得到当前action的ID方法:
Yii::app()->getController()->getAction()->id;

Yii获取ip地址
Yii::app()->request->userHostAddress;

Yii判断提交方式
Yii::app()->request->isPostRequest

得到当前域名:
Yii::app()->request->hostInfo

得到proteced目录的物理路径
Yii::app()->basePath;

获得上一页的url以返回
Yii::app()->request->urlReferrer;

得到当前url

Yii::app()->request->url;

得到当前home url

Yii::app()->homeUrl

得到当前return url

Yii::app()->user->returnUrl

项目路径

dirname(Yii::app()->BasePath)