Массовая вставка в Laravel с использованием красноречивого ORM
Как мы можем выполнять массовые вставки базы данных в Laravel, используя Eloquent ORM?
Я хочу сделать это в Laravel: /questions/13144718/php-massovaya-vstavka-foreach/13144730#13144730 но я получаю следующую ошибку.
SQLSTATE [HY093]: недопустимый номер параметра: смешанные именованные и позиционные параметры.
13 ответов
Вы можете просто использовать Eloquent::insert()
,
Например:
$data = array(
array('name'=>'Coder 1', 'rep'=>'4096'),
array('name'=>'Coder 2', 'rep'=>'2048'),
//...
);
Coder::insert($data);
Мы можем обновить ответ GTF, чтобы легко обновлять временные метки.
$data = array(
array(
'name'=>'Coder 1', 'rep'=>'4096',
'created_at'=>date('Y-m-d H:i:s'),
'modified_at'=> date('Y-m-d H:i:s')
),
array(
'name'=>'Coder 2', 'rep'=>'2048',
'created_at'=>date('Y-m-d H:i:s'),
'modified_at'=> date('Y-m-d H:i:s')
),
//...
);
Coder::insert($data);
Обновление: чтобы упростить дату, мы можем использовать углерод, как предложил @Pedro Moreira.
$now = Carbon::now('utc')->toDateTimeString();
$data = array(
array(
'name'=>'Coder 1', 'rep'=>'4096',
'created_at'=> $now,
'modified_at'=> $now
),
array(
'name'=>'Coder 2', 'rep'=>'2048',
'created_at'=> $now,
'modified_at'=> $now
),
//...
);
Coder::insert($data);
ОБНОВЛЕНИЕ 2: для Laravel 5, используйте updated_at
вместо modified_at
$now = Carbon::now('utc')->toDateTimeString();
$data = array(
array(
'name'=>'Coder 1', 'rep'=>'4096',
'created_at'=> $now,
'updated_at'=> $now
),
array(
'name'=>'Coder 2', 'rep'=>'2048',
'created_at'=> $now,
'updated_at'=> $now
),
//...
);
Coder::insert($data);
Вот как вы делаете это более красноречивым образом,
$allintests = [];
foreach($intersts as $item){ //$intersts array contains input data
$intestcat = new User_Category();
$intestcat->memberid = $item->memberid;
$intestcat->catid= $item->catid;
$allintests[] = $intestcat->attributesToArray();
}
User_Category::insert($allintests);
Для тех, кто читает это, проверьте createMany()
метод.
/**
* Create a Collection of new instances of the related model.
*
* @param array $records
* @return \Illuminate\Database\Eloquent\Collection
*/
public function createMany(array $records)
{
$instances = $this->related->newCollection();
foreach ($records as $record) {
$instances->push($this->create($record));
}
return $instances;
}
Я искал много раз, наконец, использовал обычай timestamps
как ниже:
$now = Carbon::now()->toDateTimeString();
Model::insert([
['name'=>'Foo', 'created_at'=>$now, 'updated_at'=>$now],
['name'=>'Bar', 'created_at'=>$now, 'updated_at'=>$now],
['name'=>'Baz', 'created_at'=>$now, 'updated_at'=>$now],
..................................
]);
Eloquent::insert
является правильным решением, но оно не будет обновлять временные метки, так что вы можете сделать что-то вроде ниже
$json_array=array_map(function ($a) {
return array_merge($a,['created_at'=>
Carbon::now(),'updated_at'=> Carbon::now()]
);
}, $json_array);
Model::insert($json_array);
Идея состоит в том, чтобы добавить create_at и updated_at во весь массив перед выполнением вставки
Начиная с Laravel 5.7 с Illuminate\Database\Query\Builder
вы можете использовать метод insertUsing.
$query = [];
foreach($oXML->results->item->item as $oEntry){
$date = date("Y-m-d H:i:s")
$query[] = "('{$oEntry->firstname}', '{$oEntry->lastname}', '{$date}')";
}
Builder::insertUsing(['first_name', 'last_name', 'date_added'], implode(', ', $query));
$start_date = date('Y-m-d h:m:s');
$end_date = date('Y-m-d h:m:s', strtotime($start_date . "+".$userSubscription['duration']." months") );
$user_subscription_array = array(
array(
'user_id' => $request->input('user_id'),
'user_subscription_plan_id' => $request->input('subscription_plan_id'),
'name' => $userSubscription['name'],
'description' => $userSubscription['description'],
'duration' => $userSubscription['duration'],
'start_datetime' => $start_date,
'end_datetime' => $end_date,
'amount' => $userSubscription['amount'],
'invoice_id' => '',
'transection_datetime' => '',
'created_by' => '1',
'status_id' => '1', ),
array(
'user_id' => $request->input('user_id'),
'user_subscription_plan_id' => $request->input('subscription_plan_id'),
'name' => $userSubscription['name'],
'description' => $userSubscription['description'],
'duration' => $userSubscription['duration'],
'start_datetime' => $start_date,
'end_datetime' => $end_date,
'amount' => $userSubscription['amount'],
'invoice_id' => '',
'transection_datetime' => '',
'created_by' => '1',
'status_id' => '1', )
);
dd(UserSubscription::insert($user_subscription_array));
UserSubscription
это название моей модели Это вернет "true", если вставить успешно, иначе "false".
Возможно, более простой способ решить эту проблему - использовать коллекцию и вставить ее в цикл, используя модель с использованием меток времени.
<?php
use App\Continent;
use Illuminate\Database\Seeder;
class InitialSeeder extends Seeder
{
/**
* Run the database seeds.
*
* @return void
*/
public function run()
{
collect([
['name' => 'América'],
['name' => 'África'],
['name' => 'Europa'],
['name' => 'Asia'],
['name' => 'Oceanía'],
])->each(function ($item, $key) {
Continent::forceCreate($item);
});
}
}
РЕДАКТИРОВАТЬ:
Извините за мое недоразумение. Для массовой вставки это может помочь, и, возможно, с этим вы можете сделать хорошие сеялки и немного их оптимизировать.
<?php
use App\Continent;
use Carbon\Carbon;
use Illuminate\Database\Seeder;
class InitialSeeder extends Seeder
{
/**
* Run the database seeds.
*
* @return void
*/
public function run()
{
$timestamp = Carbon::now();
$password = bcrypt('secret');
$continents = [
[
'name' => 'América'
'password' => $password,
'created_at' => $timestamp,
'updated_at' => $timestamp,
],
[
'name' => 'África'
'password' => $password,
'created_at' => $timestamp,
'updated_at' => $timestamp,
],
[
'name' => 'Europa'
'password' => $password,
'created_at' => $timestamp,
'updated_at' => $timestamp,
],
[
'name' => 'Asia'
'password' => $password,
'created_at' => $timestamp,
'updated_at' => $timestamp,
],
[
'name' => 'Oceanía'
'password' => $password,
'created_at' => $timestamp,
'updated_at' => $timestamp,
],
];
Continent::insert($continents);
}
}
Или при использовании фабрики Laravel (создание 10 пользователей):
$users = factory(App\User::class, 10)->make()->toArray();
App\User::insert($users);
Для вставки отношений категории я столкнулся с той же проблемой и понятия не имел, за исключением того, что в моей красноречивой модели я использовал Self(), чтобы иметь экземпляр того же класса в foreach для записи нескольких сохранений и идентификаторов захвата.
foreach($arCategories as $v)
{
if($v>0){
$obj = new Self(); // this is to have new instance of own
$obj->page_id = $page_id;
$obj->category_id = $v;
$obj->save();
}
}
без "$obj = new Self()" он сохраняет только одну запись (когда $ obj был $this)
РЕШЕННАЯ ПРОБЛЕМА............. ИЗМЕНЕНИЕ ТАБЛИЦЫ ДЛЯ МИГРАЦИИ...
$table->timestamp('created_at')->nullable()->useCurrent();
ЛЕГКО
Schema::create('spider_news', function (Blueprint $table) {
$table->bigIncrements('id');
$table->string('source')->nullable();
$table->string('title')->nullable();
$table->string('description')->nullable();
$table->string('daterss')->nullable();
$table->timestamp('created_at')->nullable()->useCurrent();
$table->timestamp('update_at')->nullable()->useCurrent();
});
Вы можете создать по одному и получить возвращаемую коллекцию созданного экземпляра модели.
$createdCollection= collect([
['name' => 'América'],
['name' => 'África'],
['name' => 'Europa'],
['name' => 'Asia'],
['name' => 'Oceanía'],
])->map(function ($item, $key) {
return Continent::create($item);
});