手机浏览 RSS 2.0 订阅 膘叔的简单人生 , 腾讯云RDS购买 | 超便宜的Vultr , 注册 | 登陆
浏览模式: 标准 | 列表全部文章

laravel mass update can not fire any events

 一个很简单的laravel代码:

PHP代码
  1. User::where('id',1)->update(['last_login'=>time()]);  
这样一个简单的代码,你会发现,它并没有触发:updating,updated事件,然而,你又会发现,updated_at也更新了。这是怎么回事?
跟进代码看一下,你会发现这个update,是Eloquent\Builder的方法,它的update方法代码很简单:
PHP代码
  1. /** 
  2.  * Update a record in the database. 
  3.  * 
  4.  * @param  array  $values 
  5.  * @return int 
  6.  */  
  7. public function update(array $values)  
  8. {  
  9.     return $this->toBase()->update($this->addUpdatedAtColumn($values));  
  10. }  
我擦 ,要不要这么直接?再看一下toBase,这回是调用的是Query\Builder了。这里的update,就纯粹是生成sql,然后execute,不再走EloquentORM的事件了,所以就不会触发任何事件。
 
那么要怎么样才能触发事件呢?方法有两种
1、不要这样简写:先读数据再update,就一定会触发
2、不要用where(),而是直接User::find(1)->update([]),这时候就会从ORM走了。
 
这事儿官方有解释:
https://laravel.com/docs/5.3/eloquent#updates
  1. When issuing a mass update via Eloquent, the saved and updated model events will not be fired for the updated models. This is because the models are never actually retrieved when issuing a mass update.  
同时,这事儿在stackoverflow上也有很多人在问,有人回复的就是和我上面的一样:
http://stackoverflow.com/questions/41295032/laravel-eloquent-model-update-event-is-not-fired
  1. You need to retrieve the user from the database and then save that user in order to fire the event. For example:  
  2.   
  3. This will NOT fire the update event:  
  4.   
  5. User::where('id', $id)->update(['username' => $newUsername]);  
  6. This will fire the update event:  
  7.   
  8. User::find($id)->update(['username' => $newUsername]);  
官方的解释我是看过了,下面的是我用了很多办法没法解决后才找到的。
我在想,use SoftDeletes可以注入代码,为什么我不行?直到我发现softdeletes是官方自带库,人家都准备了事件:onDelete了。。怎么就没有onUpdate事件可以让我加点字段呢?
然后我又想用其他的Scope,也没有成功(折腾了一个下午)
 
暂时放弃!就这样吧,不过我还在想想,有没有什么办法可以注入,因为在sql生成完后,它还是有触发的事件的,比如:
PHP代码
  1. /** 
  2.   * Log a query in the connection's query log. 
  3.   * 
  4.   * @param  string  $query 
  5.   * @param  array   $bindings 
  6.   * @param  float|null  $time 
  7.   * @return void 
  8.   */  
  9.  public function logQuery($query$bindings$time = null)  
  10.  {  
  11.      if (isset($this->events)) {  
  12.          $this->events->fire(new Events\QueryExecuted(  
  13.              $query$bindings$time$this  
  14.          ));  
  15.      }  
  16.   
  17.      if ($this->loggingQueries) {  
  18.          $this->queryLog[] = compact('query', 'bindings', 'time'); 
  19.      } 
  20.  } 
  21.  
  22.  /** 
  23.   * Register a database query listener with the connection. 
  24.   * 
  25.   * @param  \Closure  $callback 
  26.   * @return void 
  27.   */ 
  28.  public function listen(Closure $callback) 
  29.  { 
  30.      if (isset($this->events)) { 
  31.          $this->events->listen(Events\QueryExecuted::class, $callback); 
  32.      } 
  33.  } 
  34.  
  35.  /** 
  36.   * Fire an event for this connection. 
  37.   * 
  38.   * @param  string  $event 
  39.   * @return void 
  40.   */ 
  41.  protected function fireConnectionEvent($event) 
  42.  { 
  43.      if (! isset($this->events)) { 
  44.          return; 
  45.      } 
  46.  
  47.      switch ($event) { 
  48.          case 'beganTransaction': 
  49.              return $this->events->fire(new Events\TransactionBeginning($this)); 
  50.          case 'committed': 
  51.              return $this->events->fire(new Events\TransactionCommitted($this)); 
  52.          case 'rollingBack':  
  53.              return $this->events->fire(new Events\TransactionRolledBack($this));  
  54.      }  
  55.  }  
实在不行了,还可以在事务上面想想办法,但是总归是只能一次处理,如果是需要两次执行sql。那还不如直接写代码呢。等有空再看了
 
 
 

android studio安装APK时出错:INSTALL_FAILED_NO_MATCHING_ABIS

记录一下,真机编译的时候没问题,用模拟器的时候报:INSTALL_FAILED_NO_MATCHING_ABIS,查了一下原因。就是因为为了提速用了、x86的模拟,所以。。。

还好stackoverflow有人解决了这些问题:

1、http://stackoverflow.com/questions/24572052/install-failed-no-matching-abis-when-install-apk

2、http://stackoverflow.com/questions/24751350/install-failed-no-matching-abis-how-to-overcome

解决起来还是比较简单的,在app/build.grade的android节点下增加:

XML/HTML代码
  1.     abi {  
  2.         enable true  
  3.         reset()  
  4.         include 'x86', 'armeabi-v7a'  
  5.         universalApk true  
  6.     }  
  7. }  

一切就OK了,再编译的时候就:

XML/HTML代码
  1. 01/14 09:36:59: Launching app  
  2. $ adb install-multiple -r /xxxxx/app/build/outputs/apk/app-x86-debug.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_5.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_6.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_9.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_2.apk /xxxxx/app/build/intermediates/split-apk/debug/dep/dependencies.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_8.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_7.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_3.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_4.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_0.apk /xxxxx/app/build/intermediates/split-apk/debug/slices/slice_1.apk   
  3. Split APKs installed  
  4. $ adb shell am start -n "com.xxxxx.app/com.xxxxx.app.activity.StartActivity" -a android.intent.action.MAIN -c android.intent.category.LAUNCHER  
  5. $ adb shell am startservice com.xxxxxx.app/com.android.tools.fd.runtime.InstantRunService  
  6. Connected to process 3689 on device emulator-5554  
居然切成了10个apk.....


如何在旧版的macOS上使用双重认证

 这个标题可不是标题党,主要是我的macbook已经开启了双重认证,即我在任何一台设备上登录我的appleid,都会在手机上弹出:是否允许登录。然后输入系统给我的安全码,输进去就OK了。

然而问题发生在yosemate上,我登录apple store,结果没有弹出任何窗口来让我输入安全码。这怎么破?开始的时候一直无解。我以为把密码改成安全码就行了,然而直接告诉我密码错误。然后我又去官网后台生成一个所谓的专用密码来登录,也是告诉我密码错误。。。崩溃 !

最后google了一下,原来在官网上还真有介绍 :

https://support.apple.com/zh-cn/HT204915
  1. 如果我在运行旧版软件的设备上使用双重认证,会怎样?  
  2. 如果您在运行旧版操作系统的设备上使用双重认证,系统可能会要求您在登录时将六位验证码添加至密码末尾。从运行 iOS 9 及更高版本或 OS X El Capitan 及更高版本的受信任设备中获取验证码,或者将其发送至受信任的电话号码。然后直接在密码栏位中键入密码,后跟六位验证码。  

于是我登录apple store时,先输入密码,点击登录。然后在手机上弹出授权,点击允许,再将手机上的6位安全码,输到刚才 旧版机器 的密码框里(跟在最后),即:原密码+安全码,再点击登录。成功!

好吧。看起来不复杂,搞定就好

sablog for PHP7

 想来这东西应该没多少人用了。不过我还是上传一份吧。

sablog.zip
就是对着BUG改的,应该也没有什么太特别的玩意。有需要的就下载

Tags: sablog

旧贴:fck在chrome下能正常显示

 迁移的时候发现,后台的fck编辑器出不来。以前是用firefox的所以没问题,后来也是在使用chrome的时候才有注意(新版的fckEditor早就没有这个问题了,新版已经叫CKEditor了)

1、修改admin/editor/fckeditor_php5.php ,现在应该没有人用4了,直接改5,

同名方法覆盖一下
  1. function IsCompatible()  
  2.     {  
  3.         global $HTTP_USER_AGENT;  
  4.   
  5.         if(isset($HTTP_USER_AGENT)){  
  6.             $sAgent = $HTTP_USER_AGENT;  
  7.         } else{  
  8.             $sAgent = $_SERVER['HTTP_USER_AGENT'];  
  9.         }  
  10.   
  11.         if(strpos($sAgent'MSIE') !== false && strpos($sAgent'mac') === false && strpos($sAgent'Opera') === false){  
  12.             $iVersion = (float) substr($sAgentstrpos($sAgent'MSIE') + 5, 3);  
  13.             return ($iVersion >= 5.5);  
  14.         } else{  
  15.             if(strpos($sAgent'Gecko/') !== false){  
  16.                 $iVersion = (int) substr($sAgentstrpos($sAgent'Gecko/') + 6, 8);  
  17.                 return ($iVersion >= 20030210);  
  18.             } else{  
  19.                 if(strpos($sAgent'Opera/') !== false){  
  20.                     $fVersion = (float) substr($sAgentstrpos($sAgent'Opera/') + 6, 4);  
  21.                     return ($fVersion >= 9.5);  
  22.                 } else{  
  23.                     if(preg_match("|AppleWebKit/(\d+)|i"$sAgent$matches)){  
  24.                         $iVersion = $matches[1];  
  25.                         return ($matches[1] >= 522);  
  26.                     } else{  
  27.                         return false;  
  28.                     }  
  29.                 }  
  30.             }  
  31.         }  
  32.     }  

2、修改fckeditor.js,将两个配置的False改为true即可:

JavaScript代码
  1. this.EnableSafari   = true ;        // This is a temporary property, while Safari support is under development.  
  2. this.EnableOpera    = true ;  

这两个默认是false,改为true即可。

再打开sablog的后台,就OK了。

--------纯记录,下次再手贱的时候就不需要折腾N久了

 

Tags: fckeditor