diff --git a/process/Classification_lib/process.js b/process/Classification_lib/process.js
index eeb17f665f07de2fc81ee1452e04ff811b272a66..126a6dde2d44fd2fa1f2421e7f4dcab7acbc13c9 100644
--- a/process/Classification_lib/process.js
+++ b/process/Classification_lib/process.js
@@ -96,7 +96,7 @@ ClassificationUtils.mapToClass = function(pScore)
 ClassificationUtils.getAllGroups = function(pClassificationtype, pTwoCol)
 {
     var sql = new SqlBuilder()
-        .selectDistinct(["CLASSIFICATIONGROUP", (pTwoCol ? "CLASSIFICATIONGROUP" : "")])
+        .select(["distinct CLASSIFICATIONGROUP", (pTwoCol ? "CLASSIFICATIONGROUP" : "")])
         .from("CLASSIFICATIONTYPE")
         .where("CLASSIFICATIONTYPE.CLASSIFICATIONTYPE", pClassificationtype)
         .orderBy("CLASSIFICATIONGROUP");
diff --git a/process/SqlLib_tests/process.js b/process/SqlLib_tests/process.js
index c6a50cae3d8066a741a0b00790701af67c682039..c577854128e02049fc14fd7612578a419f2d58ea 100644
--- a/process/SqlLib_tests/process.js
+++ b/process/SqlLib_tests/process.js
@@ -777,7 +777,7 @@ var testConstantFunctions = new TestSuite([
 ]);
 
 var selectTests = new TestSuite([
-    ["test if a sql-builder in a fields-array is translated to sql correctly", function(pTester)
+    ["a sql-builder in a fields-array is translated to sql correctly", function(pTester)
     {
         var countSubQuery = newSelect("count(*)")
                                 .from("AB_ATTRIBUTEUSAGE")
@@ -791,6 +791,36 @@ var selectTests = new TestSuite([
         pTester.assert("select AB_ATTRIBUTEID, AB_ATTRIBUTEUSAGEID, (select count(*) from AB_ATTRIBUTEUSAGE where AB_ATTRIBUTEUSAGE.OBJECT_TYPE = ? and AB_ATTRIBUTEUSAGE.AB_ATTRIBUTE_ID = AB_ATTRIBUTE.AB_ATTRIBUTEID)", actual._select._sqlStorage, "prepared select-sql");
         pTester.assert(1, actual._select.preparedValues.length, "number of params");
     }],
+
+    ["a sql-builder in from is used as subselect", function(pTester)
+    {
+        var subQuery = newSelect("FIRSTNAME")
+                                .from("PERSON")
+                                .where("PERSON.LASTNAME", "Meier")
+    
+        var actual = new SqlBuilder()
+            .select("*")
+            .from(subQuery)
+            
+        pTester.assert("from (select FIRSTNAME from PERSON where PERSON.LASTNAME = ?)", actual._from._sqlStorage, "prepared select-sql");
+        pTester.assert(1, actual._from.preparedValues.length, "number of params");
+    }],
+
+    ["SqlBuilder as on-condition should only add the conditon of the builder", function(pTester)
+    {
+        var subQuery = newSelect("NAME")
+                                .from("ORGANISATION")
+                                .where("ORGANISATION.NAME", "Adito")
+    
+        var actual = new SqlBuilder()
+            .select("*")
+            .from("PERSON")
+            .join("ORGANISATION", subQuery)
+            
+        logging.log(JSON.stringify([actual._joins], null, "\t"))
+        pTester.assert("join ORGANISATION on ORGANISATION.NAME = ?", actual._joins[0]._sqlStorage, "prepared select-sql");
+        pTester.assert(1, actual._joins[0].preparedValues.length, "number of params");
+    }],
 ])
 
 var tester = new Tester("Test SqlBuilder");
diff --git a/process/Sql_lib/process.js b/process/Sql_lib/process.js
index 729e428f310070d90554d89204e13e9c64ef0e06..e8292830a9bea17613f9ae9b81cdd5075e143e01 100644
--- a/process/Sql_lib/process.js
+++ b/process/Sql_lib/process.js
@@ -13,9 +13,9 @@ import("Util_lib");
  * Creates a new SqlBuilder object and sets the select clause of the sql.
  * 
  * @param {String|Array|SqlBuilder} pFields You can pass:
- *                                          - A String is just used AS IT IS.
- *                                          - The array can also contain Strings, SqlBuilder which are just concatenated
+ *                                          - A String is just used AS IT IS. (e.g. "FIRSTNAME, LASTNAME")
  *                                          - SqlBuilder is used as Subquery
+ *                                          - The array can also contain Strings, SqlBuilder which are just concatenated (e.g. ["FIRSTNAME", "LASTNAME", someSqlBuilderContainingFullSelect])
  *                                          Please see .select() for more information and examples.
  * @param {String} [pAlias=currentAlias] This alias is used for fetching the ColumnTypes and also for the .table, .cell, .updateData, ... -functions
  * @return {SqlBuilder} A new SqlBuilder object already containing the provided fields
@@ -236,11 +236,11 @@ SqlBuilder.prototype.toString = function(pDefaultConditionIfNone)
 /**
  * Sets the select clause of the sql.
  * @param {String|Array|SqlBuilder} pFields You can pass:
- *                                          - A String is just used AS IT IS.
- *                                          - The array can also contain Strings, SqlBuilder which are just concatenated
+ *                                          - A String is just used AS IT IS. (e.g. "FIRSTNAME, LASTNAME")
  *                                          - SqlBuilder is used as Subquery
- *
- *@return {SqlBuilder} current SqlBuilder object
+ *                                          - The array can also contain Strings, SqlBuilder which are just concatenated (e.g. ["FIRSTNAME", "LASTNAME", someSqlBuilderContainingFullSelect])
+ * 
+ * @return {SqlBuilder} current SqlBuilder object
  */
 SqlBuilder.prototype.select = function(pFields)
 {
@@ -248,20 +248,14 @@ SqlBuilder.prototype.select = function(pFields)
     return this;
 }
 
-/**
- * Sets the select clause of the sql with distinct.
- * @param {String|String[]} pFields
- * @return {SqlBuilder} current SqlBuilder object
- */
-SqlBuilder.prototype.selectDistinct = function(pFields)
-{
-    this._select = SqlBuilder._getStatement(pFields, "select distinct", undefined, true);
-    return this;
-}
-
 /**
  * Sets the from clause of the sql.
- * @param {String|SqlBuilder} pTable
+ * 
+ * Note: It is recommended to add joins via the .join functions.
+ *       But in some cases you may already get a full from clause including the joins. In this case it is also possible to include them in the from-string.
+ * 
+ * @param {String|SqlBuilder} pTable if it is a String, it is used as it is as table
+ *                                   if it is a SqlBuilder, it is used as subselect: e.g. select * from (select FIRSTNAME from PERSON)
  * @param {String} [pTableAlias] table alias
  * @return {SqlBuilder} current SqlBuilder object
  */
@@ -275,9 +269,13 @@ SqlBuilder.prototype.from = function(pTable, pTableAlias)
 
 /**
  * Adds a join clause to the sql.
- * @param {String|SqlBuilder} pTable
+ * 
+ * @param {String|SqlBuilder} pTable if it is a String, it is used as it is as table
+ *                                   if it is a SqlBuilder, it is used as subselect: e.g. select * from (select FIRSTNAME from PERSON)
  * @param {String|String[]|SqlBuilder} pCondition The where condition. This can be
- *          a string (without the where keyword), a SqlBuilder or an array (for prepared queries).
+ *                                                - a string (without the where keyword)
+ *                                                - a SqlBuilder 
+ *                                                - an array (for prepared queries).
  * @param {String} [pTableAlias] table alias
  * @param {String} [pPrefix] string before the join, for example "left", "right"
  * @return {SqlBuilder} current SqlBuilder object
@@ -293,6 +291,10 @@ SqlBuilder.prototype.join = function(pTable, pCondition, pTableAlias, pPrefix)
         postfix = pTableAlias + " " + postfix;
     
     var joinPart = SqlBuilder._getStatement(pTable, prefix, postfix);
+    
+    if (pCondition instanceof SqlBuilder)
+        pCondition = [pCondition._where._sqlStorage, pCondition._where.preparedValues]
+    
     var conditionPart = SqlBuilder._getStatement(pCondition);
     
     joinPart._sqlStorage += " " + conditionPart._sqlStorage;
@@ -934,12 +936,13 @@ SqlBuilder._getStatement = function (pElement, pPrefix, pPostfix, pAutoJoin)
     {
         if (Array.isArray(pElement) && pElement.length !== undefined && pAutoJoin) //array of fields
         {
-            for (let i = 0, l = pElement.length; i < l; i++)
+            for (let i = 0; i < pElement.length; i++)
             {
                 if (typeof pElement[i] !== "string")
                     pElement[i] = _getElement(pElement[i]);
             }
-            pElement = pElement.join(", ");
+             
+            pElement = ArrayUtils.joinNonEmptyFields(pElement, ", ");
         }
         else
         {