SQL Query Data in Cache
NCache provides you with the ability to query indexed cache data through its own SQL-like querying mechanism. It lets you search for keys fulfilling the given criteria and then returns the keys of the result set. You can use ExecuteReader()
or ExecuteScalar()
to retrieve this data.
Note
No matter which projections you use in your query, the result set will only return keys when getdata
is set to false.
Using SQL Query Prerequisites
- Create a new Console Application.
- Make sure that the data being added is serializable.
- Add NCache References by locating
%NCHOME%\NCache\bin\assembly\4.0
and adding Alachisoft.NCache.Web
and Alachisoft.NCache.Runtime
as appropriate.
- Ensure the searchable objects and their attributes are indexed and added in the cache using the basic NCache API.
- Include the
Alachisoft.NCache.Web.Caching
namespace in your application.
Note
NCache also supports escape sequences including \n
,\b
, \t
, and wild cards *
and ?
as characters for all NCache-supported SQL operators. To learn more about these escape sequences, refer to the NCache SQL-Like syntax guide.
Using ExecuteReader To Query Data in Cache
The ExecuteReader
performs queries on the cache based on the user-specified criteria. It returns a list of key-value pairs in a data reader which fulfills the query criteria. This key-value pair has a cache key and its respective values.
SQL Query For Keys
Here is a query sample that returns the keys fulfilling the criteria based on ProductID.
Important
Make sure the instance of ICacheReader
is always closed after execution, as it is necessary to clean the resources.
string query = "SELECT * FROM FQN.Product WHERE ProductID > ?";
var queryCommand = new QueryCommand(query);
queryCommand.Parameters.Add("ProductID",50000);
ICacheReader reader = cache.SearchService.ExecuteReader(queryCommand, false, 0);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
string result = reader.GetValue<string>(0);
}
}
else
{
}
String query = "SELECT * FROM FQN.Product WHERE ProductID > ?";
QueryCommand queryCommand = new QueryCommand(query);
HashMap<String, Object> parameters = new HashMap<>();
parameters.put("ProductID", 50000);
queryCommand.getParameters().putAll(parameters);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
cacheReader.getString(0);
}
}
query = "SELECT * FROM FQN.Product WHERE product_id > ?"
query_command = ncache.QueryCommand(query)
parameter = {"product_id": 50000}
query_command.set_parameters(parameter)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
result = reader.get_value(str, 1)
print(result)
else:
print("Query result is None")
var query = "SELECT * FROM FQN.Product WHERE ProductID > ?";
var queryCommand = new ncache.QueryCommand(query);
let parameter = new Map();
parameter.set("ProductID", 50000);
queryCommand.setParameters(parameter);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand, false, 0);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var result = reader.getValue(0, Number());
}
}
else
{
}
string query = "SELECT Product where this.ProductID = ?";
Hashtable values = new Hashtable();
values.Add("ProductID", 1001);
ICacheReader reader = cache.ExecuteReader(query, values);
if (reader.FieldCount > 0){
while (reader.Read()){
Product result = (Product)reader.GetValue(1);
}
}
else{
}
reader.Close();
Note
To ensure the operation is fail-safe, it is recommended to handle any potential exceptions within your application, as explained in Handling Failures.
Using the SQL IN Operator
To query the cache using this SQL-like query format with the IN operator, you can specify multiple attribute values after the IN keyword. Here's an example that retrieves ProductName and UnitsInStock from the cache having either 10, 20, or 30 UnitsInStock using ExecuteReader
.
string query = "SELECT ProductName, UnitsInStock From FQN.Product WHERE UnitsInStock IN (?, ?, ?)";
var queryCommand = new QueryCommand(query);
ArrayList unitsArray = new ArrayList();
unitsArray.Add(10);
unitsArray.Add(20);
unitsArray.Add(30);
queryCommand.Parameters.Add("UnitsInStock", unitsArray);
ICacheReader reader = cache.SearchService.ExecuteReader(queryCommand);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
string productName = reader.GetValue<string>("ProductName");
int unitsInStock = reader.GetValue<int>("UnitsInStock");
Console.WriteLine($"Product '{productName}' has {unitsInStock} Units In Stock.");
}
}
String query = "SELECT ProductName, UnitsInStock FROM FQN.Product WHERE UnitsInStock IN (?, ?, ?)";
QueryCommand queryCommand = new QueryCommand(query);
ArrayList<Integer> unitsArray = new ArrayList<>();
unitsArray.add(10);
unitsArray.add(20);
unitsArray.add(30);
queryCommand.getParameters().put("UnitsInStock", unitsArray);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
String productName = null;
productName = cacheReader.getValue("ProductName", String.class);
Integer unitsInStock = null;
unitsInStock = cacheReader.getValue("UnitsInStock", Integer.class);
System.out.println("Product '" + productName + "' has " + unitsInStock + " Units In Stock.");
}
}
query = "SELECT product_name, units_in_stock From FQN.Product WHERE units_in_stock IN (10,20,30)"
query_command = ncache.QueryCommand(query)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
product_name = reader.get_value(str, 1)
units_available = reader.get_value(int, 2)
print("Name: " + str(product_name))
print("Units: " + str(units_available))
else:
print("Query result is None")
var query = "SELECT productName, UnitsInStock From FQN.Product WHERE UnitsInStock IN (?,?,?)";
var queryCommand = new ncache.QueryCommand(query);
let units = [10, 15, 20, 25];
let parameter = new Map();
parameter.set("UnitsInStock", units);
queryCommand.setParameters(parameter);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var ProductName = reader.getValue(1, Number());
var UnitsAvailable = reader.getValue(2, Number());
}
}
else
{
}
string query = "SELECT Product where this.ProductID IN (?,?,?)";
ArrayList idList = new ArrayList();
idList.Add(1001);
idList.Add(100);
idList.Add(500);Hashtable values = new Hashtable();
values.Add("ProductID", idList);
ICacheReader reader = cache.ExecuteReader(query, values);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
Product result = (Product)reader.GetValue(1);
}
}
else
{
}
reader.Close();
Using the SQL LIKE Operator
NCache allows you to query for a specific pattern in a column through SQL-like query format using the LIKE operator.
The two wildcards used with the LIKE operator are:
*
: Used as a substitute for zero or more characters in the string.
?
: Used as a substitute for a single character in the string.
Here is an example that searches the cache and retrieves ProductName and UnitPrice using ExecuteReader
.
string query = "SELECT ProductName, UnitPrice From FQN.Product WHERE ProductName LIKE ?";
var queryCommand = new QueryCommand(query);
queryCommand.Parameters.Add("ProductName", "choco*");
ICacheReader reader = cache.SearchService.ExecuteReader(queryCommand);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
string productName = reader.GetValue<string>("ProductName");
double unitPrice = reader.GetValue<double>("UnitPrice");
Console.WriteLine($"Product: {productName}, Unit Price: {unitPrice}");
}
}
else
{
Console.WriteLine($"No product found having a name that starts with 'choco'.");
}
String query = "SELECT ProductName, UnitPrice FROM FQN.Product WHERE ProductName LIKE ?";
QueryCommand queryCommand = new QueryCommand(query);
HashMap<String, Object> parameters = new HashMap<>();
parameters.put("ProductName", "choco*");
queryCommand.getParameters().putAll(parameters);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
String productName = cacheReader.getValue("ProductName", String.class);
Double unitPrice = cacheReader.getValue("UnitPrice", Double.class);
System.out.println("Product: " + productName + ", Unit Price: " + unitPrice);
}
} else {
System.out.println("No product found having a name that starts with 'choco'.");
}
query = "SELECT * FROM FQN.Product WHERE product_name LIKE ? AND category LIKE ?"
query_command = ncache.QueryCommand(query)
parameter1 = {"product_name": "Shamp*"}
parameter2 = {"category": "Househo*"}
query_command.set_parameters(parameter1)
query_command.set_parameters(parameter2)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
result = reader.get_value(str, columnname="product_name")
print(result)
else:
print("Query result is None")
var query = "SELECT * FROM FQN.Product WHERE ProductName LIKE ? AND Category LIKE ?";
var queryCommand = new ncache.QueryCommand(query);
let parameter1 = new Map();
parameter1.set("ProductName", "P*");
queryCommand.setParameters(parameter1);
let parameter2 = new Map();
parameter2.set("Category", " Beverage?");
queryCommand.setParameters(parameter2);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var result = reader.getValue(1, Number());
}
}
else
{
}
string query = "SELECT Product where this.ProductName LIKE ?";
ArrayList list = new ArrayList();
list.Add("Ch*");
Hashtable values = new Hashtable();
values.Add("ProductName", list);
ICacheReader reader = cache.ExecuteReader(query, values);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
Product result = (Product)reader.GetValue(1);
}
}
else
{
}
reader.Close();
Using the SQL GROUP BY Operator
NCache provides you with the ability to group data according to a given criteria through this SQL-like query format using the GROUP BY clause.
Important
Note that the GROUP BY clause cannot be used without an Aggregate function.
The following example retrieves all the products based on their UnitPrice and groups them according to their Category and Count.
string query = "SELECT Category, COUNT(*) FROM FQN.Product WHERE UnitPrice < ? Group By Category";
var queryCommand = new QueryCommand(query);
queryCommand.Parameters.Add("UnitPrice", 100.0);
ICacheReader reader = cache.SearchService.ExecuteReader(queryCommand);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
string category = reader.GetValue<string>("Category");
int count = reader.GetValue<int>("COUNT()");
Console.WriteLine($"Category '{category}' has '{count}' affordable products.");
}
}
else
{
Console.WriteLine($"No category contains affordable products.");
}
String query = "SELECT Category, COUNT(*) FROM FQN.Product WHERE UnitPrice < ? GROUP BY Category";
QueryCommand queryCommand = new QueryCommand(query);
HashMap<String, Object> parameters = new HashMap<>();
parameters.put("UnitPrice", 100.0);
queryCommand.getParameters().putAll(parameters);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
String productName = cacheReader.getValue("Category", String.class);
Double unitPrice = cacheReader.getValue("COUNT(*)", Double.class);
System.out.println("Category '" + category + "' has '" + count + "' affordable products.");
}
} else {
System.out.println("No category contains affordable products.");
}
query = "SELECT category, COUNT(*) FROM FQN.Product WHERE unit_price > 5 Group By category"
query_command = ncache.QueryCommand(query)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
result = reader.get_value(int, 1)
category = reader.get_value(str, 0)
print("Count: " + str(result))
print("Category: " + str(category))
else:
print("Query result is None")
var query = "SELECT Category, COUNT(*) FROM FQN.Product WHERE UnitPrice > ? Group By Category";
var queryCommand = new ncache.QueryCommand(query);
let parameter = new Map();
parameter.set("UnitPrice", 10);
queryCommand.setParameters(parameter);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var result = reader.getValue(1, ncache.JsonDataType.Object);
var category = reader.getValue(0, Number());
}
}
else
{
}
string query = "SELECT this.Category, COUNT(Product) WHERE this.ProductID > ? GROUP BY this.Category";
Hashtable values = new Hashtable();
values.Add("ProductID", 5);
ICacheReader reader = cache.ExecuteReader(query, values);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
object category = reader.GetOrdinal("Category"); object result = reader.GetValue(1);
}
}
else
{
}
reader.Close();
Using the SQL ORDER BY Operator
NCache provides you with the ability to sort the data in ascending or descending order, according to the given criteria, through SQL-like query format using the ORDER BY clause. The following example retrieves ProductName and UnitPrice, where ProductName is greater than 100 and sorts them in descending order using ExecuteReader
.
string query = "SELECT ProductName,UnitPrice FROM FQN.Product WHERE UnitPrice < ? ORDER BY UnitPrice DESC";
var queryCommand = new QueryCommand(query);
queryCommand.Parameters.Add("UnitPrice", 100.0);
ICacheReader reader = cache.SearchService.ExecuteReader(queryCommand);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
string productName = reader.GetValue<string>("ProductName");
string unitPrice = reader.GetValue<string>("UnitPrice");
Console.WriteLine($"Product: {productName}, Unit Price: {unitPrice}");
}
}
else
{
Console.WriteLine($"No products found which are cheaper than 100.");
}
String query = "SELECT ProductName, UnitPrice FROM FQN.Product WHERE UnitPrice < ? ORDER BY UnitPrice DESC";
QueryCommand queryCommand = new QueryCommand(query);
HashMap<String, Object> parameters = new HashMap<>();
parameters.put("UnitPrice", 100.0);
queryCommand.getParameters().putAll(parameters);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
String productName = cacheReader.getValue("ProductName", String.class);
Double unitPrice = cacheReader.getValue("UnitPrice", Double.class);
System.out.println("Product: " + productName + ", Unit Price: " + unitPrice);
}
} else {
System.out.println("No products found which are cheaper than 100.");
}
query = "SELECT * FROM FQN.Product WHERE unit_price > 100 ORDER BY units_in_stock DESC"
query_command = ncache.QueryCommand(query)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
result = reader.get_value(int, columnname="units_in_stock")
print("Units: " + str(result))
else:
print("Query result is None")
var query = "SELECT * FROM FQN.Product WHERE UnitPrice > ? ORDER BY UnitsInStock DESC";
var queryCommand = new ncache.QueryCommand(query);
let parameter = new Map();
parameter.set("UnitPrice", 100);
queryCommand.setParameters(parameter);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var result = reader.getValue(1, Number());
}
}
else
{
}
string query = "SELECT Product ORDER BY by this.ProductID = ?";
string query = "SELECT Product WHERE this.Category = ? ORDER BY this.Category";
Hashtable values = new Hashtable();
values.Add("ProductID", 1001);
ICacheReader reader = cache.ExecuteReader(query, values);
if (reader.FieldCount > 0)
{
while (reader.Read())
{ Product result = (Product)reader.GetValue(1);
}
}
else
{
}
reader.Close();
Using the SQL DateTime Operator
NCache provides you with ease of searching the cache by providing parameters in various formats. You can use many functions in a single query for faster cache searching. DateTime
can be used in your query to search the cache concerning a particular date or time. For more details on date formats, please refer to the Microsoft DateTime Struct Docs.
Note
Make sure that you specify the date in the correct format.
The following example executes a query that searches for all the orders in the cache whose OrderDate is as specified using ExecuteReader
.
string query = "SELECT OrderID FROM FQN.Order WHERE OrderDate = ?";
var queryCommand = new QueryCommand(query);
DateTime orderDate = new DateTime(2022, 01, 01);
queryCommand.Parameters.Add("OrderDate", orderDate);
ICacheReader reader = cache.SearchService.ExecuteReader(queryCommand);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
int orderID = reader.GetValue<int>("OrderID");
Console.WriteLine($"Order: {orderID} was placed on 1st January, 2022.");
}
}
else
{
Console.WriteLine($"No orders were placed on 1st January 2022.");
}
String query = "SELECT OrderID FROM FQN.Order WHERE OrderDate = ?";
QueryCommand queryCommand = new QueryCommand(query);
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date orderDate;
orderDate = sdf.parse("2022-01-01");
HashMap<String, Object> parameters = new HashMap<>();
parameters.put("OrderDate", orderDate);
queryCommand.getParameters().putAll(parameters);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
int orderID = cacheReader.getInt(cacheReader.getOrdinal("OrderID"));
System.out.println("Order: " + orderID + " was placed on 1st January, 2022.");
}
} else {
System.out.println("No orders were placed on 1st January 2022.");
}
query = "SELECT * FROM FQN.Order WHERE order_date < ?"
query_command = ncache.QueryCommand(query)
parameter = {"order_date": datetime.strptime('Jan 1 2022 1:33PM', '%b %d %Y %I:%M%p')}
query_command.set_parameters(parameter)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
order_id = reader.get_value(int, 1)
print("Order ID: " + str(order_id))
else:
print("Query result is None")
var query = "SELECT * FROM FQN.Order WHERE OrderDate < ?";
var queryCommand = new ncache.QueryCommand(query);
var date = new Date("2021-02-10");
let map = new Map();
var parameters = map.set("orderDate", date);
queryCommand.setParameters(parameters);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var result = reader.getValue(1, Number());
}
}
else
{
}
Using the SQL Logical Operator
NCache provides the user with the AND, OR, and NOT logical operators to specify more than one condition after the operator in these queries through this SQL-like query format. Here's an example showing how to use the combination of AND and OR operators in a single query within the WHERE clause using ExecuteReader
.
string query = "SELECT * FROM FQN.Product WHERE (Category = ? AND UnitsInStock > ?) OR (Category = ? AND UnitPrice > ?)";
var queryCommand = new QueryCommand(query);
ArrayList categoryList = new ArrayList();
categoryList.Add("Beverages");
categoryList.Add("Produce");
queryCommand.Parameters.Add("Category", categoryList);
queryCommand.Parameters.Add("UnitsInStock", 100);
queryCommand.Parameters.Add("UnitPrice", 100);
ICacheReader reader = _cache.SearchService.ExecuteReader(queryCommand);
if (reader.FieldCount > 0)
{
while (reader.Read())
{
string result = reader.GetValue<string>("ProductID");
}
}
else
{
}
catch (OperationFailedException ex)
if (ex.ErrorCode == NCacheErrorCodes.INCORRECT_FORMAT)
{
}
else
{
}
String query = "SELECT * FROM FQN.Product WHERE (Category = ? AND UnitsInStock > ?) OR (Category = ? AND UnitPrice > ?)";
QueryCommand queryCommand = new QueryCommand(query);
ArrayList<String> categoryList = new ArrayList<>();
categoryList.add("Beverages");
categoryList.add("Produce");
HashMap<String, Object> parameters = new HashMap<>();
parameters.put("Category1", categoryList.get(0));
parameters.put("Category2", categoryList.get(1));
parameters.put("UnitsInStock", 100);
parameters.put("UnitPrice", 100.0);
queryCommand.getParameters().putAll(parameters);
CacheReader cacheReader = cache.getSearchService().executeReader(queryCommand);
if (cacheReader.getFieldCount() > 0) {
while (cacheReader.read()) {
String result = cacheReader.getString(cacheReader.getOrdinal("ProductID"));
}
} else {
}
query = "SELECT * FROM FQN.Product WHERE (category = 'Beverages' AND units_in_stock > 0) OR (category = 'Produce' AND unit_price > 100)"
query_command = ncache.QueryCommand(query)
search_service = cache.get_search_service()
reader = search_service.execute_reader(query_command)
if reader.get_field_count() > 0:
while reader.read():
product_id = reader.get_value(str, 1)
print("Product ID: " + str(product_id))
else:
print("Query result is None")
var query = "SELECT * FROM FQN.Product WHERE (Category = ? AND UnitsInStock > ?) OR (Category = ? AND UnitPrice > ?)";
var queryCommand = new ncache.QueryCommand(query);
let categoryList = Array();
categoryList.push("Produce");
categoryList.push("Beverages");
let parameter = new Map();
parameter.set("Category", categories);
parameter.set("UnitsInStock", 100);
parameter.set("UnitPrice", 100);
queryCommand.setParameters(parameter);
var searchService = await this.cache.getSearchService();
var reader = await searchService.executeReader(queryCommand);
if (reader.getFieldCount() > 0)
{
while (reader.read())
{
var result = reader.getValue(1, Number());
}
}
else
{
}
Using ExecuteScalar to Search Data in Cache
The ExecuteScalar
executes user queries and returns the first column of the first row in the result set, any additional columns or rows are ignored.
string query = "SELECT AVG(UnitPrice) FROM FQN.Product WHERE Category = ?";
var queryCommand = new QueryCommand(query);
queryCommand.Parameters.Add("Category", "Beverages");
decimal average = (decimal)cache.SearchService.ExecuteScalar(queryCommand);
Console.WriteLine($"Average price of Beverages is {average}.");
String query = "SELECT AVG(UnitPrice) FROM FQN.Product WHERE Category = ?";
QueryCommand queryCommand = new QueryCommand(query);
queryCommand.getParameters().put("Category", "Beverages");
Double average = cache.getSearchService().executeScalar(queryCommand, Double.class);
if (average != null) {
System.out.println("Average price of Beverages is " + average + ".");
} else {
System.out.println("No data found for Beverages.");
}
Additional Resources
NCache provides sample application for SQL Queries on GitHub.
See Also
.NET: Alachisoft.NCache.Client.Services namespace.
Java: com.alachisoft.ncache.runtime.caching namespace.
Python: ncache.client.services class.
Node.js: Cache class.