久草资源网-久草资源网站-久草资源在线-久草资源在线播放-天天艹天天艹-天天艹夜夜

蒙狼科技logo
設為首頁| 聯系我們
咨詢熱線: 13917498722
  您的位置: 首頁 > 網站資訊 > php擴展與嵌入--資源數據類型2

php擴展與嵌入--資源數據類型2

發布日期:2017/7/26
在資源變量中存儲的復雜的數據類型通常在初始化時需要一些內存分配,CPU時間或網絡通信。但是在請求之間保留類似于數據庫連接這種資源,必須要做到持久。資源是否持久是一個必須要考慮到的因素。
首先看內存分配的問題: 在使用php的時候,偏向使用emalloc因為它是malloc的帶回收的版本。但是持久化的資源必須在請求間都存在。對于一個文件句柄類的資源來說,假如要加入一個存儲文件名的需求,那么必須在頭文件中加入如下的代碼:
typedef struct _php_sample_descriptor_data {
    char *filename;
    FILE *fp;
} php_sample_descriptor_data;
行使這個結構可以存儲文件名和文件句柄資源,從而能夠在不同的請求之間進行共享。
對應的,要在源文件中進行響應的更改:
static void php_sample_descriptor_dtor( //這個是進行資源回收的回調函數,定義在資源的初始化處。
zend_rsrc_list_entry *rsrc TSRMLS_DC)
{
    php_sample_descriptor_data *fdata =
      (php_sample_descriptor_data*)rsrc->ptr;
    fclose(fdata->fp);
    efree(fdata->filename);
    efree(fdata);
}
這個靜態函數用來進行資源的回收,需要在初始化資源的時候進行指定回調。
進行修改后的文件打開函數,需要增添給資源分配空間的操作:
PHP_FUNCTION(sample_fopen) //修改后的fopen
{
    php_sample_descriptor_data *fdata;
    FILE *fp;
    char *filename, *mode;
    int filename_len, mode_len;
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ss",
    &filename, &filename_len,
    &mode, &mode_len) == FAILURE) {// 獲取文件名和文件長度 
        RETURN_NULL();
    }
    if (!filename_len  !mode_len) {
        php_error_docref(NULL TSRMLS_CC, E_WARNING,
      "Invalid filename or mode length");
        RETURN_FALSE;
    }
    fp = fopen(filename, mode);
    if (!fp) {
        php_error_docref(NULL TSRMLS_CC, E_WARNING,
      "Unable to open %s using mode %s",
      filename, mode);
        RETURN_FALSE;
    }
    fdata = emalloc(sizeof(php_sample_descriptor_data)); //給包含了文件資源和文件名的結構分配空間
    fdata->fp = fp;
    fdata->filename = estrndup(filename, filename_len);
    ZEND_REGISTER_RESOURCE(return_value, fdata,
  le_sample_descriptor); // 注冊資源
}

對于文件寫入函數fwrite同樣需要修改:
PHP_FUNCTION(sample_fwrite)
{
    php_sample_descriptor_data *fdata;
    zval *file_resource;
    char *data;
    int data_len;
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs",
  &file_resource, &data, &data_len) == FAILURE ) {
        RETURN_NULL();
    }
    ZEND_FETCH_RESOURCE(fdata, php_sample_descriptor_data*,
        &file_resource, -1,
        PHP_SAMPLE_DESCRIPTOR_RES_NAME, le_sample_descriptor);
    RETURN_LONG(fwrite(data, 1, data_len, fdata->fp));
} 

對于sample_fclose函數并不需要改變什么,因為它沒有操作現實的資源。下面這個函數可以從資源中拿到原本的文件名:
PHP_FUNCTION(sample_fname)
{
    php_sample_descriptor_data *fdata;
    zval *file_resource;
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r",
  &file_resource) == FAILURE ) {
        RETURN_NULL();
    }
    ZEND_FETCH_RESOURCE(fdata, php_sample_descriptor_data*,
        &file_resource, -1,
        PHP_SAMPLE_DESCRIPTOR_RES_NAME, le_sample_descriptor);
    RETURN_STRING(fdata->filename, 1);
} 


在完成了內存分配之后,因為必須保持持久化,所以必須延遲析構
對于非持久的資源來說,一旦存放著資源id的變量被unset或fallen out of scope了,那么它們就被從EG(regular_list)中去除掉了。而EG(persistent_list)中使用的索引是鍵值類的,元素在請求的不會不會被主動的去除掉。只有在zend_hash_del()調用或線程/進程完全關閉的情況下才會消弭。 EG(persistent_list)也有dtor方法,但是是zend_register_list_descructors_ex()的第二個參數。一般來說,非持久和持久的資源會被注冊成兩種類型,有的時候也可以合二為一。現在在sample.c中添加一個持久的資源類型。
    static int le_sample_descriptor_persist;
    static void php_sample_descriptor_dtor_persistent(
zend_rsrc_list_entry *rsrc TSRMLS_DC)
{//這是一個持久化的資源析構函數
    php_sample_descriptor_data *fdata =
      (php_sample_descriptor_data*)rsrc->ptr;
    fclose(fdata->fp);
    pefree(fdata->filename, 1);
    pefree(fdata, 1);
}
PHP_MINIT_FUNCTION(sample)
{
    le_sample_descriptor =     zend_register_list_destructors_ex(
  php_sample_descriptor_dtor, NULL,
  PHP_SAMPLE_DESCRIPTOR_RES_NAME, module_number);
    le_sample_descriptor_persist =
    zend_register_list_destructors_ex(
  NULL, php_sample_descriptor_dtor_persistent,
  PHP_SAMPLE_DESCRIPTOR_RES_NAME, module_number);//注冊一個持久化的資源
    return SUCCESS;
} 

下面的這個fopen函數就兼容了持久與非持久的兩個資源類型:
PHP_FUNCTION(sample_fopen)
{
    php_sample_descriptor_data *fdata;
    FILE *fp;
    char *filename, *mode;
    int filename_len, mode_len;
    zend_bool persist = 0;
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC,"ssb",
      &filename, &filename_len, &mode, &mode_len,
      &persist) == FAILURE) {
        RETURN_NULL();
    }
    if (!filename_len  !mode_len) {
        php_error_docref(NULL TSRMLS_CC, E_WARNING,
      "Invalid filename or mode length");
        RETURN_FALSE;
    }
    fp = fopen(filename, mode);
    if (!fp) {
        php_error_docref(NULL TSRMLS_CC, E_WARNING,
      "Unable to open %s using mode %s",
      filename, mode);
        RETURN_FALSE;
    }
    if (!persist) {//非持久化的資源
        fdata = emalloc(sizeof(php_sample_descriptor_data));
        fdata->filename = estrndup(filename, filename_len);//這個做了申請內存和賦值兩步操作 
        fdata->fp = fp;
        ZEND_REGISTER_RESOURCE(return_value, fdata,
  le_sample_descriptor);
    } else {//持久化的資源
        list_entry le;
        char *hash_key;
        int hash_key_len;
        fdata =pemalloc(sizeof(php_sample_descriptor_data),1);
        fdata->filename = pemalloc(filename_len + 1, 1);
        memcpy(fdata->filename, filename, filename_len + 1);
        fdata->fp = fp;
        ZEND_REGISTER_RESOURCE(return_value, fdata,
    le_sample_descriptor_persist);

        /* Store a copy in the persistent_list 在persistent_list存儲一份副本 */
        le.type = le_sample_descriptor_persist;
        le.ptr = fdata;
        hash_key_len = spprintf(&hash_key, 0,
      "sample_descriptor:%s:%s", filename, mode);
        zend_hash_update(&EG(persistent_list),
  hash_key, hash_key_len + 1,
  (void*)&le, sizeof(list_entry), NULL);
        efree(hash_key);
    }
} 

對于非持久化的資源,給定了一個數字的索引,并存放在了跟請求依存的list中。 對于持久化的資源,給定了一個鍵值類型,這個hashkey可以在接下來的請求中被重新得到。然后把資源放進了persistentlist中。當一個持久的資源out of scope的時候,EG(regular_list)的析構函數會為le_sample_descriptro_persist檢查registerlist析構。發現是NULL的話不會有任何的操作。從而也就保證了持久的資源不會被釋放掉。當資源被從EG(persistent_list)中去除的時候,要么是線程進程結束了,要么是有心刪除掉了。這時候就會去找持久化的析構函數。

資源被申請為持久化的原因就是為了在其他的請求中可以復用
假如想要復用持久化的資源,那就一定要用到hash_key,當sample_fopen被調用的時候,函數會行使請求的文件名和模式重新創建hash_key,然后嘗試在persistent_list中找到它。
PHP_FUNCTION(sample_fopen)
{
    php_sample_descriptor_data *fdata;
    FILE *fp;
    char *filename, *mode, *hash_key;
    int filename_len, mode_len, hash_key_len;
    zend_bool persist = 0; //判斷是否持久
    list_entry *existing_file;
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC,"ssb",
      &filename, &filename_len, &mode, &mode_len,
      &persist) == FAILURE) {
        RETURN_NULL();
    }
    if (!filename_len  !mode_len) {
        php_error_docref(NULL TSRMLS_CC, E_WARNING,
      "Invalid filename or mode length");
        RETURN_FALSE;
    }
    /* 通過獲得一個hash_key嘗試尋找一個已經打開的文件 */
    hash_key_len = spprintf(&hash_key, 0,
  "sample_descriptor:%s:%s", filename, mode);

    if (zend_hash_find(&EG(persistent_list), hash_key,
  hash_key_len + 1, (void **)&existing_file) == SUCCESS) {
        /* 成功的找到了這個已經打開的文件句柄資源 */
        ZEND_REGISTER_RESOURCE(return_value,
  existing_file->ptr, le_sample_descriptor_persist);
        efree(hash_key);
        return;
    }
    fp = fopen(filename, mode);
    if (!fp) {
        php_error_docref(NULL TSRMLS_CC, E_WARNING,
      "Unable to open %s using mode %s",
      filename, mode);
        RETURN_FALSE;
    }
    if (!persist) {
        fdata = emalloc(sizeof(php_sample_descriptor_data));
        fdata->filename = estrndup(filename, filename_len);
        fdata->fp = fp;
        ZEND_REGISTER_RESOURCE(return_value, fdata,
  le_sample_descriptor);
    } else {
        list_entry le;
        fdata =pemalloc(sizeof(php_sample_descriptor_data),1);
        fdata->filename = pemalloc(filename_len + 1, 1);
        memcpy(data->filename, filename, filename_len + 1);
        fdata->fp = fp;
        ZEND_REGISTER_RESOURCE(return_value, fdata,
    le_sample_descriptor_persist);
        /* Store a copy in the persistent_list */
        le.type = le_sample_descriptor_persist;
        le.ptr = fdata;
        /* hash_key has already been created by now */
        zend_hash_update(&EG(persistent_list),
  hash_key, hash_key_len + 1,
  (void*)&le, sizeof(list_entry), NULL);
    }
    efree(hash_key);
}

注重因為所有的擴展都使用相同的哈希表單去存儲資源,所以命名很主要。一般都是用擴展和資源類型名作為前綴。

檢查資源可用性:
盡管像文件這種資源可以長期打開,但是類似遠程網絡資源這種假如在請求之間長期不用的話就有問題。所以在使用一個persistent資源之前,要先確定可用性。
if (zend_hash_find(&EG(persistent_list), hash_key,
        hash_key_len + 1, (void**)&socket) == SUCCESS) {
    if (php_sample_socket_is_alive(socket->ptr)) {
        ZEND_REGISTER_RESOURCE(return_value,
socket->ptr, le_sample_socket);
        return;
    }
    zend_hash_del(&EG(persistent_list),
        hash_key, hash_key_len + 1); //這里會去調用之前注冊好的析構函數
}













其他相關文章
  • 網站遷移的那些事
  • 了解網站影響力的不同階段
  • 干貨分享比較火的APP推廣方法
  • 網站建設基礎知識之老板們應該細致哪些?
  • 企業建網站要從這九方面入手
  • 天也網絡做網站技巧之內容原創




  • 企業網站后臺使用
    購物網站后臺使用
    網站產品圖片的處理



    農業銀行支付
    建設銀行支付
    郵政儲蓄銀行支付



    企業網站建設
    整站建設
    購物網站



    企業網站建設建議
    注冊適合自己的域名
    什么是虛擬主機




    售前咨詢QQ: 838821345
    售后服務QQ: 464698733
    應急手機:13917498722


    微信掃一掃
    添加24小時微信客服


    郵箱:lang@MENGL.CN
    地址:上海寶山區城銀路555弄2號樓3樓
    ICP備案:滬ICP備12042844號-3
     滬公網安備:31011402002917號
    做網站 | 企業網站建設 | 上海做網站 | 企業網站制作 | 做網站的公司 | 關于蒙狼 | 整站建設 | 購物網站 | 企業網絡營銷 | 成功案例 | 加盟代理 | 在線訂單
    服務區域: 臨港新區做網站 徐匯做網站 閔行做網站 長寧做網站 虹口做網站 黃浦做網站 盧灣做網站 靜安做網站 浦東做網站 楊浦做網站 普陀做網站 閘北做網站 寶山做網站 嘉定做網站 松江做網站 昆山做網站
    Copyright 2012-2025 上海蒙狼網絡科技有限公司 www.jnjiaoyu.cn All Rights Reserved
    午夜在线亚洲| 国产国语对白一级毛片| 国产伦精品一区二区三区无广告 | 日韩一级黄色大片| 亚洲爆爽| 国产视频一区在线| 亚久久伊人精品青青草原2020| 欧美激情一区二区三区视频高清| 免费毛片基地| 成人免费网站视频ww| 精品国产一区二区三区久 | 欧美国产日韩久久久| 日日夜夜婷婷| 香蕉视频久久| 一级女性全黄久久生活片| 欧美日本免费| 香蕉视频一级| 欧美激情影院| 精品视频一区二区| 欧美一级视频免费观看| 91麻豆爱豆果冻天美星空| 国产91素人搭讪系列天堂| 亚洲天堂在线播放| 欧美激情一区二区三区视频高清| 超级乱淫黄漫画免费| 一级毛片视频播放| 国产a毛片| 国产精品自拍亚洲| 欧美α片无限看在线观看免费| 国产网站免费观看| 亚洲www美色| 韩国妈妈的朋友在线播放| 欧美日本免费| 午夜家庭影院| 久久99中文字幕| 黄色免费三级| 一 级 黄 中国色 片| 91麻豆精品国产高清在线| 精品国产一区二区三区精东影业 | 久久99欧美| 国产成人啪精品| 日韩免费在线视频| 国产成人精品综合| 你懂的国产精品| 午夜欧美成人久久久久久| 国产一区精品| a级精品九九九大片免费看| 欧美另类videosbestsex| 日本特黄特色aaa大片免费| 欧美国产日韩精品| 韩国毛片基地| 国产一区二区精品| 久久精品店| 高清一级做a爱过程不卡视频| a级毛片免费观看网站| 九九精品在线| 天天做日日爱| 日韩欧美一及在线播放| 美女被草网站| 国产精品123| 青青久久精品| 国产91丝袜在线播放0| 国产精品自拍一区| 欧美日本免费| 天堂网中文在线| 美女免费精品视频在线观看| a级毛片免费观看网站| 国产综合成人观看在线| 久久99中文字幕久久| 国产麻豆精品免费视频| 毛片成人永久免费视频| 国产麻豆精品免费视频| 国产成人啪精品| 国产精品1024永久免费视频| 二级片在线观看| 尤物视频网站在线观看| 天天色成人| 免费国产一级特黄aa大片在线| 成人av在线播放| 日韩中文字幕在线播放| 黄视频网站免费看| 超级乱淫伦动漫| 精品国产三级a| 亚飞与亚基在线观看| 天天色成人| 美女被草网站| 国产亚洲精品成人a在线| 九九精品在线| 欧美大片a一级毛片视频| 九九九网站| 国产高清在线精品一区二区| 国产成a人片在线观看视频| 青青久久精品| 久草免费在线观看| 九九久久99综合一区二区| 成人高清免费| 日韩在线观看网站| 成人免费观看的视频黄页| 日韩中文字幕在线亚洲一区 | 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 麻豆网站在线看| 999久久久免费精品国产牛牛| 国产综合成人观看在线| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 韩国三级视频网站| 可以免费在线看黄的网站| 亚洲第一页色| 精品国产一区二区三区精东影业 | 天天色成人| 午夜欧美成人久久久久久| 一级女性全黄久久生活片| 成人高清免费| 美女免费精品高清毛片在线视 | 国产一区精品| 欧美大片aaaa一级毛片| 亚洲精品中文字幕久久久久久| 日本伦理黄色大片在线观看网站| 国产91精品系列在线观看| 国产极品精频在线观看| 黄色短视频网站| 国产91精品一区| 国产一区二区精品尤物| 精品视频一区二区| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 青青久久国产成人免费网站| 精品在线观看国产| 九九干| 日韩免费片| 四虎影视久久| 天天色成人| 免费国产在线观看| 日韩中文字幕在线播放| 成人影视在线播放| 精品国产亚洲人成在线| 日本在线不卡视频| 久草免费在线视频| 国产成人精品综合久久久| 精品视频在线观看视频免费视频| 可以免费看毛片的网站| 韩国三级视频网站| 天堂网中文在线| 天天做日日爱夜夜爽| 美女免费黄网站| 国产国语在线播放视频| 九九干| 超级乱淫伦动漫| 精品视频在线看| 超级乱淫伦动漫| 久久精品成人一区二区三区| 四虎影视库国产精品一区| 国产视频在线免费观看| 国产视频久久久久| 国产91精品系列在线观看| 精品国产一区二区三区久久久狼| 99热精品在线| 999精品在线| 欧美大片aaaa一级毛片| 国产视频一区二区在线播放| 精品国产亚一区二区三区| 欧美激情一区二区三区在线 | 黄视频网站免费| 在线观看成人网 | 国产成人精品综合| 欧美大片毛片aaa免费看| 国产网站免费| 成人影院久久久久久影院| 日本免费乱人伦在线观看| 亚洲第一色在线| 成人影院一区二区三区| 日本在线播放一区| 欧美激情一区二区三区视频 | 韩国毛片基地| 亚洲女初尝黑人巨高清在线观看| 日本伦理黄色大片在线观看网站| 深夜做爰性大片中文| 麻豆网站在线看| 日韩在线观看免费| 天天做日日爱| 日日夜人人澡人人澡人人看免| 韩国三级视频网站| 亚洲精品中文字幕久久久久久| 欧美一级视频免费| 国产一级生活片| 在线观看成人网| 免费国产在线观看| 精品视频在线观看视频免费视频| 国产网站麻豆精品视频| 欧美日本韩国| 日韩免费在线视频| 欧美国产日韩在线| 日韩在线观看免费| 99热热久久| 欧美另类videosbestsex久久| 欧美α片无限看在线观看免费| 一 级 黄 中国色 片| 91麻豆精品国产自产在线| 久久成人亚洲| 欧美激情一区二区三区在线 | 久久久成人网| 久草免费在线观看|