Bit column vs relationships?

Bit column vs relationships?



Two examples:



Table users, each user may have or not have some of ~10 possible permissions (like "can login, can post, can edit, can delete" etc).


users



Table articles, each article may have or not have some of ~10 possible attributes (like "for children", "18+", "less than 10 minutes to read", "long read").


articles



What is the canonical more efficient fastest way to store it in the database?



I can think of two ways:



Option 1:
Bit-like kind of column, have column "permissions""attributes" and store a value which can be bit-interpreted like "0101100010", i-th bit is the flag for i-th attributepermission



Option 2:
Relationship. Create table user_permissions, put 10 values with their IDs there, then create table user_permissions_map and keep this many to many (M:N) relationship in this table.


user_permissions


user_permissions_map



I'm a bit afraid of option 2 because it looks like it will require additional querieslookups in permission_map table each time I need to check user's permission. When with option 1 it's just a column belonging to the user, makes it much simpler to check the permissions.


permission_map




2 Answers
2



The bit-like field will make it harder to find all users or articles with a certain property. For users/permissions, that's not much of a problem (unless you want a list of e.g. all administrators). For articles, I can imagine that you want to filter on certain attributes (e.g. only non-18+ articles). You also have the option to store those fields (option 3) as single bit columns; then, no extra table is needed.



Another consideration: what happens when you want to add an additional permission/attribute which is by default set to '1'/'true'? For the bitmask field (option 1), this can be done by adding the right power of 2 to all values in column permissions; for option 2, you need a mass INSERT in user_permissions_map; adding an extra bit column with ADD COLUMN and DEFAULT true (option 3) is clear and easy.


permissions


INSERT


user_permissions_map


ADD COLUMN


DEFAULT true






I haven't really got how do you suggest to do this without extra table and what is your "option 3". Could you please provide more details?

– The Godfather
Sep 18 '18 at 16:36






Option 3 is adding extra columns, e.g. ALTER TABLE articles ADD COLUMN is_18_plus BIT NOT NULL DEFAULT VALUE true.

– Glorfindel
Sep 18 '18 at 16:40


ALTER TABLE articles ADD COLUMN is_18_plus BIT NOT NULL DEFAULT VALUE true






I think, for option 2 "negative" attributes it can be done via some additional column in user_permissions table like is_true_by_default (like property for attribute). However it will still require some backend logic changed. Adding columns to articles looks like non-intuitive solution as for me. First, it will spam the database with 10+ extra columns for each attribute (if use column for each attribute). Second, it can make backend logic complicated (if use column only for "negative" attribute and keep "positive" attributes in separated table)

– The Godfather
Sep 18 '18 at 16:49


user_permissions


is_true_by_default


articles






@TheGodfather Adding Y/N columns to a table shouldn't have that much of an impact on it as the data is small in footprint and will be easy to store.

– Joe W
Sep 18 '18 at 17:05







One thing to consider is how often would you be adding in a new permission/attribute?

– Joe W
Sep 18 '18 at 17:08



I would suggest doing a different design for each of those examples as I don't think that one design fits all cases even within a database design. In the case of user permissions that is something that is unlikely to change often but you may want to add article attributes more frequently as the site expands.



For user permissions I would suggest a simple user table with a column for each permission. Once it is set up it is unlikely that new permissions will be added and a simple table will allow for easy access to the data.


user_id NUMBER,
permission_1 VARCHAR2(1 CHAR) default 'N',
permission_2 VARCHAR2(1 CHAR) default 'N'



For article on the other hand I would suggest a mapping table so that it is easier to add new attributes as needed. In fact it would be possible for someone with the correct permissions to add a new attribute as they submitted an article that needed one.


attributes
attribute_id NUMBER,
attribute_description VARCHAR2(4000 CHAR)

article_attribute
article_id NUMBER,
attribute_id NUMBER



Thanks for contributing an answer to Database Administrators Stack Exchange!



But avoid



To learn more, see our tips on writing great answers.



Required, but never shown



Required, but never shown




By clicking “Post Your Answer”, you agree to our terms of service, privacy policy and cookie policy

Popular posts from this blog

𛂒𛀶,𛀽𛀑𛂀𛃧𛂓𛀙𛃆𛃑𛃷𛂟𛁡𛀢𛀟𛁤𛂽𛁕𛁪𛂟𛂯,𛁞𛂧𛀴𛁄𛁠𛁼𛂿𛀤 𛂘,𛁺𛂾𛃭𛃭𛃵𛀺,𛂣𛃍𛂖𛃶 𛀸𛃀𛂖𛁶𛁏𛁚 𛂢𛂞 𛁰𛂆𛀔,𛁸𛀽𛁓𛃋𛂇𛃧𛀧𛃣𛂐𛃇,𛂂𛃻𛃲𛁬𛃞𛀧𛃃𛀅 𛂭𛁠𛁡𛃇𛀷𛃓𛁥,𛁙𛁘𛁞𛃸𛁸𛃣𛁜,𛂛,𛃿,𛁯𛂘𛂌𛃛𛁱𛃌𛂈𛂇 𛁊𛃲,𛀕𛃴𛀜 𛀶𛂆𛀶𛃟𛂉𛀣,𛂐𛁞𛁾 𛁷𛂑𛁳𛂯𛀬𛃅,𛃶𛁼

Edmonton

Crossroads (UK TV series)