<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0">
  <channel>
    <title>twairball (twairball)</title>
    <link>https://ruby-china.org/twairball</link>
    <description>Rails / iOS</description>
    <language>en-us</language>
    <item>
      <title>Rails / PostgreSQL 使用 JSONB 属性心得</title>
      <description>&lt;h2 id="电商订单信息保存为JSON属性"&gt;电商订单信息保存为 JSON 属性&lt;/h2&gt;
&lt;blockquote&gt;
&lt;p&gt;&lt;a href="https://lioniq.com" rel="nofollow" target="_blank" title=""&gt;LionIQ 数据狮&lt;/a&gt;数据狮专注于移动端电商插件开发，让任何 APP 快速完善电商功能。本文章讨论如何用 JSON 属性储存入数据库。&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;一般电商应用数据库我们可以现象订单对象 &lt;code&gt;Order&lt;/code&gt; 与产品对象 &lt;code&gt;Item&lt;/code&gt; 的关联，但是也有些情况把订单信息直接以静态 JSON 属性储存入数据库会有优势：&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;产品价格、信息更改&lt;/li&gt;
&lt;li&gt;产品删除或下架&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;把关系数据列如订单列表项 &lt;code&gt;order_items&lt;/code&gt; 直接以 JSON 属性存入数据库会失去所谓订单 &lt;code&gt;Order&lt;/code&gt; 于产品 &lt;code&gt;Item&lt;/code&gt; 的关联。因而订单信息的互动有限，反而宁愿订单信息生成后固定更改：&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;用户查看下过的订单信息，产品，单价&lt;/li&gt;
&lt;li&gt;运营团队更新订单状态 (未付、发货、完成)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;另外，在移动 APP 会经常需要把数据转换成 JSON 格式。因此，直接把订单信息储存为 JSON 属性是个方便的选择。我们在 LionIQ 数据狮使用的技术是 Postgresql + Ruby/Rails 5.0, 以下范例以此为主。&lt;/p&gt;
&lt;h3 id="Model 数据部署"&gt;Model 数据部署&lt;/h3&gt;
&lt;p&gt;创建数据如下：&lt;/p&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;rails g model Order order_no:uniq user:belongs_to order_infos:jsonb
&lt;/code&gt;&lt;/pre&gt;
&lt;p&gt;创建一般的 &lt;code&gt;Order&lt;/code&gt; 对象，再加上了一个 &lt;code&gt;order_infos&lt;/code&gt; 以 &lt;code&gt;jsonb&lt;/code&gt; 的属性装订单信息。[1]&lt;/p&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;# app/models/order.rb
class Order &amp;lt; ApplicationRecord
  belongs_to :shop_user

  # jsonb accessors
  store :order_infos, accessors: [:address, :order_items, :total_amount]
  store_accessor :order_infos 
end
&lt;/code&gt;&lt;/pre&gt;
&lt;p&gt;数据定义里我们用了 &lt;code&gt;store_accessor&lt;/code&gt; 及 &lt;code&gt;store&lt;/code&gt; 可以透过 &lt;code&gt;order_infos&lt;/code&gt; 参数获取 &lt;code&gt;JSON&lt;/code&gt; 数据，以及相应 &lt;code&gt;order_infos&lt;/code&gt; 内的参数。 &lt;/p&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;lioniq$ rails c
Running via Spring preloader in process 29470
Loading development environment (Rails 5.0.0)
irb(main):001:0&amp;gt; o = Order.new
=&amp;gt; #&amp;lt;Order id: nil, key: nil, archived: false, user_id: nil, order_infos: {}, created_at: nil, updated_at: nil, order_no: nil&amp;gt;

irb(main):002:0&amp;gt; o.order_infos
=&amp;gt; {}

irb(main):004:0&amp;gt; o.address = "98 Yanping Lu"
=&amp;gt; "98 Yanping Lu"

irb(main):005:0&amp;gt; o.order_items = [{key: "some_item_key", title: "cool tshirt", price: 3000, qty: 1}, {key: "another_key", title: "air jordan V", price: 10000, qty: 1}]
=&amp;gt; [{"key"=&amp;gt;"some_item_key", "title"=&amp;gt;"cool tshirt", "price"=&amp;gt;3000, "qty"=&amp;gt;1}, {"key"=&amp;gt;"another_key", "title"=&amp;gt;"air jordan V", "price"=&amp;gt;10000, "qty"=&amp;gt;1}]
irb(main):006:0&amp;gt; 

&lt;/code&gt;&lt;/pre&gt;&lt;h3 id="Controller 控制器"&gt;Controller 控制器&lt;/h3&gt;
&lt;p&gt;&lt;a href="https://lioniq.com/" rel="nofollow" target="_blank" title=""&gt;LionIQ&lt;/a&gt; 插件会以 JSON 格式传订单 &lt;code&gt;Order&lt;/code&gt; 数据，之后可以传送到你的后台 API 然后保存为以上定义的 &lt;code&gt;Order&lt;/code&gt; 数据。因为我们在 &lt;code&gt;model&lt;/code&gt; 里定义了 &lt;code&gt;store_accessor&lt;/code&gt; 可以很方便的定义 &lt;code&gt;strong parameters&lt;/code&gt; 可允许的 JSON 参数传给 &lt;code&gt;Order&lt;/code&gt; 初始化方式。&lt;/p&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;# app/controllers/orders_controller.rb
class OrdersController &amp;lt; ApplicationController

    # POST /orders
    # POST /orders.json
    def create
        @order = Order.new(order_params)

        # associate user, depending on your authentication
        @order.user = current_user 

        if @order.save 
            render :show, status: :created, location: @order
        else
            render json: @order.errors, status: :unprocessable_entity
        end
    end 

    private

    def order_params
        item_params = [:key, :title, :price]
        params.require(:order).permit(:order_no, :total_amount, :address, 
            order_items: [{item: item_params}, :qty])
    end
end
&lt;/code&gt;&lt;/pre&gt;&lt;h3 id="总结"&gt;总结&lt;/h3&gt;
&lt;p&gt;本文章采用很简单的 Ruby/Rails + Postgresql 示范如何保存订单信息为 JSON 属性入数据库。你是否需要开发电商 APP 功能？欢迎来到 &lt;a href="https://lioniq.com/" rel="nofollow" target="_blank" title=""&gt;LionIQ 数据狮&lt;/a&gt;  试用我们的 &lt;a href="https://github.com/lioniq/lioniq-ios" rel="nofollow" target="_blank" title=""&gt;iOS 电商插件&lt;/a&gt;, 今天就可以完善电商功能！&lt;/p&gt;

&lt;p&gt;欢迎来到我们技术 QQ 群：258693280 或发送邮件：dev@lioniq.com&lt;/p&gt;
&lt;h4 id="参考"&gt;参考&lt;/h4&gt;
&lt;ul&gt;
&lt;li&gt;[1] Rails / Postgresql JSONB

&lt;ul&gt;
&lt;li&gt;&lt;a href="http://nandovieira.com/using-postgresql-and-jsonb-with-ruby-on-rails" rel="nofollow" target="_blank"&gt;http://nandovieira.com/using-postgresql-and-jsonb-with-ruby-on-rails&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://segmentfault.com/a/1190000002911580" rel="nofollow" target="_blank"&gt;https://segmentfault.com/a/1190000002911580&lt;/a&gt;&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;/ul&gt;</description>
      <author>twairball</author>
      <pubDate>Sat, 15 Oct 2016 14:00:16 +0800</pubDate>
      <link>https://ruby-china.org/topics/31332</link>
      <guid>https://ruby-china.org/topics/31332</guid>
    </item>
  </channel>
</rss>
