Maison développement back-end tutoriel php php的数组跟spl固定数组

php的数组跟spl固定数组

Jun 13, 2016 am 10:58 AM
gt hashtable php

php的数组和spl固定数组
php固定数组隶属于php标准库(spl)的一种数据结构。和php普通数组相比,固定数组只能用整形定义其下标,并且如名字所示,是固定长度,它的优点是比普通数组占用的内存少,而且更快速,具体原因下文会做分析,先做一个简单的测试,将10W个a放入到数组中。

define("MAX", 100000);//simple arrayfunction simple_arr(){        $i = MAX;	$arr = array();	while ($i--)		$arr[$i]= 'a';}// fix arrayfunction fix_arr(){		$i = MAX;	$arr = new SplFixedArray(MAX);	while ($i--)		$arr[$i]= 'a';		}//fix array with setfunction fix_set_arr(){	$i = MAX;	$arr = new SplFixedArray(MAX);	while ($i--)		$arr->offsetSet($i, "a");}
Copier après la connexion

时间消耗

一般数组:0.084696054458618

固定数组:0.048405885696411

固定数组调用offsetSet方法复制:0.27650499343872

内存消耗

一般数组:9324672

固定数组:4800464

固定数组调用offsetSet方法复制:4800344


空间消耗对比

从空间和时间的效率来看,固定数组的消耗都比一般数组少了很可观。固定数组通过扩展中的内置函数offsetSet赋值比通过下标赋值时间慢的多,这个因为用扩展中的内置方法給数组赋值,php内部需要多一次函数表的查询。

在空间方面,对一般数组,php内部是通过hashtable来存储,hashtable中的每一个槽位对应数组中的一个值,在php源码Zend/zend_hash.h中定义了hash相关的结构体定义和函数。

typedef struct bucket {	ulong h;						/* Used for numeric indexing */	uint nKeyLength;	void *pData;	void *pDataPtr;	struct bucket *pListNext;	struct bucket *pListLast;	struct bucket *pNext;	struct bucket *pLast;	const char *arKey;} Bucket;typedef struct _hashtable {	uint nTableSize;	uint nTableMask;	uint nNumOfElements;	ulong nNextFreeElement;	Bucket *pInternalPointer;	/* Used for element traversal */	Bucket *pListHead;	Bucket *pListTail;	Bucket **arBuckets;	dtor_func_t pDestructor;	zend_bool persistent;	unsigned char nApplyCount;	zend_bool bApplyProtection;#if ZEND_DEBUG	int inconsistent;#endif} HashTable	
Copier après la connexion

如上面代码所示,一个10个元素的php数组所站的空间是sizeof(HashTable) + 10 * size(Bucket) + 元素本身占用空间,这是代码层面的算术,其实在php内部会复杂一点,HashTable的nTableSize永远是2^n,所以即使是10个元素,php内部通过 简单算法能实现占用2^4,也即16个槽位,所以实际占用空间是sizeof(HashTable) + 16 * sizeof(Bucket)  + 元素本身占用空间。(空间的计算只考虑下标是整数的情况下)

而对应固定数组直接通过用户传人的size大小初始化数组,如下面代码所示:同样10个元素的数组,所需要的空间只有10* 元素本身占用空间。

static void spl_fixedarray_init(spl_fixedarray *array, long size TSRMLS_DC) /* {{{ */{	if (size > 0) {		array->size = 0; /* reset size in case ecalloc() fails */		array->elements = ecalloc(size, sizeof(zval *));		array->size = size;	} else {		array->elements = NULL;		array->size = 0;	}}
Copier après la connexion

时间方面对比

对于固定数组来说,对内存的申请一步到位了,当内存不够时候会报错,当内存用不完时,也就浪费在那里。

对于普通数组,因为是动态分配数组空间,由于预先不知道要有多少元素,php初始化空数组的的时候,默认8个槽位,但槽位不够的时候,会再分配*2的空间,当元素元素的数量大于hashTbale中的nTableSize的时候,会resize和rehash hashTable,在resize和rehash的过程中,时间的消耗相当可观了。

static int zend_hash_do_resize(HashTable *ht){	Bucket **t;#ifdef ZEND_SIGNALS	TSRMLS_FETCH();#endif	IS_CONSISTENT(ht);	if ((ht->nTableSize  0) {	/* Let's double the table size */		t = (Bucket **) perealloc_recoverable(ht->arBuckets, (ht->nTableSize persistent);		if (t) {			HANDLE_BLOCK_INTERRUPTIONS();			ht->arBuckets = t;			ht->nTableSize = (ht->nTableSize nTableMask = ht->nTableSize - 1;			zend_hash_rehash(ht);			HANDLE_UNBLOCK_INTERRUPTIONS();			return SUCCESS;		}		return FAILURE;	}	return SUCCESS;}ZEND_API int zend_hash_rehash(HashTable *ht){	Bucket *p;	uint nIndex;	IS_CONSISTENT(ht);	if (UNEXPECTED(ht->nNumOfElements == 0)) {		return SUCCESS;	}	memset(ht->arBuckets, 0, ht->nTableSize * sizeof(Bucket *));	p = ht->pListHead;	while (p != NULL) {		nIndex = p->h & ht->nTableMask;		CONNECT_TO_BUCKET_DLLIST(p, ht->arBuckets[nIndex]);		ht->arBuckets[nIndex] = p;		p = p->pListNext;	}	return SUCCESS;}
Copier après la connexion


通过对比发现php普通数组的内存和时间消耗和固定数组相比大部分都用在了符号表上。php内部实现的一个重要思想是通过空间来换取时间,用hashTable能够快速定位到数据的元素,它甚至把hashTable设计成双向链表,又因为php数组空间是动态分配的,而内部是用c实现的,c语言对数组空间分配只有固定分配,为了实现让用户感觉php数组是动态分配空间的,只能通过resize和rehash实现,所以和固定数组比,相同数量的元素赋值,时间就变慢了。总之,普通数组和固定数组各有优劣,不能说固定数组时间和空间消耗低就是后,具体的情况需要考虑到具体的业务场景。

Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

AI Hentai Generator

AI Hentai Generator

Générez AI Hentai gratuitement.

Article chaud

R.E.P.O. Crystals d'énergie expliqués et ce qu'ils font (cristal jaune)
3 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Meilleurs paramètres graphiques
3 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Comment réparer l'audio si vous n'entendez personne
3 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
Où trouver la courte de la grue à atomide atomique
1 Il y a quelques semaines By DDD

Outils chauds

Bloc-notes++7.3.1

Bloc-notes++7.3.1

Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise

SublimeText3 version chinoise

Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1

Envoyer Studio 13.0.1

Puissant environnement de développement intégré PHP

Dreamweaver CS6

Dreamweaver CS6

Outils de développement Web visuel

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Configuration du projet CakePHP Configuration du projet CakePHP Sep 10, 2024 pm 05:25 PM

Dans ce chapitre, nous comprendrons les variables d'environnement, la configuration générale, la configuration de la base de données et la configuration de la messagerie dans CakePHP.

Guide d'installation et de mise à niveau de PHP 8.4 pour Ubuntu et Debian Guide d'installation et de mise à niveau de PHP 8.4 pour Ubuntu et Debian Dec 24, 2024 pm 04:42 PM

PHP 8.4 apporte plusieurs nouvelles fonctionnalités, améliorations de sécurité et de performances avec une bonne quantité de dépréciations et de suppressions de fonctionnalités. Ce guide explique comment installer PHP 8.4 ou mettre à niveau vers PHP 8.4 sur Ubuntu, Debian ou leurs dérivés. Bien qu'il soit possible de compiler PHP à partir des sources, son installation à partir d'un référentiel APT comme expliqué ci-dessous est souvent plus rapide et plus sécurisée car ces référentiels fourniront les dernières corrections de bogues et mises à jour de sécurité à l'avenir.

Date et heure de CakePHP Date et heure de CakePHP Sep 10, 2024 pm 05:27 PM

Pour travailler avec la date et l'heure dans cakephp4, nous allons utiliser la classe FrozenTime disponible.

CakePHP travaillant avec la base de données CakePHP travaillant avec la base de données Sep 10, 2024 pm 05:25 PM

Travailler avec la base de données dans CakePHP est très simple. Nous comprendrons les opérations CRUD (Créer, Lire, Mettre à jour, Supprimer) dans ce chapitre.

Téléchargement de fichiers CakePHP Téléchargement de fichiers CakePHP Sep 10, 2024 pm 05:27 PM

Pour travailler sur le téléchargement de fichiers, nous allons utiliser l'assistant de formulaire. Voici un exemple de téléchargement de fichiers.

Routage CakePHP Routage CakePHP Sep 10, 2024 pm 05:25 PM

Dans ce chapitre, nous allons apprendre les sujets suivants liés au routage ?

Discuter de CakePHP Discuter de CakePHP Sep 10, 2024 pm 05:28 PM

CakePHP est un framework open source pour PHP. Il vise à faciliter grandement le développement, le déploiement et la maintenance d'applications. CakePHP est basé sur une architecture de type MVC à la fois puissante et facile à appréhender. Modèles, vues et contrôleurs gu

CakePHP créant des validateurs CakePHP créant des validateurs Sep 10, 2024 pm 05:26 PM

Le validateur peut être créé en ajoutant les deux lignes suivantes dans le contrôleur.

See all articles