# Overview Tesseract.js offers 2 ways to run recognition jobs: (1) using a worker directly, or (2) using a scheduler to run jobs on multiple workers in parallel. The syntax for the latter is more complicated, but using parallel processing via schedulers provides significantly better performance for large jobs. For more detailed documentation on each function, see the [api page](./api.md). # Option 1: Using Workers Directly Tesseract.js also supports creating and managing workers (the objects that execute recognition) manually. ``` (async () => { const worker = await Tesseract.createWorker('eng'); const { data: { text } } = await worker.recognize('https://tesseract.projectnaptha.com/img/eng_bw.png'); console.log(text); await worker.terminate(); })(); ``` In actual use, the `createWorker` step should be separated from the `worker.recognize` step. Doing so enables the following benefits: 1. Workers can be prepared ahead of time - E.g. a worker can be created when the page is first loaded, rather than waiting for a user to upload an image to recognize 1. Workers can be reused for multiple recognition jobs, rather than creating a new worker and loading language data for every image recognized - Remember to call `worker.terminate()` after all recognition is complete to free memory # Option 2: Using Schedulers + Workers Tesseract.js also supports executing jobs using schedulers. A scheduler is an object that contains multiple workers, which it uses to execute jobs in parallel. For example, the following code executes 10 jobs in parallel using 4 workers. ``` const scheduler = Tesseract.createScheduler(); // Creates worker and adds to scheduler const workerGen = async () => { const worker = await Tesseract.createWorker('eng'); scheduler.addWorker(worker); } const workerN = 4; (async () => { const resArr = Array(workerN); for (let i=0; i ( scheduler.addJob('recognize', 'https://tesseract.projectnaptha.com/img/eng_bw.png').then((x) => console.log(x.data.text)) ))) await scheduler.terminate(); // It also terminates all workers. })(); ``` While using schedulers is no more efficient for a single job, they allow for quickly executing large numbers of jobs in parallel. When working with schedulers, note that workers added to the same scheduler should all be homogenous—they should have the same language be configured with the same parameters. Schedulers assign jobs to workers in a non-deterministic manner, so if the workers are not identical then recognition results will depend on which worker the job is assigned to.