Developer on the Language Interoperability Team at InterSystems.
@Andre Larsen Barbosa - I am the author of the intiial dynamic SQL 'feature'. I no longer work in this area and many significant enhancements have been made since my initial work. That said, I applaud your article. It is quite nice.
There is one feature that not many people know about - there is an Object mode that you can use with Dynamic Statement. Object mode is interesting when your query selects column values whose underlying type is a IRIS ObjectScript Class that is an object type (Persistent, Serial, Stream). Here is a trivial example of how it might be used.
LATEST:USER>set statement = ##class(%SQL.Statement).%New() LATEST:USER>set statement.%ObjectSelectMode = 1 LATEST:USER>do statement.prepare("select name,address from person") LATEST:USER>set result = statement.execute() LATEST:USER>write result.%Next() 1 LATEST:USER>write result.name po1 LATEST:USER>write result.address 7@User.Address LATEST:USER>write result.address.city Boston LATEST:USER>write result.address.state MA
set stream = ##class(%Stream.FileCharacter).%OpenId("/Users/.../data/continents-en.json")
Did you know that you can just open a file as a stream?
And for iterating - this has been in the product for a while now. This is the code for my :pp alias.
ClassMethod pp(set As %AbstractSet, offset As %Integer = 0)
{
#define QUOTE(%val) $zu(144,1,%val)
try {
set isLabeled = set."_isLabeled"()
if (isLabeled) {
write "{"
set close = "}"
} else {
write "["
set close = "]"
}
set it = set.iterator()
while it.hasNext() {
set next = it.next()
if $isobject(next.value) {
write !,?(offset+2)
write:isLabeled $$$QUOTE(next.key),": "
do ..pp(next.value, offset + 2)
} else {
write !,?(offset+2),$select(isLabeled:$$$QUOTE(next.key)_": ",1:""),$$$QUOTE(next.value)
}
if it.hasNext() {
write ","
}
}
write !,?offset,close
} catch exc {
write !,"Exception caught: ",exc.AsSQLMessage()
}
return
}
This is in %ASQ.SetUtils. My alias, pp - pretty print, is this:
pp do ##class(%ASQ.SetUtils).pp($*)
data:image/s3,"s3://crabby-images/8141a/8141a8895995b9ce7ef1ab0ddf39157b3198a8a9" alt="DC Author"
data:image/s3,"s3://crabby-images/2c8d9/2c8d94971a70cff8ba90800b14d816dc9818d1a3" alt="DC Commenter"
data:image/s3,"s3://crabby-images/8d2bf/8d2bfb88f5369a41a816c658e3c041456559050f" alt="DC Problem Solver"
data:image/s3,"s3://crabby-images/6d39e/6d39ea300a89c95fc70a4e5bba45842efa2db7eb" alt="Bronze Commenter"
data:image/s3,"s3://crabby-images/b66c0/b66c0fcc486cd218e178d93d050cba84aecd3ac4" alt="Insightful Commenter"
data:image/s3,"s3://crabby-images/48d30/48d302c1309ebea6351c5ca674e620c8826498d7" alt="Master of Answers"
data:image/s3,"s3://crabby-images/8555b/8555bc9e846efb0ccab7a3145ef899d9e4d19115" alt="Bronze Master of Answers"
Dynamic SQL doesn't have to build the query solution every time you run the same statement. Once you initially prepare an SQL Statement, the implementation of that statement is cached and reused. The only overhead should be limited to a hopefully very brief cache resolve. There are ways that you can write a statement that makes the hashing algorithm we use very inefficient (prepare:execute ratio p:n, the larger 'n' is, the better - 1:1 being the worst). Since we now use a unified model, even embedded SQL will produce a cached implementation. Embedded reduces the cache resolve time.