Laravel4查询克隆

时间:2015-01-05 21:45:25

标签: php laravel-4 clone

这个问题让我疯狂 自从我上次作曲家更新以来,克隆查询似乎无法正常工作 奇怪的是,如果我不对第一个查询克隆执行->get(),则第二个查询执行正常。从我在两个查询克隆上执行->get的那一刻起,我收到一个错误: General error: 2031 clone应创建原始查询的深层副本,但有些内容似乎不正确 有任何想法吗?这是我的查询(它看起来比它复杂,查询本身很好):

    //Query dates 
    $query_onetime = clone $query; 
    $query_onetime = $query_onetime->join('events_dates', function($join) use ($input_date_start, $input_date_end){ 
                $join->on('events.id', '=', 'events_dates.event_id'); 
                    $join->where('events_dates.start_date', "<=", $input_date_end); 
                    $join->where('events_dates.end_date', '>=', $input_date_start); 
            }); 

    //Select fields 
    $events_onetime = $query_onetime->select('events.id AS id', 'events.name AS name', 'event_categories.id as category_id', 'event_categories.category as category', 
                    'event_subcategories.id as subcategory_id', 'event_subcategories.subcategory as subcategory', 
                    'short_description', 'time_description', 
                    'price_description', 'nr_going', 'nr_checkedin', 'homepage', 'fbpage', 'profile_pic', 
                    'places.id AS place_id', 'places.name AS place_name', 'lat', 'lng', 'address')
            ->groupBy('events.id')
            ->get(); 

    if(isset($data["include_recurrent"]) && $data["include_recurrent"]){ 

        //Query recurrent dates 
        $query_recurrent = clone $query; 
        $query_recurrent = $query_recurrent->join('events_dates_recurrent', 'events.id', '=', 'events_dates_recurrent.event_id')
            ->where(function($join) use ($input_date_start, $input_date_end) { 
                //Create a dynamic query to get all recurrent dates within the input time interval 
                $query_string = "ABS(DATEDIFF('" . $input_date_start . "', CAST(events_dates_recurrent.start_date AS DATE)) % events_dates_recurrent.repeat_interval) = 0"; 
                $temp_date_start = $input_date_start; 

                while(strtotime($temp_date_start) < strtotime($input_date_end)){ 
                    $temp_date_start = date('Y-m-d', strtotime($temp_date_start . " +1 day")); 
                    //Create a raw query string 
                    $query_string = $query_string . " OR ABS(DATEDIFF('" . $temp_date_start . "', CAST(events_dates_recurrent.start_date AS DATE)) % events_dates_recurrent.repeat_interval) = 0"; 
                } 
                $join->whereRaw($query_string); 
            }); 

        //Select fields 
        $events_recurrent = $query_recurrent->select('events.id AS id', 'events.name AS name', 'event_categories.id as category_id', 'event_categories.category as category', 
                        'event_subcategories.id as subcategory_id', 'event_subcategories.subcategory as subcategory', 
                        'short_description', 'time_description', 
                        'price_description', 'nr_going', 'nr_checkedin', 'homepage', 'fbpage', 'profile_pic', 
                        'places.id AS place_id', 'places.name AS place_name', 'lat', 'lng', 'address')
                ->groupBy('events.id')
                ->get(); 

        $events = array_merge($events_onetime, $events_recurrent); 

编辑:有关要求的信息,这是完整的查询 我避免它,因为它很长。

    // Queries events based on map bounds, category and date 
    $query = DB::table('events')
            ->join('places', function($join) use ($data){ 
                $join->on('events.place_id', '=', 'places.id')
                        ->where('places.lat', '>', $data['sw_lat'])
                        ->where('places.lat', '<', $data['ne_lat'])
                        ->where('places.lng', '>', $data['sw_lng'])
                        ->where('places.lng', '<', $data['ne_lng']); 
            })->join('event_categories', function($join) use ($data){ 
                $join->on('events.category_id', '=', 'event_categories.id'); 
            }); 

    // The category id is optional 
    if(isset($data["category_id"])){ 
        $query = $query->where('event_categories.id', '=', $data['category_id']);  
    } 
    //Query subcategory 
    $query = $query->leftJoin('event_subcategories', function($join) use ($data){ 
                $join->on('events.subcategory_id', "=", "event_subcategories.id"); 
            }); 

    //Query keywords 
    $query = $query->join('events_keywords', 'events.id', '=', 'events_keywords.event_id'); 

    //Reverse date format 
    $input_date_start = date("Y-m-d", strtotime($data["date_start"])); 
    $input_date_end = date("Y-m-d", strtotime($data["date_end"])); 

    //Query dates 
    $query_onetime = clone $query; 
    $query_onetime = $query_onetime->join('events_dates', function($join) use ($input_date_start, $input_date_end){ 
                $join->on('events.id', '=', 'events_dates.event_id'); 
                    $join->where('events_dates.start_date', "<=", $input_date_end); 
                    $join->where('events_dates.end_date', '>=', $input_date_start); 
            }); 
    //Select fields 
    $events_onetime = $query_onetime->select('events.id AS id', 'events.name AS name', 'event_categories.id as category_id', 'event_categories.category as category', 
                    'event_subcategories.id as subcategory_id', 'event_subcategories.subcategory as subcategory', 
                    'short_description', 'time_description', 
                    'price_description', 'nr_going', 'nr_checkedin', 'homepage', 'fbpage', 'profile_pic', 
                    'places.id AS place_id', 'places.name AS place_name', 'lat', 'lng', 'address')
            ->groupBy('events.id')
            ->get(); 

    foreach($events_onetime as $event){ 
        $temp_event = EventModel::find($event->id); 
        $event->keywords = $temp_event->keywords; 
    } 

    if(isset($data["include_recurrent"]) && $data["include_recurrent"]){ 

        //Query recurrent dates 
        $query_recurrent = clone $query; 
        $query_recurrent = $query_recurrent->join('events_dates_recurrent', 'events.id', '=', 'events_dates_recurrent.event_id')
            ->where(function($join) use ($input_date_start, $input_date_end) { 
                //Create a dynamic query to get all recurrent dates within the input time interval 
                $query_string = "ABS(DATEDIFF('" . $input_date_start . "', CAST(events_dates_recurrent.start_date AS DATE)) % events_dates_recurrent.repeat_interval) = 0"; 
                $temp_date_start = $input_date_start; 

                while(strtotime($temp_date_start) < strtotime($input_date_end)){ 
                    $temp_date_start = date('Y-m-d', strtotime($temp_date_start . " +1 day")); 
                    //Create a raw query string 
                    $query_string = $query_string . " OR ABS(DATEDIFF('" . $temp_date_start . "', CAST(events_dates_recurrent.start_date AS DATE)) % events_dates_recurrent.repeat_interval) = 0"; 
                } 
                $join->whereRaw($query_string); 
            }); 

        //Select fields 
        $events_recurrent = $query_recurrent->select('events.id AS id', 'events.name AS name', 'event_categories.id as category_id', 'event_categories.category as category', 
                        'event_subcategories.id as subcategory_id', 'event_subcategories.subcategory as subcategory', 
                        'short_description', 'time_description', 
                        'price_description', 'nr_going', 'nr_checkedin', 'homepage', 'fbpage', 'profile_pic', 
                        'places.id AS place_id', 'places.name AS place_name', 'lat', 'lng', 'address')
                ->groupBy('events.id')
                ->get(); 

        //At this point we just have the events, but we also need the keywords per event. 
        foreach($events_recurrent as $event){ 
            $temp_event = EventModel::find($event->id); 
            $event->keywords = $temp_event->keywords; 
        } 
        $events = array_merge($events_onetime, $events_recurrent); 
    } else { 
        //Else return only the non-recurrent events 
        $events = $events_onetime; 
    }

    return $events; 
} 

1 个答案:

答案 0 :(得分:1)

您的$query对象是Illuminate\Database\Eloquent\Builder的实例,它维护对包含实际查询的Illuminate\Database\Query\Builder的引用。根据{{​​3}},clone关键字执行对象的浅表副本,这意味着将引用复制为引用。因此,$query$query_onetime都维护对同一Illuminate\Database\Query\Builder实例的引用,而对一个实例的更改会影响另一个实例。

需要克隆基础查询的此问题已在Laravel 4.1中修复,但如果您不想升级,可以尝试以下解决方法:

$query_onetime = clone $query;
$query_onetime->setQuery(clone $query->getQuery());
// Any additional joins specific to $query_onetime