我正在尝试找出最合适的设计来在PHP 5.3中的类之间传递会话密钥。
从第三方API检索会话密钥,我的应用程序进行各种API调用,这些调用都需要传入此会话密钥。
我已经创建了用于保存相关API调用的类,例如,类卡车保存方法,这些方法在被调用时将触发对API的请求以从API_GetCart(),API_AddItem()等调用返回数据。
我将会话密钥存储在一个cookie(唯一需要的cookie)中,并且需要使该cookie的值可用于我的所有类。我不能使用数据库或$ _SESSION来保存会话数据。第三方API负责管理篮子内容等会话管理。
当用户第一次到达我的应用程序时,将没有cookie值,因此我需要能够为新cookie分配新的会话密钥并传递该值(因为cookie尚未提供)我们仍在处理与其他类相同的HTTP请求。
我的一个想法是创建一个这样的Session类,并将会话抓取/检查代码放在构造函数中。
class Session {
public $sk;
function __construct() {
//code to check if user has sessionkey (sk) in cookie
//if not, grab new sessionkey from 3rd party API and assign to new cookie
// $sk = 'abcde12345'; //example $sk value
}
}
然后在所有视图页面上,我将实例化一个新的Session实例,然后将该对象传递给需要它的每个类(几乎所有的),作为类构造函数的参数或作为方法参数。
$s = new Session;
//$s currently would only hold one variable, $sk = "abcde12345"
//but in the future may hold more info or perform more work
// what is best approach to making the sessionkey
// available to all classes? arg to constructor or method... or neither :)
$basket = new Basket;
$baskSumm = $basket->getBasketSummary();
$billing = new Billing;
$billSumm = $billing->getBillingSummary();
$delivery = new Delivery;
$delSumm = $delivery->getDeliverySummary();
//code to render as HTML the customer's basket details
//as well as their billing and delivery details
创建一个Session类(实际上只包含一个值)是最好的主意吗?鉴于它可能需要持有更多的价值并进行更多的检查,因此感觉“正确”使它成为一个阶级。在将该值传递给各个类方面,最好将Session对象传递给它们的构造函数,例如。
$se = new Session;
$basket = new Basket($se);
$baskSumm = $basket->getBasketSummary();
我是OOP的新手,所以非常感谢一些指导。
答案 0 :(得分:3)
您可以使用工厂模式。 Basket,Billing和Delivery对象应由第三方服务API包装类创建:
$svc = new The3rdPartyServiceApiWrapper();
$svc->init(); // connect, get session etc.
if ($svc->fail()) die("halp! error here!");
$basket = $svc->createBasket();
$baskSumm = $basket->getBasketSummary();
$billing = $svc->createBilling();
$billSumm = $billing->getBillingSummary();
$delivery = $svc->createDelivery();
$delSumm = $delivery->getDeliverySummary();
使用API连接Basket,Billing和Delivery类的最佳方法是存储对API类的引用,然后它们可以调用它的任何方法,而不仅仅是getSession()。
另一个优点是,如果您有一个已识别的实体,例如一个用户,然后包装类可以授予你,场景中不会有双重对象。
如果主程序创建用户,则应该有相同用户的不同对象,这是错误的:
$user1 = new User("fred12");
$user2 = new User("fred12");
VS如果API包装器创建它们,则包装器类应保留用户的“缓存”,并使用相同的User对象返回相同的请求:
$user1 = $svc->createUser("fred12");
$user2 = $svc->createUser("fred12"); // $user2 will be the same object
(也许这不是一个最好的例子,如果一个程序创建两个相同的用户,则意味着该程序的主要设计有问题。)
更新:对svc类的解释
The3rdPartyServiceApiWrapper应如下所示:
function getSessionId() {
return $this->sessionId; // initialized by constructor
} // getSessionId()
function createBasket() {
$basket = new Basket($this);
return $basket;
} // createBasket()
The Basket:
function Basket($s) { // constructor of Basket class
$this->svc = $s;
//... the rest part of constructor
} // Basket() constructor
function doSomethingUseful() {
// if you wanna use the session:
$sess = $this->svc->getSessionId();
echo("doing useful with session $session");
// you may access other api functions, I don't know what functions they provide
$this->svc->closeSession();
} // doSomethingUseful()