Write once, run anywhere

Write once, run anywhere

概述

环境:LNMP(CentOS7.2 + Nginx1.10 + MariaDB10.0 + PHP5.6),Laravel5.1,Yar;

用户Api项目:项目名:example_user_api(example可替换为具体公司的名字,代表公司的user项目,下同),域名:api.user.example.com,对应数据:user;

订单Api项目:项目名:example_order_api,域名:api.order.example.com,对应数据:order;

订单项目:项目名:example_order,域名:order.example.com;

正文

作为一名PHP后端开发工程师,时常感受到来自项目deadline的压力,而且还有需求在开发过程中的变更。

如何应对需求的变更和来自项目deadline的压力,那就要考虑系统的架构了,然后,才是编码——为何这样说,我喜欢用一个比喻解释,大厦的架构搭建好了,垒砖就容易多了。

系统架构从项目的角度来讲有两个方面,一方面是项目与项目之间的搭配,另一方面是项目本身的分层。

本文主要讲项目与项目之间的搭配这一方面的架构,也就是实现Write once, run anywhere的基础。先下结论:项目与项目之间通过RPC接口调用。

如环境描述和项目清单所述,环境为LNMP,用户相关项目有用户Api项目,订单相关项目有订单Api项目和订单项目。

读者或许有疑问了,这里用户相关的只有一个项目,为何订单相关的项目有两个,它们之间有什么区别?其实,这就是Write once, run anywhere的关键。

用户Api项目能够为订单项目提供Rpc服务,订单Api项目更是主要是为订单项目提供Rpc服务,这两者本身都是没有界面的,这也是它们为何被定义为Api项目的原因。而订单项目通过调用用户Api项目和订单Api项目的Rpc服务,为用户提供查看、修改等操作自己订单的服务,因此是有界面的。
然而,这也不足以要把订单划分成订单Api项目和订单项目两个项目。进一步的原因是订单Api项目不仅用于(用户查询、修改订单等操作的)订单项目,也可以为公司内部的ERP系统提供订单相关的Rpc服务。

代码

Talk is easy, show me the code.
用户Api项目,有t_user表,其对应的Model为User,文件名为User.php,内容如下:

<?php
namespace App\Model;

use DB;
use Log;
use Redis;

/**
 * 用户表
 * @author los_gsy
 */
class User {
    /**
     * 查询一条记录,根据id
     * @param int $id
     * @return null | object
     */
    static public function getById($id) {
        $result = DB::connection('user')->table('user')
            ->where('id', $id)
            ->first();
        if (!$result) {
            Log::warning(__METHOD__ . ': $param = ' . var_export(func_get_args(), true));
        }
        return $result;
    }

}

通过Controller封装成Rpc服务,这里取文件名为ModelController.php,内容如下:

<?php
namespace App\Http\Controllers\Rpc;

use App\Http\Controllers\Controller;
use App\Model\User;
use Illuminate\Http\Request;
use Log;
use Yar_Server;

/**
 * 模型
 * @author los_gsy
 */
class ModelController extends Controller {
    /**
     * 构造函数
     */
    public function __construct(Request $req) {
        parent::__construct();
    }


    /**
     * 用户
     */
    public function user(Request $req) {
        $service = new Yar_Server(new User());
        $service->handle();
    }

}

类似的,订单Api项目,有t_order表,其对应的Model为Order,文件名为Order.php,内容如下:

<?php
namespace App\Model;

use DB;
use Log;
use Redis;

/**
 * 订单表
 * @author los_gsy
 */
class Order {
    /**
     * 查询一条记录,根据id
     * @param int $id
     * @return null | object
     */
    static public function getById($id) {
        $result = DB::connection('order')->table('order')
            ->where('id', $id)
            ->first();
        if (!$result) {
            Log::warning(__METHOD__ . ': $param = ' . var_export(func_get_args(), true));
        }
        return $result;
    }

}

通过Controller封装成Rpc服务,这里同样取文件名为ModelController.php,内容如下:

<?php
namespace App\Http\Controllers\Rpc;

use App\Http\Controllers\Controller;
use App\Model\Order;
use Illuminate\Http\Request;
use Log;
use Yar_Server;

/**
 * 模型
 * @author los_gsy
 */
class ModelController extends Controller {
    /**
     * 构造函数
     */
    public function __construct(Request $req) {
        parent::__construct();
    }


    /**
     * 订单
     */
    public function order(Request $req) {
        $service = new Yar_Server(new Order());
        $service->handle();
    }

}

订单项目,在这里用户需要查询自己的一个订单详情,于是通过调用用户Api项目和订单Api项目的Rpc服务来实现,代码如下:

<?php
namespace App\Http\Controllers;

use App\Http\Controllers\Controller;
use Illuminate\Http\Request;
use Yar_Client;

/**
 * 订单
 * @author los_gsy
 */
class OrderController extends Controller {
    /**
     * 构造函数
     */
    public function __construct(Request $req) {
        parent::__construct();
    }


    /**
     * tmp
     * @param Request $req
     */
    public function tmp(Request $req) {
        //定义变量
        $oUser = new Yar_Client('http://api.user.example.com/Rpc/Model/user');
        $oOrder = new Yar_Client('http://api.order.example.com/Rpc/Model/order');

        //查询用户和订单
        $user = $oUser->getById(1);
        $order = $oOrder->getById(1);

        //打印用户和订单信息
        var_export($user);
        var_export($order);
    }

}

结尾

Write once, run anywhere主要的思路就如上所述,以及代码示例了。仅用于学习参考,如果是生产使用,还需考虑安全、性能等因素。
刚开始在简书上写日志,不足之处,还请各位不吝赐教。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,911评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,014评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,129评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,283评论 1 264
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,159评论 4 357
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,161评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,565评论 3 382
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,251评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,531评论 1 292
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,619评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,383评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,255评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,624评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,916评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,199评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,553评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,756评论 2 335

推荐阅读更多精彩内容