Php 如何在Codeigniter中的数据库中插入数组?

Php 如何在Codeigniter中的数据库中插入数组?,php,codeigniter,Php,Codeigniter,我在Codeigniter的数据库中插入数组时遇到问题,我尝试了以下方法,但它给出了一个错误“消息:非法字符串偏移量‘年’”,“消息:非法字符串偏移量‘月’”,等等…,我真的很困惑如何解决这个问题,请帮助 表格如下: <form action="<?php echo base_url();?>hr/Home/store_attendance" method="post" id="student_attendance_entry&

我在Codeigniter的数据库中插入数组时遇到问题,我尝试了以下方法,但它给出了一个错误“消息:非法字符串偏移量‘年’”,“消息:非法字符串偏移量‘月’”,等等…,我真的很困惑如何解决这个问题,请帮助

表格如下:

<form action="<?php echo base_url();?>hr/Home/store_attendance" method="post"
id="student_attendance_entry">
<input type="hidden" name="year" value="<?php echo $year; ?>" />
<input type="hidden" name="month" value="<?php echo $month; ?>" />
<?php foreach($staffs as $staff): ?>
<table class="table table-striped table-bordered table-hover attendance_entry"
    style="border:initial;border: 1px solid #ddd;" id="sample_editable_2">
    <thead>
        <tr>
            <th style="width: 50px !important;">شماره</th>
            <th align="center" style="width:auto%">آی دی</th>
            <th align="center" style="width:15%">نام</th>
            <th align="center" style="width:15%;"> ولد</th>
            <th align="center" style="width:auto">حاضر</th>
            <th align="center" style="width:auto">غیر حاضر</th>
            <th align="center" style="width:auto">توضیحات</th>
        </tr>
    </thead>
    <tbody>
        <?php $a=1; foreach($staffs as $st):?>
        <tr>
            <td align="center" style="font-size: 11px"><?php echo $a;?></td>
            <td align="center" style="font-size: 11px"><?php echo $st['s_id'];?></td>
            <td align="center"><?php echo $st['dari_name'];?></td>
            <td align="center"><?php echo $st['dari_fName'];?></td>
            <td>
                <input type="number" min="0" class="form-control" name="total_present_day[]"
                    value="<?php if($st['total_present']==null){echo '';}else{ echo $st['total_present'];}?>"
                    data="<?php echo $a;?>">

                <input type="hidden" name="salary_type[]" id="salary_type" value="<?php echo $st['salary_type']?>">
            </td>
            <td>
                <input type="number" min="0" class="form-control" name="total_absent_day[]"
                    value="<?php if($st['absent']==null){echo '';}else{ echo $st['absent'];}?>"
                    data="<?php echo $a;?>">
                <input type="hidden" class="form-control" name="staff_id[]" value="<?php echo $st['s_id'];?>">

            </td>
            <td>
                <textarea min="0" class="form-control" name="memo[]"
                    colspan='3' rowspan="1"
                    data="<?php echo $a;?>"><?php if($st['memo']==null){echo '';}else{ echo $st['memo'];}?></textarea>
               

            </td>
        </tr>
        <?php $a++;endforeach;?>
    </tbody>
</table>

<?php endforeach; ?>
<br>
<div class="form-actions right">
    <a href="<?php echo base_url();?>student/school/attendance" class="btn default" data-dismiss="modal"><i
            class="fa fa-close"></i> بستن</a>
    <input type="submit" name="save" class="btn blue" value="ذخیره" />
</div>
这就是模型:

public function add_staff_attendance($data)
  {
      $this->db->trans_begin();

      foreach ($data['total_present_day'] as $key => $value) {
          {
          
  $dataToSave = array(
   'year' =>  $value['year'],
   'month' => $value['month'],
   'type_id'=>$value['salary_type'][$key],
   'total_present' => $value['total_present_day'][$key],
   'absent'=>$value['total_absent_day'][$key],
   'memo'=>$value['memo'][$key],
   'staf_id' => $value['staff_id'][$key]
 );
      
        $this->db->insert('staff_attendance', $dataToSave);
    }
          if ($this->db->trans_status() === false) {
              $this->db->trans_rollback();
              return false;
          } else {
              $this->db->trans_commit();
              return true;
          }
      }
}
我在控制器中调用了
$this->dd($data)
,这是输出:

array (
'Total' => 7,
)
array (
  'year' => '1400',
  'month' => '2',
  'staff_id' => 
     array (
       0 => '3',
     ),
   'total_present_day' => 
     array (
       0 => '26',
   ),
   'total_absent_day' => 
      array (
        0 => '0',
      ),
   'salary_type' => '1',
   'memo' => 
    array (
      0 => 'dds',
    ),
)

这是
echo'的结果。似乎错误在控制器中,您必须在保存数组时检查它

 Array
 (
   [year] => 1400
   [month] => 1
   [staff_id] => Array
     (
       [0] => 3
       [1] => 1
     )

[total_present_day] => Array
    (
        [0] => 26
        [1] => 20
    )

[total_absent_day] => Array
    (
        [0] => 0
        [1] => 6
    )

[salary_type] => Array
    (
        [0] => 1
        [1] => 1
    )

[memo] => Array
    (
        [0] => asfd
        [1] => saef
    )

    )
    Array
    (
    [year] => 1400
    [month] => 1
    [staff_id] => Array
    (
        [0] => 3
        [1] => 1
    )

[total_present_day] => Array
    (
        [0] => 26
        [1] => 20
    )

[total_absent_day] => Array
    (
        [0] => 0
        [1] => 6
    )

[salary_type] => Array
    (
        [0] => 1
        [1] => 1
    )

[memo] => Array
    (
        [0] => asfd
        [1] => saef
    )

 )
<?php foreach($staffs as $index => $staff) { ?>
    <input type="hidden" name="attendance[<?php echo $index; ?>]['year']">
    <input type="hidden" name="attendance[<?php echo $index; ?>]['month']">
    <input type="number" name="attendance[<?php echo $index; ?>]['total_present']">
    <input type="hidden" name="attendance[<?php echo $index; ?>]['type_id']">
    <input type="number" name="attendance[<?php echo $index; ?>]['absent']">
    <input type="hidden" name="attendance[<?php echo $index; ?>]['staf_id']">
    <textarea name="attendance[<?php echo $index; ?>]['memo']"></textarea>
<?php } ?>

您应该尝试类似的方法

您提交的数据是以逻辑的、最低限度的、但不规则的方式构建的。通过在视图中重新配置表单字段的
name
属性,可以大大减少控制器和模型中的处理代码行

移动
foeach()循环中的隐藏字段并声明动态索引。实际上,命名字段类似于:

public function store_attendance(): void
{
    // Definitely validate and sanitize the rows of data before passing to model.
    // This shortcut is for demonstrating how to batch insert
    if ($this->stuff_model->add_staff_attendance($this->input->post('attendance'))) {
        redirect(base_url() . 'hr/home/register_employee_attendance');
    } else {
        // reload view and explain what went wrong
    }
}
型号:

$presentDays = $this->input->post('total_present_day');
if ($presentDay) {
    $rows = [];
    foreach ($presentDays as $i => $presentDay) {
        $rows[] = [
            'year' => $this->input->post('year'),
            'month' => $this->input->post('month'),
            'staf_id' => $this->input->post('staff_id')[$i],
            'total_present' => $this->input->post('total_present_day')[$i],
            'absent' => $this->input->post('total_absent_day')[$i],
            'type_id' => $this->input->post('salary_type')[$i],
            'memo' => $this->input->post('memo')[$i]
        ];
    }
    
    if ($this->stuff_model->add_staff_attendance($rows)) {
        redirect(base_url() . 'hr/home/register_employee_attendance');
    } else {
        // reload view and explain what went wrong
    }
}
这些答案都没有经过测试


要将隐藏字段保留在循环之外,您需要在发送到
batch\u insert
ion的模型之前,将行的值组合到正确的结构中

public function store_attendance()
{   
    $data = array();
    $insert_att = 0;
    $tst = $this->input->post('total_present_day');

    for ($i=0; $i < count($tst); $i++) {
      
        $data = array(
            'year' =>  $this->input->post('year'),
            'month' => $this->input->post('month'),
            'staf_id' => $this->input->post('staff_id')[$i],
            'total_present' => $this->input->post('total_present_day')[$i],
            'absent'=>$this->input->post('total_absent_day')[$i],
            'type_id'=>$this->input->post('salary_type')[$i],
            'memo'=>$this->input->post('memo')[$i],
        );
        // var_dump($data);
        
        $this->get_public_data->saveData('staff_attendance', $data);
        
    }

    echo redirect(base_url().'hr/home/rgisterAttendance');
}

我通过更改控制器和模型中的一些代码来解决我的问题。 控制器:

 public function store_attendance()
{
    $data=array(

        'year' =>  $this->input->post('year'),
        'month' => $this->input->post('month'),
        'staff_id' => $this->input->post('staff_id'),
        'total_present_day' => $this->input->post('total_present_day'),
        'total_absent_day'=>$this->input->post('total_absent_day'),
        'salary_type'=>$this->input->post('salary_type'),
        'memo'=>$this->input->post('memo')
    );
    $this->dd($data);
    // $this->dd($class);
    $insert_att = $this->stuff_model->add_staff_attendance($data);
    // var_dump($insert_att);
    if($insert_att)
    {
        echo redirect(base_url().'hr/register_employee_attendance');
    }
   
}
public function add_staff_attendance(array $data): bool
{
    return $this->db->insert_batch('staff_attendance', $data);
}
 public function add_staff_attendance($data)
  {
  $this->db->insert('staff_attendance', $data);      
}

为什么要循环,您可以直接使用$data['year'],etc@Jerson因为数据库中存储的不仅仅是员工,有时可能不止一个人。正如您所知,您的标记是无效的,因为您正在生成多个具有相同的
id
sample\u editable\u 2
的表。无论您使用
data=“”
做什么,我都会删除它,如果有必要的话——找到一种更聪明的方法。
(以及类似的重复代码)更简单地表示为
…我不相信您甚至需要
?:''
。在这种情况下,如果控制器中的两个都指向同一个目标,为什么会有条件重定向?它解决了错误,但返回除year和monthI更新了答案之外的所有值的空值。请检查这个。为了方便起见,我已经对事情进行了注释。现在它插入了数据,年和月的值是正确的,但在其他列中插入2,它是不正确的。请在插入之前转储$dataToSave并共享结果。这是结果数组([year]=>1400[month]=>1[type_id]=>2[total_present]=>2[缺席]=>2[memo]=>2[staf_id]=>2)我不知道为什么值为2您无法将数组数据保存在行的列中。没有必要在循环中移动隐藏字段,我尝试了另一种方法,现在它完全按照我的要求工作。确实,不需要将隐藏字段移动到循环中就可以完成。要保存的行可以在控制器中形成正确的结构。围绕单个/单个插入方法使用事务毫无意义。由于故障而回滚时,仅回滚一行,批处理将部分保存。因此,删除所有
trans\uu
调用将具有相同的效果。提交完全成功或不保存行对应用程序是否重要?或者该批插入部分成功是否可以接受?我假设部分保存的提交是一件坏事。在每次迭代中调用
count()
是不必要的,因为计数永远不会改变。为什么要回显
重定向()
?您会注意到CodeIgniter故意创建了一个方法来容纳多行数据的插入--
insert\u batch()
public function store_attendance()
{   
    $data = array();
    $insert_att = 0;
    $tst = $this->input->post('total_present_day');

    for ($i=0; $i < count($tst); $i++) {
      
        $data = array(
            'year' =>  $this->input->post('year'),
            'month' => $this->input->post('month'),
            'staf_id' => $this->input->post('staff_id')[$i],
            'total_present' => $this->input->post('total_present_day')[$i],
            'absent'=>$this->input->post('total_absent_day')[$i],
            'type_id'=>$this->input->post('salary_type')[$i],
            'memo'=>$this->input->post('memo')[$i],
        );
        // var_dump($data);
        
        $this->get_public_data->saveData('staff_attendance', $data);
        
    }

    echo redirect(base_url().'hr/home/rgisterAttendance');
}
 public function add_staff_attendance($data)
  {
  $this->db->insert('staff_attendance', $data);      
}