- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我的用户可以更新他们的信息,这些信息保存在表中定义数量的列中,例如:user ( id INT, email VARCHAR, phone VARCHAR, address VARCHAR)
,例如.
我见过其他实现,例如 Wordpress 的实现,它为用户将此信息存储在一个名为 usermeta
的表中,其布局为 ( umeta_id INT, user_id INT, meta_key VARCHAR,元值 VARCHAR )
.
在我想要实现的更改日志中,我正在评估是使用这样的解决方案还是制作(我认为会更好的)布局,例如:userLog ( id INT, date TIMESTAMP, email VARCHAR,电话 VARCHAR,地址 VARCHAR)
。
因此,我可以获得任何用户在给定日期拥有的所有信息的历史记录。行将只记录更改,在未更改的列上具有 NULL。
对于第一个问题:除了能够通过插入适当的meta_key
来创建新的信息类型之外,这种布局还有什么优势吗?
我有时认为,如果我的环境需要考虑性能,那么这种布局可能不太合适,因为我会为我要存储的每种数据使用 VARCHAR
。
对于第二个问题:存储和选择/插入效率真的能影响我正在考虑的两种解决方案吗?
哪个解决方案比另一个解决方案占用空间更少(或更多)和/或选择/插入效率更低(或更高),为什么?
最佳答案
一些想法,如果不一定是答案:
显然更改日志对您来说是必不可少的,因此每个用户一行的原始结构不适合您。所以我们谈论的是以下选择:
解决方案 1 对应于您的
userLog ( id INT, date TIMESTAMP, email VARCHAR, phone VARCHAR, address VARCHAR )
方案二对应Wordpress方案一:
umeta_id INT, user_id INT, meta_key VARCHAR, meta_value VARCHAR
您的问题 1: 我看不出 Solution2 有任何优势,除非您随后决定要捕获用户的(例如)网站 URL 或(例如)最喜欢的颜色作为好吧,你可以通过添加一个 meta_key 来做到这一点。但是您同样可以在 Solution1 下轻松地执行此操作,只需执行一个
ALTER TABLE userlog ADD COLUMN WebSiteURL(etc)
这并不难做到。除非您公司中的 DBA 非常像杜宾犬 (;))。因为您持有更改日志,所有现有用户(在更改时)现在将有一个空白的 WebsiteURL 列;但这正是您想要的:您不知道他们的 WebsiteURL,因为系统之前没有捕获它。当然,新列必须是 NULLABLE - 但无论如何这可能是不可避免的,即使使用“初始”数据,除非您用来捕获用户信息的方法坚持将电子邮件、电话和地址列为必需的列。
对我来说,meta_key 解决方案的缺点大于优点。缺点是:
您必须开发一段数据透视代码,将一个用户的用户信息转换为另一个用户
排。您必须在要在一行中获取用户信息的每个地方调用此代码。在相比之下,Solution1只需要
SELECT userID,[所有用户信息] FROM userLog INNER JOIN (SELECT userID,MAX(datechanged) AS LatestDAteChanged FROM userlog GROUP BY userID) a ON userlog.userid=a.userID AND userlog.DateChanged=a.LatestDAteChanged
这比枢轴更有效。使用 UserID、DateChanged 的索引,这将奔跑如风。
除非您真的想在 userinfo 表(Email、Email、Email、Email、Email)中多次保存 meta_key 值,否则您需要一个额外的 Meta_Key_Lookup 表。
第二个问题:对于最终的空间效率,是的,meta_key Solution2 是最好的。特别是如果您不使用 VARCHAR 元键,而是使用元键 ID 值,并且有一个单独的元键查找表(例如 1=Email,2=Phone 等)。但我认为这不是 meta_key 解决方案 2 的决定性论据,因为存储价格几乎为零,而且该解决方案涉及困难。
(注意/想法:恕我直言,您在解决方案 1 中保留 NULL 值的想法是一条错误的道路。尝试获取最新电子邮件的编码,然后是电话,然后是地址(分别) 对于每个用户来说,这将是一场噩梦:几乎与其他解决方案所需的枢轴一样难以编码/测试 - 以及服务器运行 - 以及存储边际的减少。每次做一件事时只保留整行变化。除非你只是举个例子,真正的用户信息集是 50 列宽...)
恕我直言,存储问题不是决定性的。那么让我们转向 SELECT/INSERT 效率:
在这个问题上,我认为还是Solution1胜出。在 Inserts 上,SOlution1 获胜:仅插入一行,即使用户更改了其信息中的每个字段。在 SELECTS 上,解决方案 1 再次获胜:您只需要查看每个用户的最新信息(上面的代码),这是 SQL 优化的类型。相比之下,解决方案 2 需要一个支点:SQL 不擅长的东西。
关于mysql - 用户数据和更改日志的哪种布局最有效且存储消耗更少?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13364790/
我有一个功能是转换 ADO Recordset 进入html: class function RecordsetToHtml(const rs: _Recordset): WideString; 该函
经过几天的研究和讨论,我想出了这种方法来收集访客的熵(你可以看到我的研究历史here) 当用户访问时,我运行此代码: $entropy=sha1(microtime().$pepper.$_SERVE
给定一个无序列表 List ,我需要查找是否存在 String与提供的字符串匹配。 所以,我循环 for (String k : keys) { if (Utils.keysM
我已经搜索过这个问题,但没有找到我正在寻找的答案。 基本上,我想将类构造函数包装在 try/except 子句中,以便它忽略构造函数内特定类型的错误(但无论如何都会记录并打印它们)。我发现做到这一点的
我有一组三个数字,我想将一组数字与另一组数字进行比较。即,第一组中的每个数字小于另一组中的至少一个数字。需要注意的是,第一组中的下一个数字必须小于第二组中的不同数字(即,{6,1,6} 对 {8,8,
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 9 年前。 Improve this
首先介绍一下背景: 我正在开发一个带有 EJB 模块和应用程序客户端模块的企业应用程序 (ear)。我还使用 hibernate JPA 来实现持久性,并使用 swingx 来实现 GUI。这些是唯一
我正在尝试在我的上网本上运行 Eclipse 以便能够为 Android 进行开发。 您可能已经猜到了,Eclipse 非常慢,并且不容易有效地开发。 我正在使用 Linux Ubuntu 并且我还有
for row, instrument in enumerate(instruments): for col, value in enumerate(instrument):
return not a and not b ^ 我如何以更好的格式表达它 最佳答案 DeMorgan's Law , 也许? return not (a or b) 我认为在这一点上已经足够简单了
我正在尝试让 Font Awesome 图标看起来更 slim https://jsfiddle.net/cliffeee/7L6ehw9r/1/ . 我尝试使用“-webkit-text-strok
假设我有一个名为 vals 的数据框,如下所示: id…………日期…………min_date…… .........最大日期 1…………2016/01/01…………2017/01/01…………2018/
是否有更 Pythonic 的方式来做到这一点?: if self.name2info[name]['prereqs'] is None: se
我有一个函数可以将一些文本打印到它接收到的 ostream&。如果 ostream 以终端为目标,我想让它适应终端宽度,否则默认为某个值。 我现在做的是: 从 ostream 中获取一个 ofstre
这个问题在这里已经有了答案: Should a retrieval method return 'null' or throw an exception when it can't produce
我有这个 bc = 'off' if c.page == 'blog': bc = 'on' print(bc) 有没有更 Pythonic(和/或更短)的方式在 Python 中编写? 最佳
输入:一个包含 50,000 行的 CSV;每行包含 910 列值 0/1。 输出:运行我的 CNN 的数据框。 我编写了一个逐行读取 CSV 的代码。对于每一行,我将数据分成两部分,称为神经元(90
据我所知,with block 会在您退出 block 后自动调用 close(),并且它通常用于确保不会忘记关闭一个文件。 好像没有技术上的区别 with open(file, 'r+') as f
我有一个使用 Entity Framework V6.1.1 的 MVC 5 网站。 Entity Framework DbContext 类和模型最初都在网站项目中。这个项目有 3 个 DbCont
我是编程新手,在尝试通过将 tableView 和关联 View 的创建移动到单独的类并将委托(delegate)和数据源从 VC 移动到单独的类来精简我的 ViewController 时遇到了一些
我是一名优秀的程序员,十分优秀!