对于我目前正在编写的应用程序,我需要创建链接。链接由文章名称和类别名称组成,必须在我想要使用链接的所有地方都这样做。重复该代码对我来说很难维护并且容易出错。因此,我希望此链接/url 是表/类中的虚拟列。
我确实找到了一个可以添加虚拟列的模块,但文档指出这只是一个“概念验证”,已有 10 多年历史,并且存在未解决的问题。DBIx ::Class::VirtualColumns
还发现了一种使用自定义方法扩展结果集的方法: 预定义搜索和创建结果集类
两者都使用预定义搜索的方法。
因此我创建了一个结果集目录和以下模块
package MySite::Schema::ResultSet::Article;
use strict;
use warnings;
use Data::Dumper;
use base 'DBIx::Class::ResultSet';
sub returnURL {
my ($self, $which) = @_;
my $blaat =$self->find(
{article_id => $which}
);
my $result = $blaat->slug;
print "\n>>$result<<\n";
return $result;
}
1;
出于调试目的调用此方法(在 Dancer2 应用程序中):
debug schema.resultset('Article')->returnURL(2);
输出结果如下
>>artikel_2<<
[MySite:94935] debug @2024-11-25 10:08:13> MySite::Schema=HASH(0x6216cb9aae60)artikel_2 in /var/www/MySite/bin/../lib/MySite.pm l. 24
第一行正是我所期望的。第二行中的“MySite::Schema=HASH(0x6216cb9aae60)”不是。这是从哪里来的?
此外:我在 TT 模板中循环文章并尝试获取方法的结果:
<% url1 = schema.resultset('Article').returnURL(1) %>
<% url2 = article.returnURL(1) %>
>><% url1 %><<
>><% url2 %><<
(是的,文章索引现在是硬编码的)url1 和 url2 都返回空。
我很确定我正在做一些非常研究的事情,只是不知道是什么。
按照 Dave 的建议尝试过(我认为)
将该方法移至 wat 为我创建的类中:
sub returnURL {
print ">>blaat<<";
my ($self, $which) = shift;
# my $blaat =$self->find(
# {article_id => $which}
# );
# my $result = $blaat->slug;
# print "\n>>$result<<\n";
# return $result;
return("hey there ".$self->slug);
}
在 Perl 代码中调用该方法会返回错误。
Can't locate object method "returnURL" via package "DBIx::Class::ResultSet" at /var/www/MySite/bin/../lib/MySite.pm line 24.
所以我想,谁在乎呢……无论如何模板中都需要它。然后注释掉了它。之后我在模板中使用的第二种方法确实返回了一个结果:
<% article.returnURL() %>
给了我
hey there artikel_4
这正是我所期望的,也是我可以使用的东西。结果我甚至不需要提供文章索引。
谢谢你戴夫
我觉得你想多了。这应该是你添加到 Result 类的方法。
更新:我意识到我原来的回答很简短。所以他提供了一些更有用的信息。
“结果”类基本上模拟了数据库表中的一行。如果您在结果类中的方法中,那么
$self
您拥有的变量就是表中的一行。它将具有与表中各个列匹配的方法。因此,这是添加方法以扩展您拥有的有关单行的信息的明显位置。例如,想象一下,一张包含一
date_of_birth
列的人员表 - 但您想要该人员的年龄。在结果类中,您创建了一个名为的新方法age()
。在其中,您可以从当前日期中减去出生日期以获取年龄:您的
url()
方法正是基于同样的原理。另一方面,“ResultSet”类是您想要放置巧妙的搜索方法的地方。这些方法返回表中行的子集。按照我们个人的示例,您可能需要返回所有男性或所有女性的搜索方法。