读取新闻条目

在上一节中,我们通过写出一个包含静态页面的类了解了一些框架的基本概念,我们也根据自定义路由规则来重定向 URI 。现在是时候向大家介绍动态内容和如何使用数据库了。

创建你的数据模型

数据库的查询操作应该放在模型里,而不是写在控制器里,这样可以很方便的重用它。模型正是用于从数据库或者其他存储中获取、新增、更新数据的地方。它就代表你的数据。

打开 application/models/ 目录,新建一个文件 News_model.php ,然后写入下面的代码。确保你的 数据库配置 正确。

  1. <?php
  2. class News_model extends CI_Model {
  3.  
  4. public function __construct()
  5. {
  6. $this->load->database();
  7. }
  8. }

这个代码和之前的控制器的代码有点类似,它通过继承 CI_Model 创建了一个新模型,并加载了数据库类。数据库类可以通过 $this->db 对象访问。

在查询数据库之前,我们要先创建一个数据库表。连接你的数据库,运行下面的 SQL 语句(MySQL),并添加一些测试数据。

  1. CREATE TABLE news (
  2. id int(11) NOT NULL AUTO_INCREMENT,
  3. title varchar(128) NOT NULL,
  4. slug varchar(128) NOT NULL,
  5. text text NOT NULL,
  6. PRIMARY KEY (id),
  7. KEY slug (slug)
  8. );

现在,数据库和模型都准备好了,你需要一个方法来从数据库中获取所有的新闻文章。为实现这点,我们使用了 CodeIgniter 的数据库抽象层 查询构造器 ,通过它你可以编写你的查询代码,并在 所有支持的数据库平台 上运行。向你的模型中添加如下代码。

  1. public function get_news($slug = FALSE)
  2. {
  3. if ($slug === FALSE)
  4. {
  5. $query = $this->db->get('news');
  6. return $query->result_array();
  7. }
  8.  
  9. $query = $this->db->get_where('news', array('slug' => $slug));
  10. return $query->row_array();
  11. }

通过这个代码,你可以执行两种不同的查询,一种是获取所有的新闻条目,另一种是根据它的 slug 来获取新闻条目。你应该注意到,$slug 变量在执行查询之前并没有做检查, 查询构造器 会自动帮你检查的。

显示新闻

现在,查询已经写好了,接下来我们需要将模型绑定到视图上,向用户显示新闻条目了。这可以在之前写的 Pages 控制器里来做,但为了更清楚的阐述,我们定义了一个新的News 控制器,创建在 application/controllers/News.php 文件中。

  1. <?php
  2. class News extends CI_Controller {
  3.  
  4. public function __construct()
  5. {
  6. parent::__construct();
  7. $this->load->model('news_model');
  8. $this->load->helper('url_helper');
  9. }
  10.  
  11. public function index()
  12. {
  13. $data['news'] = $this->news_model->get_news();
  14. }
  15.  
  16. public function view($slug = NULL)
  17. {
  18. $data['news_item'] = $this->news_model->get_news($slug);
  19. }
  20. }

阅读上面的代码你会发现,这和之前写的代码有些相似之处。首先是 _construct()方法,它调用父类(CI_Controller)中的构造函数,并加载模型。这样模型就可以在这个控制器的其他方法中使用了。另外它还加载了 [_URL 辅助函数]($7f84ca71ccd5c49d.md) ,因为我们在后面的视图中会用到它。

其次,有两个方法用来显示新闻条目,一个显示所有的,另一个显示特定的。你可以看到第二个方法中调用模型方法时传入了 $slug 参数,模型根据这个 slug返回特定的新闻条目。

现在,通过模型,控制器已经获取到数据了,但还没有显示。下一步要做的就是,将数据传递给视图。

  1. public function index()
  2. {
  3. $data['news'] = $this->news_model->get_news();
  4. $data['title'] = 'News archive';
  5.  
  6. $this->load->view('templates/header', $data);
  7. $this->load->view('news/index', $data);
  8. $this->load->view('templates/footer');
  9. }

上面的代码从模型中获取所有的新闻条目,并赋值给一个变量,另外页面的标题赋值给了$data['title'] 元素,然后所有的数据被传递给视图。现在你需要创建一个视图文件来显示新闻条目了,新建 application/views/news/index.php 文件并添加如下代码。

  1. <h2><?php echo $title; ?></h2>
  2.  
  3. <?php foreach ($news as $news_item): ?>
  4.  
  5. <h3><?php echo $news_item['title']; ?></h3>
  6. <div class="main">
  7. <?php echo $news_item['text']; ?>
  8. </div>
  9. <p><a href="<?php echo site_url('news/'.$news_item['slug']); ?>">View article</a></p>
  10.  
  11. <?php endforeach; ?>

这里,通过一个循环将所有的新闻条目显示给用户,你可以看到我们在 HTML 模板中混用了 PHP ,如果你希望使用一种模板语言,你可以使用 CodeIgniter 的 模板解析类 ,或其他的第三方解析器。

新闻的列表页就做好了,但是还缺了显示特定新闻条目的页面,之前创建的模型可以很容易的实现该功能,你只需要向控制器中添加一些代码,然后再新建一个视图就可以了。回到 News控制器,使用下面的代码替换掉 view() 方法:

  1. public function view($slug = NULL)
  2. {
  3. $data['news_item'] = $this->news_model->get_news($slug);
  4.  
  5. if (empty($data['news_item']))
  6. {
  7. show_404();
  8. }
  9.  
  10. $data['title'] = $data['news_item']['title'];
  11.  
  12. $this->load->view('templates/header', $data);
  13. $this->load->view('news/view', $data);
  14. $this->load->view('templates/footer');
  15. }

我们并没有直接调用 getnews() 方法,而是传入了一个 $slug 参数,所以它会返回相应的新闻条目。最后剩下的事是创建视图文件_application/views/news/view.php 并添加如下代码 。

  1. <?php
  2. echo '<h2>'.$news_item['title'].'</h2>';
  3. echo $news_item['text'];

路由

由于之前创建的通配符路由规则,你需要新增一条路由来显示你刚刚创建的控制器,修改你的路由配置文件(application/config/routes.php)添加类似下面的代码。该规则可以让请求访问 News 控制器而不是 Pages 控制器,第一行可以让带 slug 的 URI 重定向到 News 控制器的 view() 方法。

  1. $route['news/(:any)'] = 'news/view/$1';
  2. $route['news'] = 'news';
  3. $route['(:any)'] = 'pages/view/$1';
  4. $route['default_controller'] = 'pages/view';

把浏览器的地址改回根目录,在后面加上 index.php/news 来看看你的新闻页面吧。