LanceDB’s multivector support enables you to store and search multiple vector embeddings for a single item.
This capability is particularly valuable when working with late interaction models like ColBERT and ColPaLi that generate multiple embeddings per document.
In this tutorial, you’ll create a table with multiple vector embeddings per document and learn how to perform multivector search. For all the code - open in Colab
Each item in your dataset can have a column containing multiple vectors, which LanceDB can efficiently index and search. When performing a search, you can query using either a single vector embedding or multiple vector embeddings.
LanceDB also integrates with ConteXtualized Token Retriever (XTR) , an advanced retrieval model that prioritizes the most semantically important document tokens during search. This integration enhances the quality of search results by focusing on the most relevant token matches.
cosine metric is supported for multivector search.float16, float32, or float64.MaxSim (Maximum Similarity) is a key concept in late interaction models that:
The MaxSim calculation can be expressed as:
\[ \text{MaxSim}(Q, D) = \sum_{i=1}^{|Q|} \max_{j=1}^{|D|} \text{sim}(q_i, d_j) \]where sim is the similarity function (e.g. cosine), \(Q = \{q_1, q_2, ..., q_{|Q|}\}\) represents the query vectors, and \(D = \{d_1, d_2, ..., d_{|D|}\}\) represents the document vectors.
cosine metric for multivector search.
The vector value type can be float16, float32 or float64.
Connect to LanceDB and import required libraries for data management.
import lancedb
import numpy as np
import pyarrow as pa
db = lancedb.connect(
uri="db://your-project-slug",
api_key="your-api-key",
region="your-cloud-region"
)Define a schema that specifies a multivector field. A multivector field is a nested list structure where each document contains multiple vectors. In this case, we’ll create a schema with:
db = lancedb.connect("data/multivector_demo")
schema = pa.schema(
[
pa.field("id", pa.int64()),
# float16, float32, and float64 are supported
pa.field("vector", pa.list_(pa.list_(pa.float32(), 256))),
]
)Generate sample data where each document contains multiple vector embeddings, which could represent different aspects or views of the same document.
In this example, we create 1024 documents where each document has 2 random vectors of dimension 256, simulating a real-world scenario where you might have multiple embeddings per item.
data = [
{
"id": i,
"vector": np.random.random(size=(2, 256)).tolist(), # Each document has 2 vectors
}
for i in range(1024)
]Create a table with the defined schema and sample data, which will store multiple vectors per document for similarity search.
tbl = db.create_table("multivector_example", data=data, schema=schema)Only cosine similarity is supported as the distance metric for multivector search operations.
For faster search, build the standard IVF_PQ index over your vectors:
tbl.create_index(metric="cosine", vector_column_name="vector")When searching with a single query vector, it will be compared against all vectors in each document, and the similarity scores will be aggregated to find the most relevant documents.
query = np.random.random(256)
results_single = tbl.search(query).limit(5).to_pandas()With multiple vector queries, LanceDB calculates similarity using late interaction - a technique that computes relevance by finding the best matching pairs between query and document vectors. This approach provides more nuanced matching while maintaining fast retrieval speeds.
query_multi = np.random.random(size=(2, 256))
results_multi = tbl.search(query_multi).limit(5).to_pandas()If you still need more guidance, you can try the complete Multivector Search Notebook .