DEV Community

Michael
Michael

Posted on • Originally published at gbase.cn

Working with ROW Types in GBase 8s JDBC

GBase 8s provides support for the ROW type, a composite data type that can contain multiple fields, each with its own data type. ROW types can be used to define table columns, program variables, statement variables, and routine return values, making it more flexible and intuitive to handle nested data structures in JDBC.

Defining ROW Types

The following statements define four ROW types. Note that person_t nests fullname_t, demonstrating multi-level composite structures:

CREATE ROW TYPE r1_t (i INT, b BOOLEAN);
CREATE ROW TYPE r2_t (f FLOAT, s MULTISET(INT8 NOT NULL), i INT);
CREATE ROW TYPE fullname_t (first CHAR(20), last CHAR(20));
CREATE ROW TYPE person_t (id INT, name fullname_t, age INT);
Enter fullscreen mode Exit fullscreen mode

SQLData Implementation Classes

Use the SQLData interface to map Java objects to ROW types. Regular fields are read and written with methods like readInt and writeDouble; nested ROW types use readObject/writeObject; collection types such as MULTISET require IfmxComplexSQLInput/IfmxComplexSQLOutput.

r1_t Implementation

public class r1_t implements SQLData {
    int     int_col;
    boolean bool_col;

    public String toString() {
        String str = "int_col: " + int_col + "    bool_col: " + bool_col;
        return (str);
    }
}
Enter fullscreen mode Exit fullscreen mode

r2_t Implementation

public class r2_t implements SQLData {
    double     dbl_col;
    int        int_col;
    Collection mset_col;

    public void readSQL(SQLInput stream, String typeName) throws SQLException {
        dbl_col = stream.readDouble();
        mset_col = (Collection) ((IfmxComplexSQLInput) stream).readObject();
        int_col = stream.readInt();
    }

    public void writeSQL(SQLOutput stream) throws SQLException {
        stream.writeDouble(dbl_col);
        ((IfmxComplexSQLOutput) stream).writeObject(mset_col);
        stream.writeInt(int_col);
    }

    public String toString() {
        String str = "dbl_col: " + dbl_col + "\n";
        str += "                int_col: " + int_col + "\n";
        str += "                mset_col: \n";
        Iterator it = mset_col.iterator();
        while (it.hasNext()) {
            str += "\t\t\telement: " + it.next() + "\n";
        }
        return (str);
    }
}
Enter fullscreen mode Exit fullscreen mode

fullname Implementation

public class fullname implements SQLData {
    public String first;
    public String last;
}
Enter fullscreen mode Exit fullscreen mode

person Implementation

public class person implements SQLData {
    public int      id;
    public fullname name;
    public int      age;

    public void readSQL(SQLInput stream, String type) throws SQLException {
        id = stream.readInt();
        name = (fullname) stream.readObject();
        age = stream.readInt();
    }

    public void writeSQL(SQLOutput stream) throws SQLException {
        stream.writeInt(id);
        stream.writeObject(name);
        stream.writeInt(age);
    }

    public String toString() {
        String s = "person:";
        s += "id: " + id + "\n";
        s += "     name: " + name.toString() + "\n";
        s += "     age: " + age + "\n";
        return s;
    }
}
Enter fullscreen mode Exit fullscreen mode

Seven Practical Examples

Example 1: Fetching a Named ROW into a SQLData Object

Create the teachers table, insert a row, then use a custom TypeMap to map the person column to a Java object.

stmt.executeUpdate("create table teachers (person person_t, dept char (20))");
stmt.executeUpdate("insert into teachers values ('row(100, row(Bill, Smith), 27)', 'physics')");

java.util.Map map = conn.getTypeMap();
map.put("fullname_t", Class.forName("com.jdbc.demo.fullname"));
map.put("person_t", Class.forName("com.jdbc.demo.person"));

PreparedStatement pstmt = conn.prepareStatement("select person from teachers");
ResultSet rs = pstmt.executeQuery();
while (rs.next()) {
    person who = (person) rs.getObject(1);
    System.out.println("row: " + who.toString());
}
Enter fullscreen mode Exit fullscreen mode

Output:

row: person:id: 100
   name: fullname: first: Bill                 last: Smith               
   age: 27
Enter fullscreen mode Exit fullscreen mode

Example 2: Inserting a SQLData Object into a Named ROW Column

Build a person object and insert it with setObject. When querying, cast the ROW to lvarchar for display.

stmt.executeUpdate("create table teachers (person person_t, dept char (20))");

java.util.Map map = conn.getTypeMap();
map.put("fullname_t", Class.forName("com.jdbc.demo.fullname"));
map.put("person_t", Class.forName("com.jdbc.demo.person"));

person who = new person();
fullname name = new fullname();
name.last = "Jones";
name.first = "Sarah";
who.id = 567;
who.name = name;
who.age = 17;

String s = "insert into teachers values (?, 'physics')";
PreparedStatement pstmt = conn.prepareStatement(s);
pstmt.setObject(1, who);
pstmt.executeUpdate();

pstmt = conn.prepareStatement("select person::lvarchar from teachers");
ResultSet rs = pstmt.executeQuery();
while (rs.next()) {
    System.out.println("lvarchar: " + rs.getString(1));
}
Enter fullscreen mode Exit fullscreen mode

Output:

lvarchar: ROW(567        ,ROW('Sarah               ','Jones               '),17         )
Enter fullscreen mode Exit fullscreen mode

Example 3: Fetching an Unnamed SQLROW Column into a Struct Object

For unnamed ROW columns, use java.sql.Struct. Retrieve nested fields with getAttributes().

stmt.executeUpdate("create table teachers (person row(id int, name row(first char(20), last char(20)), age int), dept char (20))");
stmt.executeUpdate("insert into teachers values ('row(100, row(Bill, Smith), 27)', 'physics')");

PreparedStatement pstmt = conn.prepareStatement("select person from teachers");
ResultSet rs = pstmt.executeQuery();
rs.next();

Struct person = (Struct) rs.getObject(1);
System.out.println("person row description: " + person.getSQLTypeName());

Object[] elements = person.getAttributes();
Integer id = (Integer) elements[0];
Struct name = (Struct) elements[1];
Integer age = (Integer) elements[2];

System.out.println("person.id: " + id);
System.out.println("person.age: " + age);

System.out.println("name row description: " + name.getSQLTypeName());
Object[] nameAttrs = name.getAttributes();
String first = (String) nameAttrs[0];
String last = (String) nameAttrs[1];
System.out.println("name.first: " + first);
System.out.println("name.last: " + last);
Enter fullscreen mode Exit fullscreen mode

Output:

person row description:  row ( id int , name row ( first char(20) , last char(20) ) , age int ) 
person.id: 100
person.age: 27
name row description: name row ( first char(20) , last char(20) )
name.first: Bill                
name.last: Smith
Enter fullscreen mode Exit fullscreen mode

Example 4: Inserting a Struct Object into a Named ROW

Implement a custom GenericStruct class satisfying java.sql.Struct, build attributes, and insert with setObject.

public class GenericStruct implements java.sql.Struct {
    private Object[] attributes = null;
    private String   typeName   = null;

    GenericStruct() {}
    GenericStruct(String name, Object[] obj) {
        typeName = name;
        attributes = obj;
    }
    public String getSQLTypeName() { return typeName; }
    public Object[] getAttributes() { return attributes; }
    public Object[] getAttributes(Map map) throws SQLException { return attributes; }
    public void setAttributes(Object[] objArray) { attributes = objArray; }
    public void setSQLTypeName(String name) { typeName = name; }
}
Enter fullscreen mode Exit fullscreen mode
stmt.executeUpdate("create table teachers (person person_t, dept char (20))");

PreparedStatement pstmt = conn.prepareStatement("insert into teachers values (?, 'Math')");

Object[] name = new Object[2];
name[0] = new String("Jane");
name[1] = new String("Smith");
String rowType = "row(first char(20), last char(20))";
GenericStruct gs = new GenericStruct(rowType, name);

Object[] person = new Object[3];
person[0] = new Integer(99);
person[1] = gs;
person[2] = new Integer(56);
rowType = "row(id int, name row(first char(20), last char(20)), age int)";
gs = new GenericStruct(rowType, person);

pstmt.setObject(1, gs);
pstmt.executeUpdate();

pstmt = conn.prepareStatement("select person::lvarchar from teachers");
ResultSet rs = pstmt.executeQuery();
while (rs.next()) {
    System.out.println("lvarchar: " + rs.getString(1));
}
Enter fullscreen mode Exit fullscreen mode

Output:

lvarchar: ROW(99         ,ROW('Jane                ','Smith               '),56         )
Enter fullscreen mode Exit fullscreen mode

Example 5: Reading and Writing a Simple ROW Type (r1_t)

Use a custom TypeMap to map r1_t. Demonstrates reading data both as SQLData and as Struct, then re-inserting using the retrieved Struct.

stmt.executeUpdate("create table row_tab (int_col int, row_col r1_t)");

// Insert an SQLData object
PreparedStatement pstmt = conn.prepareStatement("insert into row_tab (int_col, row_col) values (?, ?)");
java.util.Map customtypemap = conn.getTypeMap();
customtypemap.put("r1_t", Class.forName("com.jdbc.demo.r1_t"));
r1_t row = new r1_t(99, true);
pstmt.setInt(1, 1);
pstmt.setObject(2, row);
pstmt.executeUpdate();

// Read as SQLData
Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("select int_col, row_col from row_tab order by int_col");
rs.next();
r1_t rowData = (r1_t) rs.getObject("row_col");
System.out.println(rowData); // int_col: 99    bool_col: true

// Read as Struct
rs = stmt.executeQuery("select int_col, row_col from row_tab order by int_col");
rs.next();
Struct rowStruct = (Struct) rs.getObject("row_col", null);
Object[] objArray = rowStruct.getAttributes();
for (int i = 0; i < objArray.length; i++) {
    System.out.println("\t\tfield: " + objArray[i]);
}

// Re-insert using the retrieved Struct objects (table truncation omitted for brevity)
pstmt = conn.prepareStatement("insert into row_tab values (?, ?)");
for (int i = 0; i < objVector.size(); i++) {
    pstmt.setInt(1, i);
    pstmt.setObject(2, objVector.get(i));
    pstmt.executeUpdate();
}
rs = stmt.executeQuery("select int_col, row_col::lvarchar from row_tab order by int_col");
rs.next();
System.out.println(rs.getString(2)); // ROW(99         ,'t')
Enter fullscreen mode Exit fullscreen mode

Sample output:

int_col: 99    bool_col: true
        field: 99
        field: true
row_col       ROW(99         ,'t')
Enter fullscreen mode Exit fullscreen mode

Example 6: Reading and Writing a ROW with MULTISET (r2_t)

r2_t includes a MULTISET(INT8 NOT NULL) field. The example builds a TreeSet, inserts the row, and queries it back.

stmt.executeUpdate("create table row_tab (int_col int, row_col r2_t)");
java.util.Map customtypemap = conn.getTypeMap();
customtypemap.put("r2_t", Class.forName("r2_t"));

String s = "insert into row_tab (int_col, row_col) values (?, ?)";
PreparedStatement pstmt = conn.prepareStatement(s);
TreeSet set = new TreeSet();
set.add(new Long(1234567890));
set.add(new Long(-1234567890));
set.add(new Long(5));
r2_t row = new r2_t(1.67e30, set, 99);
pstmt.setInt(1, 1);
pstmt.setObject(2, row);
pstmt.executeUpdate();

Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("select int_col, row_col from row_tab order by int_col");
rs.next();
r2_t rowData = (r2_t) rs.getObject("row_col");
System.out.println(rowData);
Enter fullscreen mode Exit fullscreen mode

Output:

dbl_col: 1.67E30
int_col: 99
mset_col: 
    element: -1234567890
    element: 5
    element: 1234567890
Enter fullscreen mode Exit fullscreen mode

Example 7: Reading and Writing an Unnamed ROW with LIST

For unnamed ROWs containing a LIST field, use GenericStruct in the same way. The ArrayList is passed and automatically parsed on retrieval.

stmt.executeUpdate("create table row_tab (int_col int, row_col row(f float, l list(lvarchar not null), i int))");

String s = "insert into row_tab (int_col, row_col) values (?, ?)";
PreparedStatement pstmt = conn.prepareStatement(s);

ArrayList list = new ArrayList();
list.add(new String("John Smith"));
list.add(new String("Granny Apple"));
list.add(new String("Barry White"));

Object[] objArray = new Object[3];
objArray[0] = new Double(1.67e30);
objArray[1] = list;
objArray[2] = new Integer(99);
GenericStruct struct = new GenericStruct("row(f float, l list(lvarchar not null), i int)", objArray);

pstmt.setInt(1, 1);
pstmt.setObject(2, struct);
pstmt.executeUpdate();

Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("select int_col, row_col from row_tab order by int_col");
rs.next();
Struct row = (Struct) rs.getObject("row_col");
System.out.println("Class: " + row.getClass().getName());
System.out.println("SQLTypeName: " + row.getSQLTypeName());
Object[] attrs = row.getAttributes();
for (int i = 0; i < attrs.length; i++) {
    System.out.println("field: " + attrs[i]);
}
Enter fullscreen mode Exit fullscreen mode

Output:

Class: com.gbasedbt.jdbc.IfxStruct
SQLTypeName:  row ( f float , l list (  lvarchar not null) , i int ) 
field: -1.67E30
field: [Roger Jones, Jeff Brown]
field: -99
Enter fullscreen mode Exit fullscreen mode

Through this series of examples, you can systematically master reading and writing ROW types in the GBase 8s JDBC environment, enabling efficient interaction between Java objects and nested database structures in a gbase database.

Top comments (0)