Forum Replies Created

Page 4 of 22
  • rocuinneagain

    Member
    July 22, 2024 at 9:48 am in reply to: How can I apply a func to a grouped table?

    Sample table:

    q)t:([] stock:10000?`1;AskOrder:10000?100;BidOrder:10000?100)
    q)t
    stock AskOrder BidOrder
    -----------------------
    n 85 43
    p 40 15
    n 9 47
    j 10 64
    i 87 88
    i 79 87
    m 46 41
    h 92 63
    g 86 39
    ......

    Function which returns a dictionary of percentiles:

    q)f:{(`$x,/:string 1+til y)!az -1+(where deltas y xrank az:asc z),count z}

    Running the function returns a dictionary:

    q)r:exec f["Ask_";4;AskOrder],f["Bid_";4;BidOrder] by stock from t
    q)r
    | AskOrder BidOrder
    -| -------------------------------------------------------------------------
    a| `Ask_1`Ask_2`Ask_3`Ask_4!25 50 74 99 `Bid_1`Bid_2`Bid_3`Bid_4!23 51 74 99
    b| `Ask_1`Ask_2`Ask_3`Ask_4!26 49 77 99 `Bid_1`Bid_2`Bid_3`Bid_4!23 47 74 99
    c| `Ask_1`Ask_2`Ask_3`Ask_4!23 49 74 99 `Bid_1`Bid_2`Bid_3`Bid_4!26 49 75 99
    d| `Ask_1`Ask_2`Ask_3`Ask_4!29 53 78 99 `Bid_1`Bid_2`Bid_3`Bid_4!27 50 74 99
    e| `Ask_1`Ask_2`Ask_3`Ask_4!22 46 73 99 `Bid_1`Bid_2`Bid_3`Bid_4!26 54 79 99
    f| `Ask_1`Ask_2`Ask_3`Ask_4!25 49 75 99 `Bid_1`Bid_2`Bid_3`Bid_4!23 48 74 99
    g| `Ask_1`Ask_2`Ask_3`Ask_4!26 49 73 99 `Bid_1`Bid_2`Bid_3`Bid_4!27 50 76 99
    h| `Ask_1`Ask_2`Ask_3`Ask_4!24 51 77 99 `Bid_1`Bid_2`Bid_3`Bid_4!27 52 76 99
    i| `Ask_1`Ask_2`Ask_3`Ask_4!24 48 74 99 `Bid_1`Bid_2`Bid_3`Bid_4!23 49 72 99
    j| `Ask_1`Ask_2`Ask_3`Ask_4!23 46 72 99 `Bid_1`Bid_2`Bid_3`Bid_4!31 53 75 99
    k| `Ask_1`Ask_2`Ask_3`Ask_4!21 44 72 99 `Bid_1`Bid_2`Bid_3`Bid_4!22 50 75 99
    l| `Ask_1`Ask_2`Ask_3`Ask_4!24 49 73 99 `Bid_1`Bid_2`Bid_3`Bid_4!27 50 73 99
    m| `Ask_1`Ask_2`Ask_3`Ask_4!23 45 70 99 `Bid_1`Bid_2`Bid_3`Bid_4!25 52 78 99
    n| `Ask_1`Ask_2`Ask_3`Ask_4!26 52 77 99 `Bid_1`Bid_2`Bid_3`Bid_4!25 53 76 99
    o| `Ask_1`Ask_2`Ask_3`Ask_4!22 50 75 99 `Bid_1`Bid_2`Bid_3`Bid_4!24 47 74 99
    p| `Ask_1`Ask_2`Ask_3`Ask_4!24 51 76 99 `Bid_1`Bid_2`Bid_3`Bid_4!23 48 74 99

    Transforming the dictionary in to a table:

    q){`stock xcols update stock:key x from (value x)[`AskOrder],'(value x)[`BidOrder]} r
    stock Ask_1 Ask_2 Ask_3 Ask_4 Bid_1 Bid_2 Bid_3 Bid_4
    -----------------------------------------------------
    a 25 50 74 99 23 51 74 99
    b 26 49 77 99 23 47 74 99
    c 23 49 74 99 26 49 75 99
    d 29 53 78 99 27 50 74 99
    e 22 46 73 99 26 54 79 99
    f 25 49 75 99 23 48 74 99
    g 26 49 73 99 27 50 76 99
    h 24 51 77 99 27 52 76 99
    i 24 48 74 99 23 49 72 99
    j 23 46 72 99 31 53 75 99
    k 21 44 72 99 22 50 75 99
    l 24 49 73 99 27 50 73 99
    m 23 45 70 99 25 52 78 99
    n 26 52 77 99 25 53 76 99
    o 22 50 75 99 24 47 74 99
    p 24 51 76 99 23 48 74 99
  • rocuinneagain

    Member
    July 17, 2024 at 12:53 pm in reply to: Using the partition type as a parameter

    It is advised to avoid using name of column or globals as function param names as much as possible. Here you are seeing issues as a result of this.

    In a HDB the issues caused by this scoping confusion can differ from in memory tables.

    This is due to map-reduce in the HDB, which means the query execution path is more complicated.

    1. type error:

    In the query path date would be expected to be a list but if the function local date atom is used a type error is being raised

    q){2025 select cnt:count i from Orders where date=2024.07.15}[.z.d] //Error
    q){[] select cnt:count i from Orders where date=2024.07.15}[.z.d] //Okay

    2. count = 0:

    You enlist the data param in one of the uses which means = returns a list and does not get the type error of query 1. However this list value does not make sense so an unexpected value is returned.

    Returns the count of the table in the first partition of the HDB

    q){2025 select cnt:count i from Orders where date=enlist 2024.07.15}[2024.07.15]

    Same result

    q){2025 select cnt:count i from Orders where date=2024.07.15}[enlist 2024.07.15]

    Returns the counts of the first 2 partitions

    q){2025 select cnt:{enlist count x}i from Orders where date=.z.d}[2#.z.d]

    You don’t even need to put in dates as the date param is being compared to itself you can put anything in:

    q){2025 select cnt:{enlist count x}i from Orders where date=1b}[11b]

    3. Works: Not using date as a param name – no issues

  • rocuinneagain

    Member
    July 11, 2024 at 1:48 pm in reply to: pykx is returning different output compared to embedQ

    PyKX also will convert a hsym directly to a Path type object

    q).pykx.print[.pykx.eval["lambda x:x"]`:/path/to/file]
    /path/to/file
    q).pykx.print[.pykx.eval["lambda x:type(x)"]`:/path/to/file]
    <class 'pathlib.PosixPath'>

    If you want a string of the Path you can call:

    streamer:.pylibs.security[`:smStreamer] .pykx.eval["lambda x:str(x)"] file
  • rocuinneagain

    Member
    July 11, 2024 at 1:26 pm in reply to: pykx is returning different output compared to embedQ

    See: Upgrading from embedPy – PyKX

    PyKX converts Python strings to q symbols which differs from embedPy

    q).pykx.eval["'hello'"]`
    `hello

    Pass your data back as Bytes for it to become q character vectors

    q)string_to_bytes:.pykx.eval["lambda x: bytes(x, 'utf-8')"]
    q)string_to_bytes[.pykx.eval["'hello'"]]`
    "hello"

  • rocuinneagain

    Member
    July 8, 2024 at 8:24 am in reply to: create random symbol with length >8

    8 is a documented limit

    https://code.kx.com/q/ref/deal/#generate

    symbols, each of n chars (n≤8)

    To create 3 random symbols of length 20:

    q){`$y cut (x*y)?16#.Q.a}[3;20]

    `hgpfbbonnjlpppnpjago`mebadmhpkcjgoheikpmh`pbkbabaofjhmacenfabc

  • rocuinneagain

    Member
    July 3, 2024 at 12:20 pm in reply to: Compression for null string column

    I expect this entry in 4.0 README for 2022.04.15 is the version from where you will see the improvement:


    2022.04.15
    NEW
    anymap write now detects consecutive deduplicated (address matching) toplevel objects, skipping them to save space
    q)a:("hi";"there";"world");`:a0 set a;`:a1 set a@where 1000 2000 3000;(hcount`$":a0#")=hcount`$":a1#"
    improved memory efficiency of writing nested data sourced from a type 77 file, commonly encountered during compression of files. e.g.
    q)`:a set 500000 100#"abc";system"ts `:b set get`:a" / was 76584400 bytes, now 8390720
  • rocuinneagain

    Member
    July 3, 2024 at 12:07 pm in reply to: Compression for null string column

    <div>Can you test against a newer version of 4.0? </div>

    My 4.1 gets much improved numbers:

    q)(.z.K;.z.k)
    4.1
    2024.04.29
    q)n:10000000;tab:([]time:n#.z.p;val:n?1000;str:n#enlist "");(`:tab/;17;2;5) set tab
    `:tab/
    q)-21!`:tab/str
    compressedLength | 136807
    uncompressedLength| 80004096
    algorithm | 2i
    logicalBlockSize | 17i
    zipLevel | 5i
    //Your compression 5.6x
    q)80004096%14074225
    5.684441
    //Compression now 584x
    80004096%136807
    584.7953
    q)-21!`$":tab/str#"
    compressedLength | 93
    uncompressedLength| 4098
    algorithm | 2i
    logicalBlockSize | 17i
    zipLevel | 5i
  • rocuinneagain

    Member
    July 2, 2024 at 10:56 am in reply to: Apply attributes on-disk in 4.1

    Yes. The website changelogs cover main/highlighted features only.

    For full changelog reference the README.txt for your version.

  • rocuinneagain

    Member
    July 1, 2024 at 12:57 pm in reply to: help with VWAP function
    Sample table for testing:
    q)data:([] timestamp:til 10;bq0:til 10;bq1:til 10;bq2:til 10;bp0:til 10;bp1:til 10;bp2:til 10;aq0:til 10;aq1:til 10;aq2:til 10;ap0:til 10;ap1:til 10;ap2:til 10) 
    q)data 
    timestamp bq0 bq1 bq2 bp0 bp1 bp2 aq0 aq1 aq2 ap0 ap1 ap2 
    --------------------------------------------------------- 
    0         0   0   0   0   0   0   0   0   0   0   0   0 
    1         1   1   1   1   1   1   1   1   1   1   1   1 
    2         2   2   2   2   2   2   2   2   2   2   2   2 
    3         3   3   3   3   3   3   3   3   3   3   3   3 
    4         4   4   4   4   4   4   4   4   4   4   4   4 
    5         5   5   5   5   5   5   5   5   5   5   5   5 
    6         6   6   6   6   6   6   6   6   6   6   6   6 
    7         7   7   7   7   7   7   7   7   7   7   7   7 
    8         8   8   8   8   8   8   8   8   8   8   8   8 
    9         9   9   9   9   9   9   9   9   9   9   9   9
    Test provided query:
    q)select timestamp, depth_vwap1: (bq0; bq1; aq0; aq1) wavg (bp0; bp1; ap0; ap1) from data 
    timestamp depth_vwap1 
    --------------------- 
    0 
    1         1 
    2         2 
    3         3 
    4         4 
    5         5 
    6         6 
    7         7 
    8         8 
    9         9
    parse query to functional form:
    q)parse"select timestamp, depth_vwap1: (bq0; bq1; aq0; aq1) wavg (bp0; bp1; ap0; ap1) from data" 
    ? 
    `data 
    () 
    0b 
    `timestamp`depth_vwap1!(`timestamp;(wavg;
    (enlist;`bq0;`bq1;`aq0;`aq1);
    (enlist;`bp0;`bp1;`ap0;`ap1)))
    Test functional query:
    q)?[data;();0b;`timestamp`depth_vwap1!(`timestamp;(wavg;
    (enlist;`bq0;`bq1;`aq0;`aq1);
    (enlist;`bp0;`bp1;`ap0;`ap1)))] 
    timestamp depth_vwap1 
    --------------------- 
    0 
    1         1 
    2         2 
    3         3 
    4         4 
    5         5 
    6         6 
    7         7 
    8         8 
    9         9
    Set up variables:
    q)maxDepth:2 
    q)quantities: `$raze(("bq"; "aq"),/:\:string til maxDepth) // Note `$ to convert strings to symbols 
    q)prices : `$raze(("bp"; "ap"),/:\:string til maxDepth) // Note `$ to convert strings to symbols
    Create needed list to pass to functional query:
    q)enlist,quantities 
    enlist 
    `bq0 
    `bq1 
    `aq0 
    `aq1 
    q)-1 .Q.s1 enlist,quantities; //.Q.s1 very useful to show full structure of object 
    (enlist;`bq0;`bq1;`aq0;`aq1)
    Test new dynamic functional select:
    q)?[data;();0b;`timestamp`depth_vwap1!(`timestamp;
    (wavg;enlist,quantities;enlist,prices))] 
    timestamp depth_vwap1 
    --------------------- 
    0 
    1         1 
    2         2 
    3         3 
    4         4 
    5         5 
    6         6 
    7         7 
    8         8 
    9         9
  • rocuinneagain

    Member
    July 1, 2024 at 10:24 am in reply to: Question about TCP and IPC handles limit

    https://code.kx.com/q/ref/dotz/#zpo-open

    https://code.kx.com/q/ref/dotz/#zpc-close

    You can use callbacks to see when handles open and close

    .z.po:{string[.z.p]," opened ",string x;x}
    .z.pc:{string[.z.p]," closed ",string x;x}

    Test this against your c.Close() calls to confirm it is closing the handles.

    If still seeing a problem raising an issue can be done at: https://github.com/KxSystems/csharpkdb/issues

  • rocuinneagain

    Member
    July 1, 2024 at 10:14 am in reply to: Question about TCP and IPC handles limit

    This limitation was removed in 4.1 release

    https://code.kx.com/q/basics/errors/#runtime-errors

    conn

    Too many connections. Max connections was 1022 prior to 4.1t 2023.09.15, otherwise the limit imposed by the operating system (operating system configurable for system/protocol).

    https://code.kx.com/q/releases/ChangesIn4.1/#unlimited-ipcwebsocket-connections

    The number of connections is now limited only by the operating system and protocol settings (system configurable).

  • rocuinneagain

    Member
    June 18, 2024 at 12:34 pm in reply to: Best way to access to kdb+ personally on windows

    Command prompt is the default, but the new Windows Terminal app is much nicer. I also use it to open sessions to WSL (Windows Subsystem for Linux).

    Visual Studio Code is a great option if building up bigger projects. An official kdb plugin is available for it.

  • rocuinneagain

    Member
    June 18, 2024 at 9:48 am in reply to: Help understanding scan adverb

    Testing speed and equivalence:

    fun_ema:{[lambda; liste];({[lambda; x; y]; (lambda*y)+ (x*1-lambda)}[lambda]\) liste}
    expma1:{[lambda;vector] {[x;y;z] (x*y)+z}\[ first vector; 1-lambda; vector * lambda]};
    q)a:10
    q)b:til 1000000
    q)\ts r1:fun_ema[a;b]
    174 32777680
    q)\ts r2:expma1[a;b]
    91 41166288
    q)r1~r2
    1b

    Your fun_ema performs *, +, *, a total of count[b] times. The operations are all on atom type data.

    expma1 performs *, + a total of count[b] times and , * once. The single use of * is on vector type data.

    q uses vector instructions for speed. These take advantage of CPU instructions to be more efficient

    q)a:til 1000000

    q)b: til 1000000

    q)\ts a+b // + operates on vectors and is fast

    1 8388800

    q)\ts a+'b // Using ' to force + to loop through each pair of atoms is much slower

    26 32777488

    q)\ts {x+y}'[a;b] //Wrapping in an unnecessary lambda is slower again

    62 32777792

    Since 4.0, kdb further sped up large vector operations by adding Multithreaded primitives which will spread the vector operations across multiple threads when available.

  • rocuinneagain

    Member
    June 17, 2024 at 7:54 am in reply to: select with ssr function

    https://code.kx.com/q/ref/each/

    Use each:

    q)select {[x]ssr[x;".C";""]} each orderID, cancelTime from t

    https://code.kx.com/q/basics/application/#projection

    Neater with projection:

    select ssr[;".C";""] each orderID, cancelTime from t

    https://code.kx.com/q/ref/drop/

    If you know .C is always at the end it’s much better to use drop _ rather than ssr as ssr is compute intensive

    select {-2_ x} each orderID, cancelTime from t

    https://code.kx.com/q/ref/maps/#each-left-and-each-right

    each-right /: is neater

    q)select -2_/:orderID, cancelTime from t
  • rocuinneagain

    Member
    June 18, 2024 at 9:15 am in reply to: when the backtick is required
Page 4 of 22