喵了个咪

喵了个咪的技术博客

期待已久的 PHP7.4 终于发布了,个人最期待的功能还是 Opcache Preloading

  • 为了预加载文件,您需要编写自定义PHP脚本
  • 此脚本在服务器启动时执行一次
  • 所有预加载的文件都可在内存中用于所有请求
  • 在重新启动服务器之前,对源文件所做的更改不会产生任何影响
阅读全文 »

烦恼从何而来

用 Laravel 的小伙伴应该都会用到 Queue, 从手册 Job 失败后会将队列信息记录到 failed_jobs 表,可以通过

1
2
php artisan queue:failed-table
php artisan migrate

作用主要是为了方便分析失败原因和 Job 重试(本身支持 retry),这都很好理解。当 failed_jobs 没有被创建的时候,会报这样一个异常:

1
SQLSTATE[42S02]: Base table or view not found: 1146 Table 'a8591.failed_jobs' doesn't exist

并且记录到 log 文件,如果你的项目中有异常通知,那是相当困扰。

但是有一个场景,也许我并不关心任务执行是否成功,或者说因为某种特定不可控因素允许任务存在执行失败的情况,而我又不希望被这类异常打扰要怎么办呢?

阅读全文 »

安装 illuminate/notifications

1
composer require illuminate/notifications

bootstrap/app.php 注册服务

1
$app->register(Illuminate\Notifications\NotificationServiceProvider::class);

⚠️ 必须在 AppServiceProvider 注册之前,不然自定义 Channel 会无法找到,提升 InvalidArgumentException with message 'Driver [xxxx] not supported.'

AppServiceProvider.php 注册 Channel

1
2
3
$this->app->make(Illuminate\Notifications\ChannelManager::class)->extend('your-channel', function() {
return $this->app->make(App\Channels\YourChannel::class);
});

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
class Foo
{
public function __call($name, $arguments)
{
echo 1;
}

public static function __callStatic($name, $arguments)
{
echo 2;
}

public function bar()
{
Foo::abc();
}
}

(new Foo)->bar();

结果是 1 还是 2?答案是 1

接下来解释一下:

  1. __call 方法关注方法能否被访问到,而不仅仅是关注是否存在
  2. __callStatic 方法关注的是方法能否被静态的访问到,而不是关注方法是否存在,是否是静态方法。
  3. 具体执行 __call__callStatic ,是根据调用的上下文。如

表结构

字段 数据类型
id int(11)
created_at datetime

期望结果

时间段 数量
2019-09-29 12:00:00 ~ 2019-09-29 12:05:00 100
2019-09-29 12:05:00 ~ 2019-09-29 12:10:00 120
2019-09-29 12:10:00 ~ 2019-09-29 12:15:00 131
2019-09-29 12:15:00 ~ 2019-09-29 12:20:00 189
2019-09-29 12:20:00 ~ 2019-09-29 12:25:00 134
2019-09-29 12:25:00 ~ 2019-09-29 12:30:00 155
2019-09-29 12:30:00 ~ 2019-09-29 12:35:00 198
2019-09-29 12:35:00 ~ 2019-09-29 12:40:00 133

SQL语句

1
2
3
4
5
6
-- 300 为 5 分钟
select
concat(from_unixtime(unix_timestamp(created_at) - unix_timestamp(created_at) % 300), ' ~ ', from_unixtime(unix_timestamp(created_at) - unix_timestamp(created_at) % 300 + 300)) as per,
count(1)
from table
group by per;

如果 created_at 类型为 int,还可以减少一次转换

1
2
3
4
5
6
-- 300 为 5 分钟
select
concat(from_unixtime(created_at - created_at % 300), ' ~ ', from_unixtime(created_at - created_at % 300 + 300)) as per,
count(1)
from table
group by per;

思路

时间分片 MySQL没有相关的时间分片函数,还好可以用取模(%)实现

CI/CD (continuous integration and continuous deployment)

最近同事分享了 GitHub Actions ,发现除了我们常听到的自动部署(项目上线)外还可以做很多事情,比如说就可以自动生成 Hexo 博客,我擦!这不就是我想要的吗?以后就不用每次写完文章之后再执行 hexo clean && hexo g -d 了。

回来各种百度,有几篇针对 HEXO 的文章,照着开始写脚本,发现怎么都不能成功,各种错误。折腾了 3 个小时,终于顺利通过了。

阅读全文 »

很不幸,因为VPS忘记续费,博客的资料没有备份,导致博客资料丢失。
经过深思熟虑,觉得将博客迁移至GitHub,丢失的文章和资料我尽量找回。

含义

字符 含义
*         全部。意思是在该时间的任意点都应当执行
? 不指定,任意。仅用于 日(月)和日(周)。0 0 5 * ? 代表每个月的第5天零点,不论星期几。
0 0 ? * 1 代表每周一,不论是当月的哪天。
, 多个值的分隔符,例如1,5,10 - 代表连续值,例如1-20
/ 步长。例如 5/15,代表从5开始,以15为步长。因此,当5/15位于分钟的位置时,表示小时内的第5、20、35和50分钟
L 最后一天。可以是每月最后一天或者每周最后一天。如果用在 天(周)字段,并且前面加数字,则表示最后一个周N。例如5L,表示最后一个周五(5表示周五,L表示最后)。
W 工作日,指周一到周五的任意一天
# 表示第几个的意思,例如 6#3,表示当月第3个星期六(6表示周六,3表示第3个)

例子

1
2
* 0 9 5L 最后一个周五早上9点
* 0 9 6#3 当月第3个星期六
0%