TValue是lua中的通用数据结构,lua中的所有数据都可以使用这个结构表示。下面看一下与TValue定义相关的数据结构:

/*
** {======================================================
** types and prototypes
** =======================================================
*/
union Value {
  		GCObject *gc;    /* collectable objects */
  		void *p;         /* light userdata */
  		int b;           /* booleans */
  		lua_CFunction f; /* light C functions */
  		lua_Integer i;   /* integer numbers */
  		lua_Number n;    /* float numbers */
};

/*
** Union of all Lua values
*/
typedef union Value Value;

/*
** Tagged Values. This is the basic representation of values in Lua,
** an actual value plus a tag with its type.
*/
#define TValuefields	Value value_; int tt_

struct lua_TValue {
  		TValuefields;
};

typedef struct lua_TValue TValue;

/*
** Common type for all collectable objects
*/
typedef struct GCObject GCObject;


/*
** Common Header for all collectable objects (in macro form, to be
** included in other objects)
*/
#define CommonHeader	GCObject *next; lu_byte tt; lu_byte marked


/*
** Common type has only the common header
*/
struct GCObject {
 		CommonHeader;
};

下面把TValue结构中相关的宏展开:

typedef struct lua_TValue {
	union {
		struct GCObject {
			GCObject *next;
			lu_byte tt;
			lu_byte marked;
		} *gc;
		void *p;
		int b;
		lua_CFunction f;
		lua_Integer i;
		lua_Number n;
	} value_;
	int tt_;
} TValue;

这个结构可以分为两部分:一部分tt_用来标识数据类型,而另一部分value_用来存放相关的数据。联合体gc部分用来存放可gc对象的gc相关的信息:next指针将可gc对象连成链表,tt表示数据类型,marked表示gc处理时的颜色值。从源码时可以看出各种可gc的对象的结构都会包含CommonHeader这个头信息来处理gc相关的信息。

lua中stack相关的信息被表示为TValue的指针,如果:

typedef TValue *StkId;  /* index to stack elements */

下面是字符串结构:

/*
** Header for string value; string bytes follow the end of this structure
** (aligned according to 'UTString'; see next).
*/
typedef struct TString {
  		CommonHeader;
  		lu_byte extra;  /* reserved words for short strings; "has hash" for longs */
  		lu_byte shrlen;  /* length for short strings */
  		unsigned int hash;
  		union {
		size_t lnglen;  /* length for long strings */
		struct TString *hnext;  /* linked list for hash table */
  		} u;
} TString;

/*
** Ensures that address after this type is always fully aligned.
*/
typedef union UTString {
  		L_Umaxalign dummy;  /* ensures maximum alignment for strings */
  		TString tsv;
} UTString;

字符串一旦创建,则不可被改写。lua的值对象若这字符串类型,则以引用方式存在。属于被gc管理的对象。当一个字符串没有任何地方引用就可以回收它。 除了用于gc的CommonHeader外,还有extra shrlen hash和u域。extra对于短字符串来说用于记录这个字符串是否为保留字,这个标识用于词法分析器对保留字的快速判断,对于长字符串存hash值。 hash记录字符串的hash值用来加快字符串的相关操作。shrlen用于记录短字符串的长度。u为联合体其中lnglen记录长字符串的长度,hnext为链表指针。 字符串的实际值并没有分配独立的内在来保存,而是直接加在UTString后面。用下面的宏就可以取到实际的C字符串指针:

/*
** Get the actual string (array of bytes) from a 'TString'.
** (Access to 'extra' ensures that value is really a 'TString'.)
*/
#define getaddrstr(ts)	(cast(char *, (ts)) + sizeof(UTString))
#define getstr(ts)  \
  		check_exp(sizeof((ts)->extra), cast(const char*, getaddrstr(ts)))

所有的短字符串被存放在全局状态机(global_State)的strt域中,strt的数据类型如下:

typedef struct stringtable {
  		TString **hash;
  		int nuse;  /* number of elements */
  		int size;
} stringtable;

相同的短字符串在同一个lua state中存在唯一一份,这样不仅能减小内在而且还提高了比较相关的操作的效率。 长字符串则独立存在,从外部压入一个长字符串时,简单的复制一遍字符串。但不立即计算其hash值,推迟到对字符串作匹配时。

Userdata在lua中并没有特别之处,在储存形式上和字符串相同。可以看成是拥有独立的元表,不被内部化处理,也不需要追加\0的字符串。如下:

/*
** Header for userdata; memory area follows the end of this structure
** (aligned according to 'UUdata'; see next).
*/
typedef struct Udata {
  		CommonHeader;
  		lu_byte ttuv_;  /* user value's tag */
  		struct Table *metatable;
  		size_t len;  /* number of bytes */
  		union Value user_;  /* user value */
} Udata;

/*
** Ensures that address after this type is always fully aligned.
*/
typedef union UUdata {
	L_Umaxalign dummy;  /* ensures maximum alignment for 'local' udata */
  		Udata uv;
} UUdata;