dongtaotao19830418 2016-11-17 17:36
浏览 44
已采纳

SilverStripe唯一URL(调试)

In this DataObject there is a user supplied field Title which has to be converted to a unique URL slug.

Desired Result: Duplicate URL's should get a suffix to its value. So saving 2 records with Title Foo should result in one record with foo as its value for column URL and the second record should have value foo-2 for the same column.

public function onBeforeWrite() {
    parent::onBeforeWrite();

    // Sanitize Title field to use for URL
    $filter = URLSegmentFilter::create();
    $this->URL = $filter->filter($this->Title);

    // If URL is not unique, add suffix
    $i = 1;
    while($this->uniqueURL($this->URL)) {
        $i++;
        $this->URL = $this->URL . "-" . $i;
    }   
}

method: uniqueURL (within same class)

public function uniqueURL($URL) {

    // Check if there is a record with the same URL
    $existingURL = DataObject::get('NewsArticle', "URL = '$URL'");
    if ($existingURL) {
        // this is a duplicate URL
        return false;
    } else {
        // this is a unique url
        return true;
    }
}

Saving Foo twice would result in foo and foo-2.

When saving two records with the same Title Foo results in two URL fields with foo

  • 写回答

1条回答 默认 最新

  • douwu0335 2016-11-17 17:53
    关注

    Why do you have two foo urls?

    If you check your DB before inserting all records, this means that the check will not work on your record batch.

    Don't use a loop to count unique urls

    You don't need to loop and check every time and increment the count ($i). Performance wise youre far better off doing a COUNT() in a query and just use that value for your next insert.

    // The following does exactly the same with just 1 query. No loop needed.
    $count = DB::query("SELECT COUNT(*) FROM Table WHERE Title LIKE '{$filteredTitle}'")->value();
    if ($count > 1) {
        $filteredTitle .= "-" .  $count;
    }
    $this->URL = $filteredTitle
    

    Solutions

    To do it onBeforeWrite() the only possibility is to Query your data AND check your records before they are saved.

    Or a simpler solution with the same results is that you can change the url in an onAfterWrite() , and check use the amount of same titles as number.

    public function onAfterWrite() {
        parent::onAfterWrite();
    
        // Sanitize Title field to use for URL
        $filter = URLSegmentFilter::create();
        $filteredTitle= $filter->filter($this->Title);
    
        $count = DB::query("SELECT COUNT(*) FROM Table WHERE Title LIKE '{$filteredTitle}'")->value();
        if ($count > 1) {
            $filteredTitle .= "-" .  $count;
        }
        $this->URL = $filteredTitle
    }
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 目前主流的音乐软件,像网易云音乐,QQ音乐他们的前端和后台部分是用的什么技术实现的?求解!
  • ¥60 pb数据库修改与连接
  • ¥15 spss统计中二分类变量和有序变量的相关性分析可以用kendall相关分析吗?
  • ¥15 拟通过pc下指令到安卓系统,如果追求响应速度,尽可能无延迟,是不是用安卓模拟器会优于实体的安卓手机?如果是,可以快多少毫秒?
  • ¥20 神经网络Sequential name=sequential, built=False
  • ¥16 Qphython 用xlrd读取excel报错
  • ¥15 单片机学习顺序问题!!
  • ¥15 ikuai客户端多拨vpn,重启总是有个别重拨不上
  • ¥20 关于#anlogic#sdram#的问题,如何解决?(关键词-performance)
  • ¥15 相敏解调 matlab