description |
---|
Discover ObjectBox: The Lightning-Fast Mobile Database for Persistent Object Storage. Streamline Your Workflow, Eliminate Repetitive Tasks, and Enjoy a User-Friendly Data Interface. |
{% tabs %} {% tab title="Java/Kotlin (Android)" %} {% embed url="https://www.youtube.com/watch?v=flmAeYY-u9I" %} Video Tutorial on Getting Started with ObjectBox for Android and Java {% endembed %}
You can get ObjectBox from the Central repository. To add ObjectBox to your Android project, follow these steps:
- Open the Gradle build file of your root project (not the ones for your app or module subprojects) and add a global variable for the version and the ObjectBox Gradle plugin:
{% code title="/build.gradle(.kts)" %}
buildscript {
ext.objectboxVersion = "4.0.3" // For Groovy build scripts
// val objectboxVersion by extra("4.0.3") // For KTS build scripts
repositories {
mavenCentral()
}
dependencies {
// Android Gradle Plugin 4.1.0 or later supported
classpath("com.android.tools.build:gradle:8.1.0")
classpath("io.objectbox:objectbox-gradle-plugin:$objectboxVersion")
}
}
{% endcode %}
- Open the Gradle build file for your app or module subproject and, after the
com.android.application
plugin, apply theio.objectbox
plugin:
{% code title="/app/build.gradle(.kts)" %}
// Using plugins syntax:
plugins {
id("com.android.application")
id("kotlin-android") // Only for Kotlin projects
id("kotlin-kapt") // Only for Kotlin projects
id("io.objectbox") // Apply last
}
// Or using the old apply syntax:
apply plugin: "com.android.application"
apply plugin: "kotlin-android" // Only for Kotlin projects
apply plugin: "kotlin-kapt" // Only for Kotlin projects
apply plugin: "io.objectbox" // Apply last
{% endcode %}
{% hint style="info" %} If you encounter any problems in this or later steps, check out the FAQ and Troubleshooting pages. {% endhint %}
- Then do "Sync Project with Gradle Files" in Android Studio so the Gradle plugin automatically adds the required ObjectBox libraries and code generation tasks.
The ObjectBox plugin uses reasonable defaults and detects most configurations automatically. However, if needed you can configure the model file path, the MyObjectBox package, enable debug mode and more using advanced setup options. {% endtab %}
{% tab title="Java/Kotlin (JVM)" %} ObjectBox for Java supports JVM (Linux, macOS, Windows) Java or Kotlin projects.
It ships with a Gradle plugin. To apply it, your project needs to use Gradle as the build system. The instructions assume the recommended multi-project directory structure is used.
{% hint style="info" %} There is an experimental Maven plugin available. See the Java Maven example. We welcome your feedback if supporting Maven is of interest to you. {% endhint %}
ObjectBox is available from the Maven Central repository.
To add the ObjectBox plugin:
- Open the Gradle build file of your root project and add a global variable for the version and the ObjectBox Gradle plugin:
{% code title="/build.gradle(.kts)" %}
buildscript {
ext.objectboxVersion = "4.0.3" // For Groovy build scripts
// val objectboxVersion by extra("4.0.3") // For KTS build scripts
repositories {
mavenCentral()
}
dependencies {
classpath("io.objectbox:objectbox-gradle-plugin:$objectboxVersion")
}
}
{% endcode %}
- Open the Gradle build file for your subproject and, after other plugins, apply the
io.objectbox
plugin:
{% code title="/app/build.gradle(.kts)" %}
// Using plugins syntax:
plugins {
id("java-library") // or org.jetbrains.kotlin.jvm for Kotlin projects.
id("io.objectbox") // Apply last.
}
// Or using the old apply syntax:
apply plugin: "java-library" // or org.jetbrains.kotlin.jvm for Kotlin projects.
apply plugin: "io.objectbox" // Apply last.
{% endcode %}
{% hint style="info" %} Using your IDE of choice with a Gradle project might require additional configuration. E.g.
- For IntelliJ IDEA see the help page for Gradle. Also make sure to use version 2019.1 or newer as it has improved Gradle support, like delegating build actions to Gradle.
- For Eclipse see the Buildship project and Getting Started article. {% endhint %}
ObjectBox is an object database running mostly in native code written in C/C++ for optimal performance. Thus, ObjectBox will load a native library: a “.dll” on Windows, a “.so” on Linux, and a “.dylib” on macOS. By default, the ObjectBox Gradle plugin adds a dependency to the native library matching your system. This means that your app is already set up to run on your system.
{% hint style="info" %} On Windows you might have to install the latest Microsoft Visual C++ Redistributable package (X64) to use the ObjectBox DLL. {% endhint %}
{% hint style="info" %} ObjectBox binaries are built for 64-bit systems for best performance. Talk to us if you require 32-bit support. {% endhint %}
To add native libraries for all platforms that your app supports and additional configuration options (configure the model file path, the MyObjectBox package, enable debug mode) see:
{% content-ref url="advanced/advanced-setup.md" %} advanced-setup.md {% endcontent-ref %}
For examples and how to write unit tests see:
{% content-ref url="java-desktop-apps.md" %} java-desktop-apps.md {% endcontent-ref %} {% endtab %}
{% tab title="Flutter" %} {% embed url="https://www.youtube.com/watch?v=AxYbdriXKI8" %} Video Tutorial on Getting Started with ObjectBox for Flutter {% endembed %}
{% hint style="info" %} You can watch these video tutorials as well 😀:
To add ObjectBox to your Flutter project:
- Run these commands:
flutter pub add objectbox objectbox_flutter_libs:any
flutter pub add --dev build_runner objectbox_generator:any
# Or when using ObjectBox Sync instead run:
flutter pub add objectbox objectbox_sync_flutter_libs:any
flutter pub add --dev build_runner objectbox_generator:any
{% hint style="info" %} To run unit tests on your machine, download the latest native ObjectBox library for your machine by running this script in a bash shell (e.g. Git Bash on Windows):
bash <(curl -s https://raw.githubusercontent.com/objectbox/objectbox-dart/main/install.sh)
For ObjectBox Sync, append the --sync
argument to above command.
{% endhint %}
- This should add lines like this to your
pubspec.yaml
:
dependencies:
objectbox: ^4.0.3
objectbox_flutter_libs: any
# For ObjectBox Sync this dependency should appear instead:
# objectbox_sync_flutter_libs: any
dev_dependencies:
build_runner: ^2.0.0
objectbox_generator: any
- If you added the above lines manually, then install the packages with
flutter pub get
.
{% hint style="info" %} For Android increase the NDK version:
{% code title="/android/app/build.gradle" %}
android {
// ObjectBox: Flutter defaults to NDK 23.1.7779620, but
// - objectbox_flutter_libs requires Android NDK 25.1.8937393
// - path_provider_android requires Android NDK 25.1.8937393
// Until Flutter uses a newer version (https://github.com/flutter/flutter/commit/919bed6e0a18bd5b76fb581ede10121f8c14a6f7)
// manually set the required one:
// ndkVersion flutter.ndkVersion
ndkVersion = "25.1.8937393"
}
{% endcode %} {% endhint %}
{% hint style="info" %}
For all macOS apps need to target macOS 10.15: in Podfile
change the platform and in the Runner.xcodeproj/poject.pbxproj
file update MACOSX_DEPLOYMENT_TARGET
.
For sandboxed macOS apps specify an application group. Check all macos/Runner/*.entitlements
files if they contain a <dict>
section with the group ID. If necessary, change the string value to the DEVELOPMENT_TEAM
you can find in your Xcode settings, plus an application-specific suffix. Due to macOS restrictions the complete string must be 19 characters or shorter. For example:
{% code title="macos/Runner/*.entitlements" %}
<dict>
<key>com.apple.security.application-groups</key>
<array>
<string>FGDTDLOBXDJ.demo</string>
</array>
...
</dict>
{% endcode %}
Then, in your app code, pass the same string when opening the Store. For example: openStore(macosApplicationGroup: 'FGDTDLOBXDJ.demo')
.
{% endhint %}
{% hint style="info" %} For Linux Desktop apps: the Flutter snap ships with an outdated version of CMake. Install Flutter manually instead to use the version of CMake installed on your system. {% endhint %}
{% hint style="info" %} For Android using Flutter 3.19 or older and the ObjectBox Sync-enabled library: increase minSdkVersion to at least 21.
{% code title="/android/app/build.gradle" %}
android {
defaultConfig {
// ObjectBox Sync requires at least SDK 21 (Android 5.0)
minSdkVersion 21
}
}
{% endcode %} {% endhint %}
{% hint style="info" %}
For iOS using Flutter 3.0 or older: increase the deployment target in Xcode to iOS 12 and, under Architectures, replace ${ARCHS_STANDARD}
with arm64
(or $ARCHS_STANDARD_64_BIT
).
{% endhint %}
{% endtab %}
{% tab title="Dart Native" %}
- Run these commands:
dart pub add objectbox
dart pub add --dev build_runner objectbox_generator:any
- This should add lines like this to your
pubspec.yaml
:
dependencies:
objectbox: ^4.0.3
dev_dependencies:
build_runner: ^2.0.0
objectbox_generator: any
- If you added the above lines manually, then install the packages with
dart pub get
- Install the ObjectBox C library for your system (on Windows you can use "Git Bash"):
bash <(curl -s https://raw.githubusercontent.com/objectbox/objectbox-dart/main/install.sh)
# Or when using ObjectBox Sync instead run:
bash <(curl -s https://raw.githubusercontent.com/objectbox/objectbox-dart/main/install.sh) --sync
{% hint style="info" %}
By default the library is downloaded into the lib
subdirectory of the working directory. It's not necessary to install the library system-wide. This also allows to use different versions for different projects. For details see below.
{% endhint %}
Deploying Dart Native projects
Natively compiled Dart applications that use ObjectBox Dart require a reference to the objectbox-c library. Hence, the shared library file downloaded with install.sh
needs to be shipped with the executable.
The install.sh
script downloads the library by default to the lib
subdirectory of the working directory. An executable using ObjectBox Dart looks for the library in this lib
directory.
If it is not found there, it falls back to using system directories (using Dart's DynamicLibrary.open
):
- Windows: working directory and
%WINDIR%\system32
. - macOS:
/usr/local/lib
(and maybe others). - Linux:
/lib
and/usr/lib
(again, possibly others). {% endtab %}
{% tab title="Python" %}
ObjectBox for Python is available via PyPI:
Stable Version (4.0.0):
pip install --upgrade objectbox
{% endtab %} {% endtabs %}
Define your model by adding an @Entity
(internal name for database objects) annotation to at least one class and an @Id
annotation to one of the class variables. Learn more about the ObjectBox model here.
A simple entity representing a user could look like this:
Important:
- Entities must have one ID property of type
long
(orLong
in Kotlin,int
in Dart). If you need to use other types, like a String ID, see the @Id annotation docs. Also, it must have non-private visibility (or non-private getter and setter methods). - Entities must also have a no-args constructor, or for better performance, a constructor with all properties as arguments. In the above example, a default, no-args constructor is generated by the compiler.
Support for many property types is already built-in, but almost any type can be stored with a converter.
{% hint style="info" %} For a deeper explanation and a look at all other available annotations (e.g. for relations and indexes) check the Entity Annotations page. {% endhint %}
Next, we generate some binding code based on the model defined in the previous step.
{% tabs %} {% tab title="Java/Kotlin" %} Build your project to generate the classes required to use ObjectBox, for example using Build > Make Project in Android Studio.
{% hint style="info" %} Note: If you make significant changes to your entities, e.g. by moving them or modifying annotations, make sure to rebuild the project so generated ObjectBox code is updated. {% endhint %} {% endtab %}
{% tab title="Flutter/Dart Native" %} To generate the binding code required to use ObjectBox run
dart run build_runner build
ObjectBox generator will look for all @Entity
annotations in your lib
folder and create
- a single database definition
lib/objectbox-model.json
and - supporting code in
lib/objectbox.g.dart
.
To customize the directory where generated files are written see advanced-setup.md.
{% hint style="info" %} If you make changes to your entities, e.g. by adding a property or modifying annotations, or after the ObjectBox library has updated make sure to re-run the generator so generated ObjectBox code is updated. {% endhint %}
{% hint style="success" %}
You typically commit the generated code file objectbox.g.dart
to your version control system (e.g. git) to avoid having to re-run the generator unless there are changes.
{% endhint %}
{% hint style="info" %}
Actually we lied above. The generator will process lib
and test
folders separately and generate files for each one (if @Entity
classes exist there). This allows to create a separate test database that does not share any of the entity classes with the main database.
{% endhint %}
{% endtab %}
{% tab title="Python" %}
{% hint style="info" %}
Python bindings offer a convenient default Model to which Entity definitions are automatically associated if not specified otherwise.
Similar to the other bindings, a JSON model file is also used for management of Schema history (i.e. to handle add/remove/rename of Entity and Property).
{% endhint %}
{% endtab %}
{% endtabs %}
Among other files ObjectBox generates a JSON model file, by default to
app/objectbox-models/default.json
for Android projects,lib/objectbox-model.json
for Dart/Flutter projects, or<user-module-dir>/objectbox-model.json
for Python projects
{% hint style="info" %}
In Android Studio you might have to switch the Project view from Android to Project to see the default.json
model file.
Python checks for the call-stack to determine the user-module directory in which the JSON file is stored.
{% endhint %}
This JSON file changes when you change your entity classes (or sometimes with a new version of ObjectBox).
Keep this JSON file, commit the changes to version control!
This file keeps track of unique IDs assigned to your entities and properties. This ensures that an older version of your database can be smoothly upgraded if your entities or properties change.
{% hint style="success" %} The model file also enables you to keep data when renaming entities or properties or to resolve conflicts when two of your developers make changes at the same time. {% endhint %}
BoxStore (Java) or Store (Dart) is the entry point for using ObjectBox. It is the direct interface to the database and manages Boxes. Typically, you want to only have a single Store (single database) and keep it open while your app is running, not closing it explicitly.
{% tabs %}
{% tab title="Java (Android)" %}
Create it using the builder returned by the generated MyObjectBox
class, for example in a small helper class like this:
public class ObjectBox {
private static BoxStore store;
public static void init(Context context) {
store = MyObjectBox.builder()
.androidContext(context)
.build();
}
public static BoxStore get() { return store; }
}
{% hint style="warning" %}
If you encounter UnsatisfiedLinkError
or LinkageError
on the build call, see App Bundle, split APKs and Multidex for solutions.
{% endhint %}
The best time to initialize ObjectBox is when your app starts. We suggest to do it in the onCreate
method of your Application class:
public class ExampleApp extends Application {
@Override
public void onCreate() {
super.onCreate();
ObjectBox.init(this);
}
}
{% endtab %}
{% tab title="Kotlin (Android)" %} Create it using the builder returned by the generated MyObjectBox class, for example in a small helper class like this:
object ObjectBox {
lateinit var store: BoxStore
private set
fun init(context: Context) {
store = MyObjectBox.builder()
.androidContext(context)
.build()
}
}
{% hint style="warning" %}
If you encounter UnsatisfiedLinkError
or LinkageError
on the build call, see App Bundle, split APKs and Multidex for solutions.
{% endhint %}
The best time to initialize ObjectBox is when your app starts. We suggest to do it in the onCreate
method of your Application class:
class ExampleApp : Application() {
override fun onCreate() {
super.onCreate()
ObjectBox.init(this)
}
}
{% endtab %}
{% tab title="JVM" %}
public class ObjectBox {
private static BoxStore store;
public static void init(Context context) {
store = MyObjectBox.builder()
.name("objectbox-notes-db")
.build();
}
public static BoxStore get() { return store; }
}
The best time to initialize ObjectBox is when your app starts. For a command line app this is typically inside the main method. {% endtab %}
{% tab title="Flutter" %}
Create it using the generated openStore()
method, for example in a small helper class like this:
import 'package:path/path.dart' as p;
import 'package:path_provider/path_provider.dart';
import 'objectbox.g.dart'; // created by `flutter pub run build_runner build`
class ObjectBox {
/// The Store of this app.
late final Store store;
ObjectBox._create(this.store) {
// Add any additional setup code, e.g. build queries.
}
/// Create an instance of ObjectBox to use throughout the app.
static Future<ObjectBox> create() async {
final docsDir = await getApplicationDocumentsDirectory();
// Future<Store> openStore() {...} is defined in the generated objectbox.g.dart
final store = await openStore(directory: p.join(docsDir.path, "obx-example"));
return ObjectBox._create(store);
}
}
The best time to initialize ObjectBox is when your app starts. We suggest to do it in your app's main()
function:
/// Provides access to the ObjectBox Store throughout the app.
late ObjectBox objectbox;
Future<void> main() async {
// This is required so ObjectBox can get the application directory
// to store the database in.
WidgetsFlutterBinding.ensureInitialized();
objectbox = await ObjectBox.create();
runApp(MyApp());
}
{% hint style="info" %}
On mobile devices or sandboxed apps data should be stored in the app's documents directory. See Flutter: read & write files for more info. This is exactly what openStore()
does, if the directory
argument is not specified.
On desktop systems it is recommended to specify a directory
to create a custom sub-directory to avoid conflicts with other apps.
If your code passes a directory that the application can't write to, you get an error that looks somewhat like this: failed to create store: 10199 Dir does not exist: objectbox (30)
.
{% endhint %}
{% hint style="info" %} When using Dart isolates, note that each Dart isolate has its own global fields, they do not share state on the Dart level.
However, as ObjectBox runs on the native or process level (so one native instance shared across all isolates), instead of creating a new Store in another isolate your code should instead attach to the open native store. {% endhint %} {% endtab %}
{% tab title="Dart Native" %}
Create it using the generated openStore()
method, for example like this:
import 'objectbox.g.dart'; // created by `dart pub run build_runner build`
void main() {
// Store openStore() {...} is defined in the generated objectbox.g.dart
final store = openStore();
// your app code ...
store.close(); // don't forget to close the store
}
The above minimal example omits the argument to (directory: )
, using the default - ./objectbox
- in the current working directory.
{% hint style="info" %} When using Dart isolates, note that each Dart isolate has its own global fields, they do not share state on the Dart level.
However, as ObjectBox runs on the native or process level (so one native instance shared across all isolates), instead of creating a new Store in another isolate your code should instead attach to the open native store. {% endhint %} {% endtab %}
{% tab title="Python" %}
from objectbox import Store
store = Store()
{% endtab %} {% endtabs %}
It is possible to specify various options when building a store. Notably for testing or caching, to use an in-memory database that does not create any files:
{% tabs %} {% tab title="Java" %}
BoxStore inMemoryStore = MyObjectBox.builder()
.androidContext(context)
.inMemory("test-db")
.build();
{% endtab %}
{% tab title="Dart" %}
final inMemoryStore =
Store(getObjectBoxModel(), directory: "memory:test-db");
{% endtab %}
{% tab title="Python" %}
store = Store(directory="memory:testdata")
{% endtab %} {% endtabs %}
For more store configuration options: for Java see the BoxStoreBuilder and for Dart the Store documentation. (Python APIs will be published soon)
The Box class is likely the class you interact with most. A Box instance gives you access to objects of a particular type. For example, if you have User
and Order
entities, you need a Box object to interact with each:
{% tabs %} {% tab title="Java" %}
Box<User> userBox = store.boxFor(User.class);
Box<Order> orderBox = store.boxFor(Order.class);
{% endtab %}
{% tab title="Kotlin" %}
val userBox = store.boxFor(User::class)
val orderBox = store.boxFor(Order::class)
{% endtab %}
{% tab title="Dart" %}
final userBox = store.box<User>();
final orderBox = store.box<Order>();
{% endtab %}
{% tab title="Python" %}
user_box = store.box(User)
order_box = store.box(Order)
{% endtab %} {% endtabs %}
These are some of the operations offered by the Box class:
put inserts a new object or updates an existing one (with the same ID). When inserting, an ID will be assigned to the just inserted object (this will be explained below) and returned. put
also supports putting multiple objects, which is more efficient.
{% tabs %} {% tab title="Java" %}
User user = new User("Tina");
userBox.put(user);
List<User> users = getNewUsers();
userBox.put(users);
{% endtab %}
{% tab title="Kotlin" %}
val user = User(name = "Tina")
userBox.put(user)
val users: List<User> = getNewUsers()
userBox.put(users)
{% endtab %}
{% tab title="Dart" %}
final user = User(name: 'Tina');
userBox.put(user);
final users = getNewUsers();
userBox.putMany(users);
{% endtab %}
{% tab title="Python" %}
user = User(name="Tina")
user_box.put(user)
users = get_new_users()
user_box.put(*users)
{% endtab %} {% endtabs %}
get and getAll: Given an object’s ID, get
reads it from its box. To get all objects in the box use getAll
.
{% tabs %} {% tab title="Java" %}
User user = userBox.get(userId);
List<User> users = userBox.getAll();
{% endtab %}
{% tab title="Kotlin" %}
val user = userBox[userId]
val users = userBox.all
{% endtab %}
{% tab title="Dart" %}
final user = userBox.get(userId);
final users = userBox.getMany(userIds);
final users = userBox.getAll();
{% endtab %}
{% tab title="Python" %}
user = user_box.get(user_id)
users = user_box.get_all()
{% endtab %} {% endtabs %}
query: Starts building a query to return objects from the box that match certain conditions. See queries for details.
{% tabs %} {% tab title="Java" %}
Query<User> query = userBox
.query(User_.name.equal("Tom"))
.order(User_.name)
.build();
List<User> results = query.find();
query.close();
{% endtab %}
{% tab title="Kotlin" %}
val query = userBox
.query(User_.name.equal("Tom"))
.order(User_.name)
.build()
val results = query.find()
query.close()
{% endtab %}
{% tab title="Dart" %}
final query =
(userBox.query(User_.name.equals('Tom'))..order(User_.name)).build();
final results = query.find();
query.close();
{% endtab %}
{% tab title="Python" %}
query = user_box \
.query(User.name.equals('Tom')) \
.build()
results = query.find()
{% endtab %} {% endtabs %}
remove and removeAll: Remove a previously put object from its box (deletes it). remove
also supports removing multiple objects, which is more efficient. removeAll
removes (deletes) all objects in a box.
{% tabs %} {% tab title="Java" %}
boolean isRemoved = userBox.remove(userId);
userBox.remove(users);
// alternatively:
userBox.removeByIds(userIds);
userBox.removeAll();
{% endtab %}
{% tab title="Kotlin" %}
val isRemoved = userBox.remove(userId)
userBox.remove(users)
// alternatively:
userBox.removeByIds(userIds)
userBox.removeAll()
{% endtab %}
{% tab title="Dart" %}
final isRemoved = userBox.remove(userId);
userBox.removeMany(userIds);
userBox.removeAll();
{% endtab %}
{% tab title="Python" %}
is_removed = user_box.remove(user_id)
user_box.remove_all()
{% endtab %} {% endtabs %}
count: Returns the number of objects stored in this box.
{% tabs %} {% tab title="Java" %}
long userCount = userBox.count();
{% endtab %}
{% tab title="Kotlin" %}
val userCount = userBox.count()
{% endtab %}
{% tab title="Dart" %}
final userCount = userBox.count();
{% endtab %}
{% tab title="Python" %}
user_box.count()
{% endtab %} {% endtabs %}
For a complete list of methods available in the Box class, check the API reference documentation for Java or Dart.
ObjectBox has built-in support to run (typically multiple or larger) database operations asynchronously.
{% tabs %} {% tab title="Java" %} runInTxAsync and callInTxAsync: runs the given Runnable/Callable in a transaction on a background thread (the internal ObjectBox thread pool) and calls the given callback once done. In case of callInTxAsync the callback also receives the returned result.
store.callInTxAsync(() -> {
Box<User> box = store.boxFor(User.class);
String name = box.get(userId).name;
box.remove(userId);
return text;
}, (result, error) -> {
if (error != null) {
System.out.println("Failed to remove user with id " + userId);
} else {
System.out.println("Removed user with name: " + result);
}
});
awaitCallInTx (Kotlin Coroutines only): wraps callInTxAsync in a coroutine that suspends until the transaction has completed. Likewise, on success the return value of the given callable is returned, on failure an exception is thrown.
try {
val name = store.awaitCallInTx {
val box = store.boxFor(User::class.java)
val name = box.get(userId).name
box.remove(userId)
name
}
println("Removed user with name $name")
} catch (e: Exception) {
println("Failed to remove user with id $userId")
}
{% endtab %}
{% tab title="Dart" %} Most Box methods do have async versions which run the operation in a worker isolate.
For example putAsync: asynchronously inserts a new object or updates an existing one (with the same ID). The returned future completes when the object is successfully written to the database.
final user = User(name: 'Tina');
Future<int> idFuture = userBox.putAsync(user);
...
final id = await idFuture;
userBox.get(id); // after the future completed, the object is inserted
To run multiple operations, it is more efficient to wrap the synchronous calls in an asynchronous transaction with runInTransactionAsync (API reference): run a callback with multiple database operations within a write or read transaction in the background without blocking the user interface. Can return results.
// The callback must be a function that can be sent to an isolate:
// either a top-level function, static method or a closure that only
// captures objects that can be sent to an isolate.
String? readNameAndRemove(Store store, int objectId) {
var box = store.box<User>();
final nameOrNull = box.get(objectId)?.name;
box.remove(objectId);
return nameOrNull;
}
final nameOrNull =
await store.runInTransactionAsync(TxMode.write, readNameAndRemove, objectId);
There is also runAsync (API reference): like runInTransactionAsync but does not start a transaction, leaving that to your callback code. This allows to supply a callback that is an async function.
If it is necessary to call put many times in a row, take a look at putQueued: Schedules the given object to be put later on, by an asynchronous queue, returns the id immediately even though the object may not have been written yet. You can use Store's awaitQueueCompletion()
or awaitQueueSubmitted()
to wait for the async queue to finish.
for (int i = 0; i < 100; i++) {
userBox.putQueued(User(name: 'User $i'));
}
// Optional: wait until submitted items are processed.
store.awaitQueueSubmitted();
expect(userBox.count(), equals(100));
{% endtab %}
{% tab title="Python" %} {% hint style="info" %} Currently work in progress. {% endhint %} {% endtab %} {% endtabs %}
By default IDs for new objects are assigned by ObjectBox. When a new object is put, it will be assigned the next highest available ID:
{% tabs %} {% tab title="Java" %}
User user = new User();
// user.id == 0
box.put(user);
// user.id != 0
long id = user.id;
{% endtab %}
{% tab title="Kotlin" %}
val user = User()
// user.id == 0
box.put(user)
// user.id != 0
val id = user.id
{% endtab %}
{% tab title="Dart" %}
final user = User();
// user.id == 0
box.put(user);
// user.id != 0
final id = user.id;
{% endtab %}
{% tab title="Python" %}
user = User()
box.put(user)
id: int = user.id
{% endtab %} {% endtabs %}
For example, if there is an object with ID 1 and another with ID 100 in a box, the next new object that is put will be assigned ID 101.
If you try to assign a new ID yourself and put the object, ObjectBox will throw an error.
{% hint style="info" %} If you need to assign IDs by yourself, have a look at how to switch to self-assigned IDs and what side effects apply. {% endhint %}
Object IDs can not be:
0
(zero) ornull
(if using java.lang.Long) As said above, when putting an object with ID zero it will be assigned an unused ID (not zero).0xFFFFFFFFFFFFFFFF
(-1 in Java) Reserved for internal use.
For a detailed explanation see the page on Object IDs.
While ObjectBox offers powerful transactions, it is sufficient for many apps to consider just some basics guidelines about transactions:
- A
put
runs an implicit transaction. - Prefer
put
bulk overloads for lists (likeput(entities)
) when possible. - For a high number of DB interactions in loops, consider explicit transactions, such as using
runInTx()
.
For more details check the separate transaction documentation.
DaoCompat is a compatibility layer that gives you a greenDAO like API for ObjectBox. It makes switching from greenDAO to ObjectBox simple. Have a look at the documentation and the example. Contact us if you have any questions!
- Check out the ObjectBox example projects on GitHub.
- Learn about Queries and Relations.