Updated 2018-09-28 13:12:58 by PeterLewerin

Created by CecilWesterhof.

Default you cannot check if something is a dict, That is why I thought it a good idea to write a proc for it.

A simple check is to check if something is a list and if so if the list contains an even number of elements.

But I find this not good enough, because this would validate:
{[} \] {[} \]

In my opinion this is not a correct dict. (Opinions differ about this.) It can certainly not be created with dict operations. And when doing:
dict set d \{ }

The list is cleaned up and you get:
{[} \] \{ \}

That is why I check if the length of the list is two times the number of the keys. The code becomes:
proc isDict d {
    expr {[string is list $d]
        && !([llength $d] % 2)
        && ((2 * [llength [dict keys $d]]) == [llength $d])
    }
}

isDictKeysValsUnique

I also wanted a proc to check if all keys and values are unique. Keys are unique of-course, but the values should also be unique and no value should also be a key. For this I created the following proc:
proc isDictKeysValsUnique d {
    expr {[isDict $d]
        && [llength [lsort -unique $d]] == [llength $d]
    }
}

PYK 2018-07-24: There's a reason there has never been an example of such on the wiki. This routine would never be needed in a working script because with dict one creates dictionaries guaranteed not to be lists with duplicate values that would be keys, and when consuming key/value pairs, one simply chooses to treate it either as a list of key-value pairs or as a dictionary.

EMJ 2018-07-24: Nor is there any reason, in general, for values to be unique or for values not to happen to have the same value as a key.

CecilWesterhof 2018-07-24: In general not, no. But in my case I had a good reason for that. See Balanced Brackets.

As always: comments, tips and questions are appreciated.

2018-07-22: see also TIP 501 "string is dict".

EMJ 2018-07-22: which I think would also validate the above four-element list.

EMJ 2018-07-22: Put the original four-element list above in a variable and you can use that variable as a dict. Of course the duplicate key disappears because that is a property of dicts (and if the values were different it is the last which would be taken).

Also the proc above will error if fed, for example, a three-element list, so you need a catch or try in there.

CecilWesterhof 2018-07-23: No, it will go OK:
$ set d [list a b c]
a b c
$ isDict $d
0

EMJ 2018-07-23: Ah, yes - lazy evaluation of && .

PYK 2018-07-23: This topic is also discussed at dict tips and tricks. In a nutshell, just ask Tcl:
proc isDict value {
    expr {![catch {dict size $value}]}
}

Often one just wants a script to produce an error if a value is not a dictionary. In this case its sufficient to add this at the right spot, without doing any additional assignment or error production:
dict size $value

Also, opinions don't differ about what is or isn't a correct dictionary. By definition, a list with an even number of elements is dictionary.