മലയാളം

ജാവാസ്ക്രിപ്റ്റിന്റെ പുതിയ എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റ് `using`, `await using` എന്നിവ ഉപയോഗിച്ച് പഠിക്കാം. ക്ലീനപ്പ് ഓട്ടോമേറ്റ് ചെയ്യാനും, റിസോഴ്സ് ലീക്കുകൾ തടയാനും, മികച്ച കോഡ് എഴുതാനും പഠിക്കുക.

ജാവാസ്ക്രിപ്റ്റിന്റെ പുതിയ സൂപ്പർ പവർ: എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റിലേക്ക് ഒരു ആഴത്തിലുള്ള பார்வை

സോഫ്റ്റ്‌വെയർ ഡെവലപ്‌മെന്റിന്റെ ചലനാത്മകമായ ലോകത്ത്, ശക്തവും വിശ്വസനീയവും മികച്ച പ്രകടനം കാഴ്ചവയ്ക്കുന്നതുമായ ആപ്ലിക്കേഷനുകൾ നിർമ്മിക്കുന്നതിന്റെ ഒരു പ്രധാന ഭാഗമാണ് വിഭവങ്ങൾ കാര്യക്ഷമമായി കൈകാര്യം ചെയ്യുക എന്നത്. പതിറ്റാണ്ടുകളായി, ജാവാസ്ക്രിപ്റ്റ് ഡെവലപ്പർമാർ ഫയൽ ഹാൻഡിലുകൾ, നെറ്റ്‌വർക്ക് കണക്ഷനുകൾ, ഡാറ്റാബേസ് സെഷനുകൾ തുടങ്ങിയ നിർണായക വിഭവങ്ങൾ ശരിയായി റിലീസ് ചെയ്യുന്നുവെന്ന് ഉറപ്പാക്കാൻ try...catch...finally പോലുള്ള മാനുവൽ പാറ്റേണുകളെയാണ് ആശ്രയിച്ചിരുന്നത്. ഈ രീതി പ്രവർത്തനക്ഷമമാണെങ്കിലും, ഇത് പലപ്പോഴും ദൈർഘ്യമേറിയതും പിശകുകൾക്ക് സാധ്യതയുള്ളതുമാണ്. സങ്കീർണ്ണമായ സാഹചര്യങ്ങളിൽ, ചിലപ്പോൾ "ദുരന്തങ്ങളുടെ പിരമിഡ്" എന്ന് വിളിക്കപ്പെടുന്ന ഒരു പാറ്റേണായി ഇത് മാറാനും സാധ്യതയുണ്ട്.

ഇവിടെയാണ് ഭാഷയ്ക്ക് ഒരു പുതിയ മാതൃകാപരമായ മാറ്റം വരുന്നത്: എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റ് (ERM). സി#, പൈത്തൺ, ജാവ തുടങ്ങിയ ഭാഷകളിലെ സമാനമായ ആശയങ്ങളിൽ നിന്ന് പ്രചോദനം ഉൾക്കൊണ്ട്, ECMAScript 2024 (ES2024) സ്റ്റാൻഡേർഡിൽ അന്തിമരൂപം നൽകിയ ഈ ശക്തമായ ഫീച്ചർ, റിസോഴ്സ് ക്ലീനപ്പ് കൈകാര്യം ചെയ്യുന്നതിനായി ഒരു ഡിക്ലറേറ്റീവും ഓട്ടോമേറ്റഡുമായ മാർഗ്ഗം അവതരിപ്പിക്കുന്നു. പുതിയ using, await using എന്നീ കീവേഡുകൾ ഉപയോഗിക്കുന്നതിലൂടെ, കാലാതീതമായ ഒരു പ്രോഗ്രാമിംഗ് വെല്ലുവിളിക്ക് ജാവാസ്ക്രിപ്റ്റ് ഇപ്പോൾ കൂടുതൽ ലളിതവും സുരക്ഷിതവുമായ ഒരു പരിഹാരം നൽകുന്നു.

ഈ സമഗ്രമായ ഗൈഡ് ജാവാസ്ക്രിപ്റ്റിന്റെ എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റിലൂടെ നിങ്ങളെ ഒരു യാത്ര കൊണ്ടുപോകും. ഇത് പരിഹരിക്കുന്ന പ്രശ്നങ്ങൾ, അതിന്റെ പ്രധാന ആശയങ്ങൾ, പ്രായോഗിക ഉദാഹരണങ്ങൾ, കൂടാതെ നിങ്ങൾ ലോകത്ത് എവിടെയിരുന്നാലും വൃത്തിയുള്ളതും കൂടുതൽ കരുത്തുറ്റതുമായ കോഡ് എഴുതാൻ നിങ്ങളെ പ്രാപ്തരാക്കുന്ന നൂതന പാറ്റേണുകൾ എന്നിവയെല്ലാം നമ്മൾ ഇതിൽ പര്യവേക്ഷണം ചെയ്യും.

പഴയ രീതി: മാനുവൽ റിസോഴ്സ് ക്ലീനപ്പിന്റെ വെല്ലുവിളികൾ

പുതിയ സിസ്റ്റത്തിന്റെ ഭംഗി മനസ്സിലാക്കുന്നതിന് മുമ്പ്, നമ്മൾ പഴയതിന്റെ പോരായ്മകൾ മനസ്സിലാക്കണം. ജാവാസ്ക്രിപ്റ്റിൽ റിസോഴ്സ് മാനേജ്മെന്റിനുള്ള ക്ലാസിക് പാറ്റേൺ try...finally ബ്ലോക്ക് ആണ്.

ഇതിന്റെ ആശയം ലളിതമാണ്: നിങ്ങൾ try ബ്ലോക്കിൽ ഒരു റിസോഴ്സ് നേടുന്നു, അത് finally ബ്ലോക്കിൽ റിലീസ് ചെയ്യുന്നു. try ബ്ലോക്കിലെ കോഡ് വിജയിച്ചാലും പരാജയപ്പെട്ടാലും അല്ലെങ്കിൽ നേരത്തെ റിട്ടേൺ ചെയ്താലും finally ബ്ലോക്ക് പ്രവർത്തിക്കുമെന്ന് ഉറപ്പാണ്.

ഒരു സാധാരണ സെർവർ-സൈഡ് സാഹചര്യം പരിഗണിക്കാം: ഒരു ഫയൽ തുറക്കുക, അതിൽ കുറച്ച് ഡാറ്റ എഴുതുക, തുടർന്ന് ഫയൽ അടച്ചുവെന്ന് ഉറപ്പാക്കുക.

ഉദാഹരണം: try...finally ഉപയോഗിച്ചുള്ള ഒരു ലളിതമായ ഫയൽ ഓപ്പറേഷൻ


const fs = require('fs/promises');

async function processFile(filePath, data) {
  let fileHandle;
  try {
    console.log('ഫയൽ തുറക്കുന്നു...');
    fileHandle = await fs.open(filePath, 'w');
    console.log('ഫയലിലേക്ക് എഴുതുന്നു...');
    await fileHandle.write(data);
    console.log('ഡാറ്റ വിജയകരമായി എഴുതി.');
  } catch (error) {
    console.error('ഫയൽ പ്രോസസ്സിംഗിനിടെ ഒരു പിശക് സംഭവിച്ചു:', error);
  } finally {
    if (fileHandle) {
      console.log('ഫയൽ അടയ്ക്കുന്നു...');
      await fileHandle.close();
    }
  }
}

ഈ കോഡ് പ്രവർത്തിക്കുന്നു, പക്ഷേ ഇത് പല ബലഹീനതകളും വെളിപ്പെടുത്തുന്നു:

ഇനി, ഒരു ഡാറ്റാബേസ് കണക്ഷനും ഒരു ഫയൽ ഹാൻഡിലും പോലെ ഒന്നിലധികം റിസോഴ്സുകൾ കൈകാര്യം ചെയ്യുന്നത് സങ്കൽപ്പിക്കുക. കോഡ് പെട്ടെന്ന് തന്നെ ഒരു നെസ്റ്റഡ് രൂപത്തിലേക്ക് മാറുന്നു:


async function logQueryResultToFile(query, filePath) {
  let dbConnection;
  try {
    dbConnection = await getDbConnection();
    const result = await dbConnection.query(query);

    let fileHandle;
    try {
      fileHandle = await fs.open(filePath, 'w');
      await fileHandle.write(JSON.stringify(result));
    } finally {
      if (fileHandle) {
        await fileHandle.close();
      }
    }
  } finally {
    if (dbConnection) {
      await dbConnection.release();
    }
  }
}

ഇത്തരത്തിലുള്ള നെസ്റ്റിംഗ് പരിപാലിക്കാനും വികസിപ്പിക്കാനും പ്രയാസമാണ്. ഒരു മികച്ച അബ്സ്ട്രാക്ഷൻ ആവശ്യമാണെന്നതിന്റെ വ്യക്തമായ സൂചനയാണിത്. എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റ് രൂപകൽപ്പന ചെയ്തിരിക്കുന്നത് കൃത്യമായും ഈ പ്രശ്നം പരിഹരിക്കാനാണ്.

ഒരു പുതിയ മാതൃക: എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റിന്റെ തത്വങ്ങൾ

എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റ് (ERM) ഒരു റിസോഴ്സ് ഒബ്ജക്റ്റും ജാവാസ്ക്രിപ്റ്റ് റൺടൈമും തമ്മിൽ ഒരു കരാർ ഉണ്ടാക്കുന്നു. ഇതിന്റെ പ്രധാന ആശയം ലളിതമാണ്: ഒരു ഒബ്ജക്റ്റിന് അത് എങ്ങനെ വൃത്തിയാക്കണമെന്ന് സ്വയം പ്രഖ്യാപിക്കാൻ കഴിയും, ഒപ്പം ആ ഒബ്ജക്റ്റ് സ്കോപ്പിന് പുറത്തുപോകുമ്പോൾ ആ ക്ലീനപ്പ് യാന്ത്രികമായി നടപ്പിലാക്കാനുള്ള സിന്റാക്സ് ഭാഷ നൽകുന്നു.

ഇത് രണ്ട് പ്രധാന ഘടകങ്ങളിലൂടെയാണ് സാധ്യമാക്കുന്നത്:

  1. ഡിസ്പോസിബിൾ പ്രോട്ടോക്കോൾ: ഒബ്ജക്റ്റുകൾക്ക് അവയുടെ സ്വന്തം ക്ലീനപ്പ് ലോജിക് നിർവചിക്കുന്നതിനുള്ള ഒരു സ്റ്റാൻഡേർഡ് മാർഗ്ഗം. സിൻക്രണസ് ക്ലീനപ്പിനായി Symbol.dispose, അസിൻക്രണസ് ക്ലീനപ്പിനായി Symbol.asyncDispose എന്നിങ്ങനെ പ്രത്യേക ചിഹ്നങ്ങൾ ഉപയോഗിക്കുന്നു.
  2. using, await using ഡിക്ലറേഷനുകൾ: ഒരു റിസോഴ്സിനെ ഒരു ബ്ലോക്ക് സ്കോപ്പിലേക്ക് ബന്ധിപ്പിക്കുന്ന പുതിയ കീവേഡുകൾ. ബ്ലോക്കിൽ നിന്ന് പുറത്തുകടക്കുമ്പോൾ, റിസോഴ്സിന്റെ ക്ലീനപ്പ് മെത്തേഡ് യാന്ത്രികമായി വിളിക്കപ്പെടുന്നു.

പ്രധാന ആശയങ്ങൾ: `Symbol.dispose`, `Symbol.asyncDispose`

ERM-ന്റെ ഹൃദയഭാഗത്ത് രണ്ട് പുതിയ സുപ്രസിദ്ധമായ സിംബലുകളുണ്ട്. ഈ സിംബലുകളിലൊന്ന് കീ ആയിട്ടുള്ള ഒരു മെത്തേഡ് ഉള്ള ഒബ്ജക്റ്റിനെ "ഡിസ്പോസിബിൾ റിസോഴ്സ്" എന്ന് കണക്കാക്കുന്നു.

Symbol.dispose ഉപയോഗിച്ചുള്ള സിൻക്രണസ് ഡിസ്പോസൽ

Symbol.dispose സിംബൽ ഒരു സിൻക്രണസ് ക്ലീനപ്പ് മെത്തേഡിനെ വ്യക്തമാക്കുന്നു. ഒരു ഫയൽ ഹാൻഡിൽ സിൻക്രണസായി അടയ്ക്കുകയോ അല്ലെങ്കിൽ ഒരു ഇൻ-മെമ്മറി ലോക്ക് റിലീസ് ചെയ്യുകയോ പോലുള്ള അസിൻക്രണസ് പ്രവർത്തനങ്ങൾ ആവശ്യമില്ലാത്ത റിസോഴ്സുകൾക്ക് ഇത് അനുയോജ്യമാണ്.

സ്വയം വൃത്തിയാക്കുന്ന ഒരു താൽക്കാലിക ഫയലിനായി ഒരു റാപ്പർ ഉണ്ടാക്കാം.


const fs = require('fs');
const path = require('path');

class TempFile {
  constructor(content) {
    this.path = path.join(__dirname, `temp_${Date.now()}.txt`);
    fs.writeFileSync(this.path, content);
    console.log(`താൽക്കാലിക ഫയൽ ഉണ്ടാക്കി: ${this.path}`);
  }

  // ഇതാണ് സിൻക്രണസ് ഡിസ്പോസിബിൾ മെത്തേഡ്
  [Symbol.dispose]() {
    console.log(`താൽക്കാലിക ഫയൽ ഡിസ്പോസ് ചെയ്യുന്നു: ${this.path}`);
    try {
      fs.unlinkSync(this.path);
      console.log('ഫയൽ വിജയകരമായി ഇല്ലാതാക്കി.');
    } catch (error) {
      console.error(`ഫയൽ ഇല്ലാതാക്കുന്നതിൽ പരാജയപ്പെട്ടു: ${this.path}`, error);
      // ഡിസ്പോസിനുള്ളിലും പിശകുകൾ കൈകാര്യം ചെയ്യേണ്ടത് പ്രധാനമാണ്!
    }
  }
}

`TempFile`-ന്റെ ഏതൊരു ഇൻസ്റ്റൻസും ഇപ്പോൾ ഒരു ഡിസ്പോസിബിൾ റിസോഴ്സാണ്. ഡിസ്കിൽ നിന്ന് ഫയൽ ഇല്ലാതാക്കാനുള്ള ലോജിക് അടങ്ങുന്ന `Symbol.dispose` എന്ന കീ ഉള്ള ഒരു മെത്തേഡ് അതിനുണ്ട്.

Symbol.asyncDispose ഉപയോഗിച്ചുള്ള അസിൻക്രണസ് ഡിസ്പോസൽ

ആധുനിക ക്ലീനപ്പ് ഓപ്പറേഷനുകളിൽ പലതും അസിൻക്രണസ് ആണ്. ഒരു ഡാറ്റാബേസ് കണക്ഷൻ അടയ്ക്കുന്നതിന് നെറ്റ്‌വർക്കിലൂടെ ഒരു QUIT കമാൻഡ് അയയ്‌ക്കേണ്ടി വന്നേക്കാം, അല്ലെങ്കിൽ ഒരു മെസേജ് ക്യൂ ക്ലയിന്റിന് അതിന്റെ ഔട്ട്‌ഗോയിംഗ് ബഫർ ഫ്ലഷ് ചെയ്യേണ്ടി വന്നേക്കാം. ഈ സാഹചര്യങ്ങൾക്കായി നമ്മൾ Symbol.asyncDispose ഉപയോഗിക്കുന്നു.

Symbol.asyncDispose-മായി ബന്ധപ്പെട്ട മെത്തേഡ് ഒരു Promise റിട്ടേൺ ചെയ്യണം (അല്ലെങ്കിൽ ഒരു async ഫംഗ്ഷൻ ആയിരിക്കണം).

ഒരു പൂളിലേക്ക് അസിൻക്രണസായി തിരികെ നൽകേണ്ട ഒരു മോക്ക് ഡാറ്റാബേസ് കണക്ഷൻ മോഡൽ ചെയ്യാം.


// ഒരു മോക്ക് ഡാറ്റാബേസ് പൂൾ
const mockDbPool = {
  getConnection: () => {
    console.log('ഡിബി കണക്ഷൻ ലഭിച്ചു.');
    return new MockDbConnection();
  }
};

class MockDbConnection {
  query(sql) {
    console.log(`ക്വറി എക്സിക്യൂട്ട് ചെയ്യുന്നു: ${sql}`);
    return Promise.resolve({ success: true, rows: [] });
  }

  // ഇതാണ് അസിൻക്രണസ് ഡിസ്പോസിബിൾ മെത്തേഡ്
  async [Symbol.asyncDispose]() {
    console.log('ഡിബി കണക്ഷൻ പൂളിലേക്ക് തിരികെ നൽകുന്നു...');
    // കണക്ഷൻ റിലീസ് ചെയ്യുന്നതിനുള്ള നെറ്റ്‌വർക്ക് കാലതാമസം സിമുലേറ്റ് ചെയ്യുന്നു
    await new Promise(resolve => setTimeout(resolve, 50));
    console.log('ഡിബി കണക്ഷൻ റിലീസ് ചെയ്തു.');
  }
}

ഇപ്പോൾ, ഏതൊരു `MockDbConnection` ഇൻസ്റ്റൻസും ഒരു അസിങ്ക് ഡിസ്പോസിബിൾ റിസോഴ്സ് ആണ്. ആവശ്യമില്ലാതെ വരുമ്പോൾ എങ്ങനെ അസിൻക്രണസായി സ്വയം റിലീസ് ചെയ്യണമെന്ന് അതിനറിയാം.

പുതിയ സിന്റാക്സ്: `using`, `await using` എന്നിവയുടെ പ്രവർത്തനം

നമ്മുടെ ഡിസ്പോസിബിൾ ക്ലാസുകൾ നിർവചിച്ചുകഴിഞ്ഞാൽ, അവയെ യാന്ത്രികമായി കൈകാര്യം ചെയ്യാൻ നമുക്ക് പുതിയ കീവേഡുകൾ ഉപയോഗിക്കാം. ഈ കീവേഡുകൾ `let`, `const` എന്നിവ പോലെ ബ്ലോക്ക്-സ്കോപ്പ്ഡ് ഡിക്ലറേഷനുകൾ സൃഷ്ടിക്കുന്നു.

`using` ഉപയോഗിച്ചുള്ള സിൻക്രണസ് ക്ലീനപ്പ്

`using` കീവേഡ് ഉപയോഗിക്കുന്നത് `Symbol.dispose` നടപ്പിലാക്കുന്ന റിസോഴ്സുകൾക്കാണ്. `using` ഡിക്ലറേഷൻ നടത്തിയ ബ്ലോക്കിൽ നിന്ന് കോഡ് എക്സിക്യൂഷൻ പുറത്തുപോകുമ്പോൾ, `[Symbol.dispose]()` മെത്തേഡ് യാന്ത്രികമായി വിളിക്കപ്പെടുന്നു.

നമുക്ക് നമ്മുടെ `TempFile` ക്ലാസ് ഉപയോഗിക്കാം:


function processDataWithTempFile() {
  console.log('ബ്ലോക്കിലേക്ക് പ്രവേശിക്കുന്നു...');
  using tempFile = new TempFile('This is some important data.');

  // നിങ്ങൾക്ക് ഇവിടെ tempFile ഉപയോഗിച്ച് പ്രവർത്തിക്കാം
  const content = fs.readFileSync(tempFile.path, 'utf8');
  console.log(`താൽക്കാലിക ഫയലിൽ നിന്ന് വായിച്ചത്: "${content}"`);

  // ഇവിടെ ക്ലീനപ്പ് കോഡ് ആവശ്യമില്ല!
  console.log('...കൂടുതൽ ജോലികൾ ചെയ്യുന്നു...');
} // <-- tempFile.[Symbol.dispose]() ഇവിടെ യാന്ത്രികമായി വിളിക്കപ്പെടുന്നു!

processDataWithTempFile();
console.log('ബ്ലോക്കിൽ നിന്ന് പുറത്തുകടന്നു.');

ഔട്ട്പുട്ട് ഇങ്ങനെയായിരിക്കും:

ബ്ലോക്കിലേക്ക് പ്രവേശിക്കുന്നു...
താൽക്കാലിക ഫയൽ ഉണ്ടാക്കി: /path/to/temp_1678886400000.txt
താൽക്കാലിക ഫയലിൽ നിന്ന് വായിച്ചത്: "This is some important data."
...കൂടുതൽ ജോലികൾ ചെയ്യുന്നു...
താൽക്കാലിക ഫയൽ ഡിസ്പോസ് ചെയ്യുന്നു: /path/to/temp_1678886400000.txt
ഫയൽ വിജയകരമായി ഇല്ലാതാക്കി.
ബ്ലോക്കിൽ നിന്ന് പുറത്തുകടന്നു.

ഇത് എത്ര വൃത്തിയുള്ളതാണെന്ന് നോക്കൂ! റിസോഴ്സിന്റെ മുഴുവൻ ലൈഫ് സൈക്കിളും ആ ബ്ലോക്കിനുള്ളിൽ ഒതുങ്ങിയിരിക്കുന്നു. നമ്മൾ അത് ഡിക്ലയർ ചെയ്യുന്നു, ഉപയോഗിക്കുന്നു, പിന്നെ അതിനെക്കുറിച്ച് മറക്കുന്നു. ഭാഷ ക്ലീനപ്പ് കൈകാര്യം ചെയ്യുന്നു. ഇത് വായനാക്ഷമതയിലും സുരക്ഷയിലും ഒരു വലിയ മെച്ചപ്പെടുത്തലാണ്.

ഒന്നിലധികം റിസോഴ്സുകൾ കൈകാര്യം ചെയ്യൽ

ഒരേ ബ്ലോക്കിൽ നിങ്ങൾക്ക് ഒന്നിലധികം `using` ഡിക്ലറേഷനുകൾ ഉണ്ടാവാം. അവയുടെ നിർമ്മാണത്തിന്റെ വിപരീത ക്രമത്തിൽ (ഒരു LIFO അല്ലെങ്കിൽ "സ്റ്റാക്ക്-പോലുള്ള" സ്വഭാവം) അവ ഡിസ്പോസ് ചെയ്യപ്പെടും.


{
  using resourceA = new MyDisposable('A'); // ആദ്യം ഉണ്ടാക്കിയത്
  using resourceB = new MyDisposable('B'); // രണ്ടാമത് ഉണ്ടാക്കിയത്
  console.log('ബ്ലോക്കിനുള്ളിൽ, റിസോഴ്സുകൾ ഉപയോഗിക്കുന്നു...');
} // resourceB ആദ്യം ഡിസ്പോസ് ചെയ്യപ്പെടുന്നു, തുടർന്ന് resourceA

`await using` ഉപയോഗിച്ചുള്ള അസിൻക്രണസ് ക്ലീനപ്പ്

`await using` കീവേഡ് `using`-ന്റെ അസിൻക്രണസ് പതിപ്പാണ്. `Symbol.asyncDispose` നടപ്പിലാക്കുന്ന റിസോഴ്സുകൾക്കാണ് ഇത് ഉപയോഗിക്കുന്നത്. ക്ലീനപ്പ് അസിൻക്രണസ് ആയതിനാൽ, ഈ കീവേഡ് ഒരു `async` ഫംഗ്ഷനുള്ളിലോ അല്ലെങ്കിൽ ഒരു മൊഡ്യൂളിന്റെ ടോപ്പ് ലെവലിലോ മാത്രമേ ഉപയോഗിക്കാൻ കഴിയൂ (ടോപ്പ്-ലെവൽ await പിന്തുണയ്ക്കുന്നുണ്ടെങ്കിൽ).

നമുക്ക് നമ്മുടെ `MockDbConnection` ക്ലാസ് ഉപയോഗിക്കാം:


async function performDatabaseOperation() {
  console.log('അസിങ്ക് ഫംഗ്ഷനിലേക്ക് പ്രവേശിക്കുന്നു...');
  await using db = mockDbPool.getConnection();

  await db.query('SELECT * FROM users');

  console.log('ഡാറ്റാബേസ് പ്രവർത്തനം പൂർത്തിയായി.');
} // <-- await db.[Symbol.asyncDispose]() ഇവിടെ യാന്ത്രികമായി വിളിക്കപ്പെടുന്നു!

(async () => {
  await performDatabaseOperation();
  console.log('അസിങ്ക് ഫംഗ്ഷൻ പൂർത്തിയായി.');
})();

ഔട്ട്പുട്ട് അസിൻക്രണസ് ക്ലീനപ്പ് കാണിക്കുന്നു:

അസിങ്ക് ഫംഗ്ഷനിലേക്ക് പ്രവേശിക്കുന്നു...
ഡിബി കണക്ഷൻ ലഭിച്ചു.
ക്വറി എക്സിക്യൂട്ട് ചെയ്യുന്നു: SELECT * FROM users
ഡാറ്റാബേസ് പ്രവർത്തനം പൂർത്തിയായി.
ഡിബി കണക്ഷൻ പൂളിലേക്ക് തിരികെ നൽകുന്നു...
(50ms കാത്തിരിക്കുന്നു)
ഡിബി കണക്ഷൻ റിലീസ് ചെയ്തു.
അസിങ്ക് ഫംഗ്ഷൻ പൂർത്തിയായി.

`using`-നെപ്പോലെ തന്നെ, `await using` സിന്റാക്സും മുഴുവൻ ലൈഫ് സൈക്കിളും കൈകാര്യം ചെയ്യുന്നു, പക്ഷേ ഇത് അസിൻക്രണസ് ക്ലീനപ്പ് പ്രക്രിയയെ ശരിയായി `awaits` ചെയ്യുന്നു. സിൻക്രണസായി മാത്രം ഡിസ്പോസ് ചെയ്യാൻ കഴിയുന്ന റിസോഴ്സുകളെ പോലും ഇതിന് കൈകാര്യം ചെയ്യാൻ കഴിയും - അത് അവയെ await ചെയ്യില്ലെന്ന് മാത്രം.

നൂതന പാറ്റേണുകൾ: `DisposableStack`, `AsyncDisposableStack`

ചിലപ്പോൾ, `using`-ന്റെ ലളിതമായ ബ്ലോക്ക്-സ്കോപ്പിംഗ് വേണ്ടത്ര ഫ്ലെക്സിബിൾ ആയിരിക്കില്ല. ഒരു കൂട്ടം റിസോഴ്സുകളുടെ ലൈഫ് ടൈം ഒരു പ്രത്യേക ലെക്സിക്കൽ ബ്ലോക്കുമായി ബന്ധിപ്പിച്ചിട്ടില്ലെങ്കിൽ എന്തുചെയ്യും? അല്ലെങ്കിൽ `Symbol.dispose` ഉള്ള ഒബ്ജക്റ്റുകൾ നൽകാത്ത ഒരു പഴയ ലൈബ്രറിയുമായി നിങ്ങൾ സംയോജിപ്പിക്കുകയാണെങ്കിലോ?

ഈ സാഹചര്യങ്ങൾക്കായി, ജാവാസ്ക്രിപ്റ്റ് രണ്ട് സഹായ ക്ലാസുകൾ നൽകുന്നു: `DisposableStack`, `AsyncDisposableStack`.

`DisposableStack`: ഫ്ലെക്സിബിൾ ക്ലീനപ്പ് മാനേജർ

ഒരു `DisposableStack` എന്നത് ക്ലീനപ്പ് ഓപ്പറേഷനുകളുടെ ഒരു ശേഖരം കൈകാര്യം ചെയ്യുന്ന ഒരു ഒബ്ജക്റ്റാണ്. ഇത് സ്വയം ഒരു ഡിസ്പോസിബിൾ റിസോഴ്സ് ആയതിനാൽ, അതിന്റെ മുഴുവൻ ലൈഫ് ടൈമും ഒരു `using` ബ്ലോക്ക് ഉപയോഗിച്ച് നിങ്ങൾക്ക് കൈകാര്യം ചെയ്യാൻ കഴിയും.

ഇതിന് ഉപയോഗപ്രദമായ നിരവധി മെത്തേഡുകളുണ്ട്:

ഉദാഹരണം: കണ്ടീഷണൽ റിസോഴ്സ് മാനേജ്മെന്റ്

ഒരു പ്രത്യേക വ്യവസ്ഥ പാലിക്കുമ്പോൾ മാത്രം ഒരു ലോഗ് ഫയൽ തുറക്കുന്ന ഒരു ഫംഗ്ഷൻ സങ്കൽപ്പിക്കുക, എന്നാൽ എല്ലാ ക്ലീനപ്പും അവസാനം ഒരിടത്ത് നടക്കണമെന്ന് നിങ്ങൾ ആഗ്രഹിക്കുന്നു.


function processWithConditionalLogging(shouldLog) {
  using stack = new DisposableStack();

  const db = stack.use(getDbConnection()); // എപ്പോഴും ഡിബി ഉപയോഗിക്കുക

  if (shouldLog) {
    const logFileStream = fs.createWriteStream('app.log');
    // സ്ട്രീമിനായുള്ള ക്ലീനപ്പ് മാറ്റിവയ്ക്കുക
    stack.defer(() => {
      console.log('ലോഗ് ഫയൽ സ്ട്രീം അടയ്ക്കുന്നു...');
      logFileStream.end();
    });
    db.logTo(logFileStream);
  }

  db.doWork();

} // <-- സ്റ്റാക്ക് ഡിസ്പോസ് ചെയ്യപ്പെടുന്നു, രജിസ്റ്റർ ചെയ്ത എല്ലാ ക്ലീനപ്പ് ഫംഗ്ഷനുകളെയും LIFO ക്രമത്തിൽ വിളിക്കുന്നു.

`AsyncDisposableStack`: അസിൻക്രണസ് ലോകത്തിനായി

നിങ്ങൾ ഊഹിക്കുന്നതുപോലെ, `AsyncDisposableStack` അസിൻക്രണസ് പതിപ്പാണ്. ഇതിന് സിൻക്രണസ്, അസിൻക്രണസ് ഡിസ്പോസിബിളുകൾ കൈകാര്യം ചെയ്യാൻ കഴിയും. ഇതിന്റെ പ്രധാന ക്ലീനപ്പ് മെത്തേഡ് `.disposeAsync()` ആണ്, ഇത് എല്ലാ അസിൻക്രണസ് ക്ലീനപ്പ് ഓപ്പറേഷനുകളും പൂർത്തിയാകുമ്പോൾ റിസോൾവ് ചെയ്യുന്ന ഒരു `Promise` തിരികെ നൽകുന്നു.

ഉദാഹരണം: പലതരം റിസോഴ്സുകൾ കൈകാര്യം ചെയ്യൽ

ഒരു ഡാറ്റാബേസ് കണക്ഷനും (അസിങ്ക് ക്ലീനപ്പ്) ഒരു താൽക്കാലിക ഫയലും (സിങ്ക് ക്ലീനപ്പ്) ആവശ്യമുള്ള ഒരു വെബ് സെർവർ റിക്വസ്റ്റ് ഹാൻഡ്‌ലർ ഉണ്ടാക്കാം.


async function handleRequest() {
  await using stack = new AsyncDisposableStack();

  // ഒരു അസിങ്ക് ഡിസ്പോസിബിൾ റിസോഴ്സ് കൈകാര്യം ചെയ്യുക
  const dbConnection = await stack.use(getAsyncDbConnection());

  // ഒരു സിങ്ക് ഡിസ്പോസിബിൾ റിസോഴ്സ് കൈകാര്യം ചെയ്യുക
  const tempFile = stack.use(new TempFile('request data'));

  // ഒരു പഴയ API-ൽ നിന്ന് ഒരു റിസോഴ്സ് സ്വീകരിക്കുക
  const legacyResource = getLegacyResource();
  stack.adopt(legacyResource, () => legacyResource.shutdown());

  console.log('അഭ്യർത്ഥന പ്രോസസ്സ് ചെയ്യുന്നു...');
  await doWork(dbConnection, tempFile.path);

} // <-- stack.disposeAsync() വിളിക്കപ്പെടുന്നു. ഇത് അസിങ്ക് ക്ലീനപ്പ് ശരിയായി await ചെയ്യും.

സങ്കീർണ്ണമായ സെറ്റപ്പ്, ടിയർഡൗൺ ലോജിക്കുകൾ വൃത്തിയുള്ളതും പ്രവചിക്കാവുന്നതുമായ രീതിയിൽ ക്രമീകരിക്കുന്നതിനുള്ള ശക്തമായ ഒരു ഉപകരണമാണ് `AsyncDisposableStack`.

`SuppressedError` ഉപയോഗിച്ച് ശക്തമായ എറർ ഹാൻഡ്ലിംഗ്

ERM-ന്റെ ഏറ്റവും സൂക്ഷ്മവും എന്നാൽ പ്രധാനപ്പെട്ടതുമായ മെച്ചപ്പെടുത്തലുകളിലൊന്ന് അത് പിശകുകൾ എങ്ങനെ കൈകാര്യം ചെയ്യുന്നു എന്നതാണ്. `using` ബ്ലോക്കിനുള്ളിൽ ഒരു പിശക് സംഭവിക്കുകയും, തുടർന്ന് യാന്ത്രിക ഡിസ്പോസൽ സമയത്ത് *മറ്റൊരു* പിശക് സംഭവിക്കുകയും ചെയ്താൽ എന്ത് സംഭവിക്കും?

പഴയ `try...finally` രീതിയിൽ, `finally` ബ്ലോക്കിലെ പിശക് സാധാരണയായി `try` ബ്ലോക്കിലെ യഥാർത്ഥവും കൂടുതൽ പ്രധാനപ്പെട്ടതുമായ പിശകിനെ മറികടക്കുകയോ "അടിച്ചമർത്തുകയോ" ചെയ്യുമായിരുന്നു. ഇത് പലപ്പോഴും ഡീബഗ്ഗിംഗ് അവിശ്വസനീയമാംവിധം ബുദ്ധിമുട്ടുള്ളതാക്കിയിരുന്നു.

ERM ഇതിന് ഒരു പുതിയ ഗ്ലോബൽ എറർ ടൈപ്പ് ഉപയോഗിച്ച് പരിഹാരം കാണുന്നു: `SuppressedError`. മറ്റൊരു പിശക് ഇതിനകം നിലവിലിരിക്കെ ഡിസ്പോസൽ സമയത്ത് ഒരു പിശക് സംഭവിച്ചാൽ, ഡിസ്പോസൽ പിശക് "അടിച്ചമർത്തപ്പെടുന്നു". യഥാർത്ഥ പിശക് തന്നെ ത്രോ ചെയ്യപ്പെടും, പക്ഷേ അതിന് ഇപ്പോൾ ഡിസ്പോസൽ പിശക് അടങ്ങുന്ന ഒരു `suppressed` പ്രോപ്പർട്ടി ഉണ്ടാകും.


class FaultyResource {
  [Symbol.dispose]() {
    throw new Error('ഡിസ്പോസൽ സമയത്ത് പിശക്!');
  }
}

try {
  using resource = new FaultyResource();
  throw new Error('പ്രവർത്തന സമയത്ത് പിശക്!');
} catch (e) {
  console.log(`പിടിച്ചെടുത്ത പിശക്: ${e.message}`); // പ്രവർത്തന സമയത്ത് പിശക്!
  if (e.suppressed) {
    console.log(`അടിച്ചമർത്തപ്പെട്ട പിശക്: ${e.suppressed.message}`); // ഡിസ്പോസൽ സമയത്ത് പിശക്!
    console.log(e instanceof SuppressedError); // false
    console.log(e.suppressed instanceof Error); // true
  }
}

ഈ സ്വഭാവം യഥാർത്ഥ പരാജയത്തിന്റെ സന്ദർഭം നിങ്ങൾക്ക് ഒരിക്കലും നഷ്ടപ്പെടുന്നില്ലെന്ന് ഉറപ്പാക്കുന്നു, ഇത് കൂടുതൽ കരുത്തുറ്റതും ഡീബഗ് ചെയ്യാൻ എളുപ്പമുള്ളതുമായ സിസ്റ്റങ്ങളിലേക്ക് നയിക്കുന്നു.

ജാവാസ്ക്രിപ്റ്റ് ഇക്കോസിസ്റ്റത്തിലുടനീളമുള്ള പ്രായോഗിക ഉപയോഗങ്ങൾ

എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റിന്റെ പ്രയോഗങ്ങൾ വളരെ വലുതാണ്, അത് ബാക്ക്-എൻഡ്, ഫ്രണ്ട്-എൻഡ്, അല്ലെങ്കിൽ ടെസ്റ്റിംഗ് എന്നിവയിൽ പ്രവർത്തിക്കുന്ന ലോകമെമ്പാടുമുള്ള ഡെവലപ്പർമാർക്ക് പ്രസക്തമാണ്.

ബ്രൗസർ, റൺടൈം പിന്തുണ

ഒരു ആധുനിക ഫീച്ചർ എന്ന നിലയിൽ, നിങ്ങൾക്ക് എവിടെയൊക്കെ എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റ് ഉപയോഗിക്കാൻ കഴിയുമെന്ന് അറിയേണ്ടത് പ്രധാനമാണ്. 2023-ന്റെ അവസാനത്തിലും 2024-ന്റെ തുടക്കത്തിലും, പ്രധാന ജാവാസ്ക്രിപ്റ്റ് എൻവയോൺമെന്റുകളുടെ ഏറ്റവും പുതിയ പതിപ്പുകളിൽ ഇതിന് വ്യാപകമായ പിന്തുണയുണ്ട്:

പഴയ എൻവയോൺമെന്റുകൾക്കായി, `using` സിന്റാക്സ് ട്രാൻസ്ഫോം ചെയ്യുന്നതിനും ആവശ്യമായ സിംബലുകളും സ്റ്റാക്ക് ക്ലാസുകളും പോളിഫിൽ ചെയ്യുന്നതിനും അനുയോജ്യമായ പ്ലഗിനുകളുള്ള Babel പോലുള്ള ട്രാൻസ്പൈലറുകളെ ആശ്രയിക്കേണ്ടി വരും.

ഉപസംഹാരം: സുരക്ഷയുടെയും വ്യക്തതയുടെയും ഒരു പുതിയ യുഗം

ജാവാസ്ക്രിപ്റ്റിന്റെ എക്സ്പ്ലിസിറ്റ് റിസോഴ്സ് മാനേജ്മെന്റ് വെറുമൊരു സിന്റാക്റ്റിക് ഷുഗർ മാത്രമല്ല; ഇത് ഭാഷയുടെ സുരക്ഷ, വ്യക്തത, പരിപാലനക്ഷമത എന്നിവ പ്രോത്സാഹിപ്പിക്കുന്ന ഒരു അടിസ്ഥാനപരമായ മെച്ചപ്പെടുത്തലാണ്. വിഭവങ്ങൾ വൃത്തിയാക്കുന്നതിന്റെ മടുപ്പിക്കുന്നതും പിശകുകൾക്ക് സാധ്യതയുള്ളതുമായ പ്രക്രിയ ഓട്ടോമേറ്റ് ചെയ്യുന്നതിലൂടെ, ഡെവലപ്പർമാരെ അവരുടെ പ്രധാന ബിസിനസ്സ് ലോജിക്കിൽ ശ്രദ്ധ കേന്ദ്രീകരിക്കാൻ ഇത് സഹായിക്കുന്നു.

പ്രധാന ആശയങ്ങൾ ഇവയാണ്:

നിങ്ങൾ പുതിയ പ്രോജക്റ്റുകൾ ആരംഭിക്കുമ്പോഴോ നിലവിലുള്ള കോഡ് റീഫാക്ടർ ചെയ്യുമ്പോഴോ ഈ ശക്തമായ പുതിയ പാറ്റേൺ സ്വീകരിക്കുന്നത് പരിഗണിക്കുക. ഇത് നിങ്ങളുടെ ജാവാസ്ക്രിപ്റ്റ് വൃത്തിയുള്ളതും, നിങ്ങളുടെ ആപ്ലിക്കേഷനുകൾ കൂടുതൽ വിശ്വസനീയവും, ഒരു ഡെവലപ്പർ എന്ന നിലയിൽ നിങ്ങളുടെ ജീവിതം കുറച്ചുകൂടി എളുപ്പമുള്ളതുമാക്കും. ആധുനികവും പ്രൊഫഷണലുമായ ജാവാസ്ക്രിപ്റ്റ് എഴുതുന്നതിനുള്ള ഒരു ആഗോള നിലവാരമാണിത്.