qxx 项目总结
一、账号池功能
1. 右边注释添加格式
private $client; // Jyeoo的接口
2. 将一些配置放到某个配置文件中,然后通过加载配置环境,来获取相关的变量
app()->configure(xxx);
$this->config = config(xxx);
3. app->configure()的加载过程
function app($make = null)
{
if (is_null($make)) {
return Container::getInstance();
} return Container::getInstance()->make($make);
}
public function configure($name)
{
if (isset($this->loadedConfigurations[$name])) {
return;
} $this->loadedConfigurations[$name] = true; $path = $this->getConfigurationPath($name); if ($path) {
$this->make('config')->set($name, require $path);
}
}
4. 功能要写单元测试,postman测数据测得并不准。
5. 写代码的时候要考虑到查询数据库的次数,避免懒查询:就是查询的次数特别的多。
6. 把自己以前写算法的劲头拿出来写代码,考虑代码的效率和可读性。
7. 使用面向对象的思路来为每个类定义它的属性值。
Class JyeooAccount extends Model
{
const IS_USING_YES = 1;
const IS_USING_NO = 2;
}
8. 避免资源抢占,同时尝试三次抢占。
// 2.redis中没值,就从资源池中选择,抢占
$retryTimes = 0;
while ($retryTimes++ < 3) {
$jyeooAccount = JyeooAccount::query()
->inRandomOrder()
->where('is_using', JyeooAccount::IS_USING_NO)
->first();
if (empty($jyeooAccount)) {
break;
}
$result = JyeooAccount::query()
->where('id', $jyeooAccount->id)
->where('is_using', JyeooAccount::IS_USING_NO)
->update([
'is_using' => JyeooAccount::IS_USING_YES,
'deadline' => $timeStamp,
'usage_time' => $jyeooAccount->usage_time+1
]);
if ($result) {
$accountId = $jyeooAccount->teacher_id;
break;
}
}
9. 变量名要统一,别到最后自己看不懂自己的变量名了,这样就很容易出bug,而且测试也不好测出因变量名出错而出现的bug。
10. if......else.....判断 尽量嵌套的层级要少些,多数情况下就是通过if判断某些特殊的情况,对这些特殊的情况进行特殊的处理,然后其余的情况走大逻辑。
11. 账号随机获取
// 如果账号已经存在,那么就重新获取;
if (!is_null($jyeooAccount)) {
$min = JyeooAccount::query()
->max('teacher_id');
$teacherId = random_int($min, $min + 1000);
}
12. 重要的数据库写操作要用到事务
try{
$timeStamp = Carbon::now()->addMinutes($this->config['cacheMinute'])->timestamp;
DB::beginTransaction();
$newJyeooAccount = new JyeooAccount();
$newJyeooAccount->teacher_id = $teacherId;
$newJyeooAccount->is_using = JyeooAccount::IS_USING_YES;
$newJyeooAccount->deadline = $timeStamp;
$newJyeooAccount->usage_time = 1;
$newJyeooAccount->save();
DB::commit();
} catch (Exception $e) {
Log::error($e);
DB::rollback();
}
13. 账号池通知预警,10分钟通知一次,而且要判断使用了多少剩余多少
if ($usedNum >= $warningNum) {
$cacheKey = 'xxxxxxx';
if (!Cache::has($cacheKey)) {
$message = sprintf("菁优网账号池预警,已用帐号:%d,帐号池数量:%d", $usedNum, $totalNum);
$token = $this->config['dingTalkToken'];
$dingTalk = app(DingTalk::class);
$dingTalk->setToken($token)->sendTextMessage($message);
Cache::put($cacheKey, 1, 10); // 10分钟只通知一次
}
}
14. 像发送请求这个东西,可以放到一个类里,然后通过调用这个类中的方法,这样可以做好封装,对用户黑盒。
$dingTalk = app(DingTalk::class);
$dingTalk->setToken($token)->sendTextMessage($message);
15. 给钉钉聊天群发送信息
class DingTalk
{
private $token; private $url = 'https://oapi.dingtalk.com/robot/send'; public function __construct($token = '')
{
if (!empty($token)) {
$this->token = $token;
}
} public function sendTextMessage($message)
{
$data = array('msgtype' => 'text', 'text' => array('content' => $message));
$data_string = json_encode($data);
$url = $this->url . '?access_token=' . $this->token;
return $this->httpRequest($url, $data_string);
} public function setToken($token)
{
$this->token = $token;
return $this;
} /**
* 发送curl请求
*
* @param $remote_server
* @param $post_string
* @return mixed
*/
private function httpRequest($remote_server, $post_string)
{
$env = app()->environment();
if ($env != 'production') {
Log::info('SKIP_SEND_DING_TALK_MESSAGE', [$remote_server, $post_string]);
return false;
} $ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $remote_server);
curl_setopt($ch, CURLOPT_POST, 1);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 5);
curl_setopt($ch, CURLOPT_HTTPHEADER, array ('Content-Type: application/json;charset=utf-8'));
curl_setopt($ch, CURLOPT_POSTFIELDS, $post_string);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); // 线下环境不用开启curl证书验证, 未调通情况可尝试添加该代码
curl_setopt ($ch, CURLOPT_SSL_VERIFYHOST, 0);
curl_setopt ($ch, CURLOPT_SSL_VERIFYPEER, 0);
$data = curl_exec($ch);
curl_close($ch); return $data;
}
}
16. laravel自动创建日志。日志功能很重要,和高手的差距之一就是通过日志排查错误。而且自己在写日志的时候,一定要能知道,谁,在哪,什么时候,事情,事情的结果。就跟记叙文一样
use App\Console\Commands\LogTrait;
use LogTrait;
$this->log($result, $teachingMaterial->toArray()); public function printProgress()
{
$this->info(sprintf(
"上传进度:%.2f%%(%d/%d), 成功:%d 失败:%d",
$this->finishNum / $this->totalNum * 100,
$this->finishNum,
$this->totalNum,
$this->successNum,
$this->failNum
));
}
17. 把任务放到队列中,当任务失败的时候,队列可以尝试重新执行,6666666。
18. while循环往往会比递归的效率更高,所以有时候为了提高系统的效率就使用while。
二、laravel相关知识点
19. firstOrCreat、updateorCreate
20. whereIn一次查询替换where数组,能省好多次查询。
21. 懒加载功能,是指在查询的过程中,一次性把与其相关的数据查询出来。具体如下:
public function childFiles()
{
if ($this->type == 1) {
return $this->hasMany(TeachingMaterialModel::class, 'father_id');
} return $this->hasMany(LectureNoteModel::class, 'father_id');
} public function childCatalogues()
{
return $this->hasMany(CatalogueModel::class, 'father_id');
}
22. 方法
/**
*
* $instance->is_empty
*
* @return bool
*/
public function getIsEmptyAttribute()
{
if (count($this->childFiles) > 0) {
return false;
} return true;
} public function getCatalogueEmptyAttribute()
{
if (count($this->childCatalogues) > 0) {
return false;
} return true;
}
23. 读取一条随机数
User::inRandomOrder()->first();
24. 获取某个时间点的时间戳
echo Carbon::now()->addDays(25); //2016-11-09 14:00:01
echo Carbon::now()->addWeeks(3); //2016-11-05 14:00:01
echo Carbon::now()->addHours(25); //2016-10-16 15:00:01
echo Carbon::now()->subHours(2); //2016-10-15 12:00:01
echo Carbon::now()->addHours(2)->addMinutes(12); //2016-10-15 16:12:01
echo Carbon::now()->modify('+15 days'); //2016-10-30 14:00:01
echo Carbon::now()->modify('-2 days'); //2016-10-13 14:00:01 $time = Carbon::now()->timestamp;
- // 直接使用字符串
- echo Carbon::now('Europe/London'); //2016-10-14 20:21:20
- // 或者
- echo Carbon::now(new DateTimeZone('Europe/London'));
- echo Carbon::now(); // 2016-10-14 15:18:34
- echo Carbon::today(); // 2016-10-14 00:00:00
- echo Carbon::tomorrow('Europe/London'); // 2016-10-14 00:00:00
- echo Carbon::yesterday();
//要想获取字符串类型的日期,可以使用下面的代码:
- echo Carbon::today()->toDateTimeString();
- echo Carbon::yesterday()->toDateTimeString();
- echo Carbon::tomorrow()->toDateTimeString();
//可以使用 parse
方法解析任何顺序和类型的日期(结果为 Carbon 类型的日期时间对象):
- echo Carbon::parse('2016-10-15')->toDateTimeString(); //2016-10-15 00:00:00
- echo Carbon::parse('2016-10-15')->toDateTimeString(); //2016-10-15 00:00:00
- echo Carbon::parse('2016-10-15 00:10:25')->toDateTimeString(); //2016-10-15 00:10:25
- echo Carbon::parse('today')->toDateTimeString(); //2016-10-15 00:00:00
- echo Carbon::parse('yesterday')->toDateTimeString(); //2016-10-14 00:00:00
- echo Carbon::parse('tomorrow')->toDateTimeString(); //2016-10-16 00:00:00
- echo Carbon::parse('2 days ago')->toDateTimeString(); //2016-10-13 20:49:53
- echo Carbon::parse('+3 days')->toDateTimeString(); //2016-10-18 20:49:53
- echo Carbon::parse('+2 weeks')->toDateTimeString(); //2016-10-29 20:49:53
- echo Carbon::parse('+4 months')->toDateTimeString(); //2017-02-15 20:49:53
- echo Carbon::parse('-1 year')->toDateTimeString(); //2015-10-15 20:49:53
- echo Carbon::parse('next wednesday')->toDateTimeString(); //2016-10-19 00:00:00
- echo Carbon::parse('last friday')->toDateTimeString(); //2016-10-14 00:00:00
//构造日期
- $year = '2015';
- $month = '04';
- $day = '12';
- echo Carbon::createFromDate($year, $month, $day); //2015-04-12 20:55:59
- $hour = '02';
- $minute = '15':
- $second = '30';
- echo Carbon::create($year, $month, $day, $hour, $minute, $second); //2015-04-12 02:15:30
- echo Carbon::createFromDate(null, 12, 25); // 年默认为当前年份
// 日期操作
日期操作可以通过add
(增加)或sub
(减去)跟上要增加或减去的单位来完成。例如,你想给一个日期增加指定的天数,你可以使用addDays
方法。此外还提供了一个modify
方法,参数格式为+
或-
跟上值及单位。所以,如果你想给当前日期增加一年,你可以传递+1 year
:
echo Carbon::now()->addDays(25); //2016-11-09 14:00:01
echo Carbon::now()->addWeeks(3); //2016-11-05 14:00:01
echo Carbon::now()->addHours(25); //2016-10-16 15:00:01
echo Carbon::now()->subHours(2); //2016-10-15 12:00:01
echo Carbon::now()->addHours(2)->addMinutes(12); //2016-10-15 16:12:01
echo Carbon::now()->modify('+15 days'); //2016-10-30 14:00:01
echo Carbon::now()->modify('-2 days'); //2016-10-13 14:00:01
//
eq
– 判断两个日期是否相等。- gt – 判断第一个日期是否比第二个日期大。
- it – 判断第一个日期是否比第二个日期小。
- gte – 判断第一个日期是否大于等于第二个日期。
- ite – 判断第一个日期是否小于等于第二个日期。
- echo Carbon::now()->tzName; // America/Toronto
- $first = Carbon::create(2012, 9, 5, 23, 26, 11);
- $second = Carbon::create(2012, 9, 5, 20, 26, 11, 'America/Vancouver');
- echo $first->toDateTimeString(); // 2012-09-05 23:26:11
- echo $first->tzName; // America/Toronto
- echo $second->toDateTimeString(); // 2012-09-05 20:26:11
- echo $second->tzName; // America/Vancouver
- var_dump($first->eq($second)); // bool(true)
- var_dump($first->ne($second)); // bool(false)
- var_dump($first->gt($second)); // bool(false)
- var_dump($first->gte($second)); // bool(true)
- var_dump($first->lt($second)); // bool(false)
- var_dump($first->lte($second)); // bool(true)
qxx 项目总结的更多相关文章
- qxx项目大文件上传
1. 在做大文件上传的时候,要注意修改文件的配置,php.ini的配置,还有连接时间.这些东西都记不清了,明天需要问一下芳哥,然后遇到问题的时候就能自己解决了. 2. 然后就遇到一个很尴尬的问题:大文 ...
- Fis3前端工程化之项目实战
Fis3项目 项目目录结构: E:. │ .gitignore │ fis-conf.js │ index.html │ package.json │ README.md │ ├─material │ ...
- 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新
本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...
- 最近帮客户实施的基于SQL Server AlwaysOn跨机房切换项目
最近帮客户实施的基于SQL Server AlwaysOn跨机房切换项目 最近一个来自重庆的客户找到走起君,客户的业务是做移动互联网支付,是微信支付收单渠道合作伙伴,数据库里存储的是支付流水和交易流水 ...
- Hangfire项目实践分享
Hangfire项目实践分享 目录 Hangfire项目实践分享 目录 什么是Hangfire Hangfire基础 基于队列的任务处理(Fire-and-forget jobs) 延迟任务执行(De ...
- Travis CI用来持续集成你的项目
这里持续集成基于GitHub搭建的博客为项目 工具: zqz@ubuntu:~$ node --version v4.2.6 zqz@ubuntu:~$ git --version git versi ...
- 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新
[原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...
- 【原】Android热更新开源项目Tinker源码解析系列之二:资源文件热更新
上一篇文章介绍了Dex文件的热更新流程,本文将会分析Tinker中对资源文件的热更新流程. 同Dex,资源文件的热更新同样包括三个部分:资源补丁生成,资源补丁合成及资源补丁加载. 本系列将从以下三个方 ...
- Angular企业级开发(5)-项目框架搭建
1.AngularJS Seed项目目录结构 AngularJS官方网站提供了一个angular-phonecat项目,另外一个就是Angular-Seed项目.所以大多数团队会基于Angular-S ...
随机推荐
- 文件系统、服务、防火墙、SELINUX——安全四大金刚
一提到安全,大家都会想到防火墙,和文件系统权限.而实际工作环境中,我们在Linux的安全配置,会涉及到四个级别.我们思考一个场景,你要在百度盘中存放一个文件,这个动作需要考虑下面四个权限. 1 fir ...
- django session 使用案例
django session 使用案例 HTTP被设计为”无状态”,每次请求都处于相同的空间中. 在一次请求和下一次请求之间没有任何状态保持,我们无法根据请求的任何方面(IP地址,用户代理等)来识别来 ...
- IOS项目中的细节处理,如更改状态栏等等
一,状态栏更改为白色 1 在info.plist中添加一个字段:view controller -base status bar 为NO 2 在需要改变状态栏颜色的ViewController中在Vi ...
- [VS 2015] VS2015 完整ISO镜像包
区别 :https://www.visualstudio.com/zh-cn/products/compare-visual-studio-2015-products-vs 完整ISO镜像:下载 VS ...
- LightOJ 1296 Again Stone Game(sg函数)题解
题意:每次必须拿且只能拿不超过一半的石头,不能拿为败 思路:显然算出每个的sg函数,但是范围1e9显然不能直接打表.所以先打表找规律,发现偶数一直是自己的一半,奇数好像没规律.偶数x的sg函数值是x/ ...
- Unity3D学习笔记(九):摄像机
3D数学复习 using System.Collections; using System.Collections.Generic; using UnityEngine; public class w ...
- iis发布,部署
1.项目发布:选择iis:文件系统:文件路径:realese 2.iis添加: 3.host文件添加 问题1: 不能在此路径中使用此配置节.如果在父级别上锁定了该节,便会出现这种情况.锁定 在全新安装 ...
- java代码实现highchart与数据库数据结合完整案例分析(二)---折线图
作者原创:未经博主允许不许转载 在上一篇的博客中,展示和分析了如何做一个饼状图,有疑问可以参考上一篇博客. 现在分析和展示折线图的绘制和案例分析, 先展示效果图: 与饼状图不同的是,折线图展现更多的数 ...
- Python的替换函数——replace(),strip(),和re.sub()
在Python中常用的三个"替换"函数是strip(),replace()和re.sub(),下面来讲讲这三个函数的用法. 一.replace() 基本用法:对象.replace( ...
- ThreadPool开启多线程时支持最大连接200个(默认为2个),不加则会超时
//ThreadPool System.Net.ServicePointManager.DefaultConnectionLimit = 200;