search

Home  >  Q&A  >  body text

Improve performance of MYSQL queries with one-to-many relationships

I have a query in my database that takes 25 seconds to return results, which is too long. It seems like it should be pretty simple. Two tables; the main table (document) is a standard table with some data columns, and the connection table is a mapping table with only two columns (parent_id, division_id). Previously there was no index on the mapping table, so I added an index and changed the "interpretation" to include the index, but it seems to have no impact on performance.

The query is as follows:

explain SELECT DISTINCT doc.*  
FROM document doc  
LEFT JOIN multi_division_mapper divisions ON doc.id = divisions.parent_id  
WHERE doc.clientId = 'SOME_GUID'  
AND (divisions.division_id IS NULL OR divisions.division_id  IN ('SOME_GUID'));

The result of the explanation is:

Total number of lines in document: 6720 Total number of rows in mapper: 6173

Based on the information I gathered, I need to improve "type" or "extra" to make the query faster. What can I do here?

Create table statement:

CREATE TABLE `document` (
   `id` varchar(36) NOT NULL,
   `addedBy` varchar(255) DEFAULT NULL,
   `addedDate` datetime NOT NULL,
   `editedBy` varchar(255) DEFAULT NULL,
   `editedDate` datetime NOT NULL,
   `deleted` bit(1) DEFAULT NULL,
   `clientId` varchar(36) NOT NULL,
   `departmentId` varchar(36) DEFAULT NULL,
   `documentParentId` varchar(36) DEFAULT NULL,
   `documentParent` varchar(50) DEFAULT NULL,
   `fileId` varchar(255) DEFAULT NULL,
   `fileUrl` varchar(600) DEFAULT NULL,
   `documentName` varchar(500) NOT NULL,
   `displayName` varchar(255) NOT NULL,
   `documentId` varchar(45) DEFAULT NULL,
   `notes` varchar(1000) DEFAULT NULL,
   `visibility` varchar(45) NOT NULL DEFAULT 'PRIVATE',
   `documentType` varchar(45) NOT NULL,
   `restrictDelete` bit(1) NOT NULL,
   `customData` text,
   `releaseDate` datetime NOT NULL,
   `expirationDate` datetime NOT NULL,
   `isApproved` bit(1) NOT NULL DEFAULT b'0',
   `userSupplier` varchar(36) DEFAULT NULL,
   `complianceCertificateId` varchar(36) DEFAULT NULL,
   `Status` varchar(50) DEFAULT 'NEUTRAL',
   PRIMARY KEY (`id`),
   KEY `idx_client` (`clientId`)
 ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
 CREATE TABLE `multi_division_mapper` (
    `parent_id` varchar(36) NOT NULL,
    `division_id` varchar(36) NOT NULL,
    PRIMARY KEY (`parent_id`,`division_id`),
    KEY `idx_parent` (`parent_id`)
 ) ENGINE=InnoDB DEFAULT CHARSET=latin1;

P粉993712159P粉993712159288 days ago511

reply all(1)I'll reply

  • P粉226667290

    P粉2266672902024-03-29 13:06:40

    I was able to get a more favorable EXPLAIN report in my tests by creating the following index:

    ALTER TABLE multi_division_mapper
      DROP INDEX idx_parent,
      ADD INDEX (division_id, parent_id);

    I also removed idx_parent because it was redundant; it is a prefix for the primary key.

    id Choose a type surface Partition type Possible keys key key_len refer to OK Filtered additional
    1 Simple document null refer to idx_client idx_client 110 constant 1 100.00 Use temporary
    1 Simple department null refer to Main,division_id Department ID 38 constant 1 100.00 Use location;Use index;Unique

    reply
    0
  • Cancelreply