PHP OOP ::在类之间传递会话密钥

时间:2011-01-04 09:19:38

标签: php oop class sessionid

我正在尝试找出最合适的设计来在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实例,然后将该对象传递给需要它的每个类(几乎所有的),作为类构造函数的参数或作为方法参数。

orderSummary.php

$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的新手,所以非常感谢一些指导。

1 个答案:

答案 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()