BUILDING ANGULAR APPS USING FLUX ARCHITECTURE
Flux is an architectural pattern based on unidirectional data flow. Although it is originated in the React community, it is general and can be used with any non-opinionated framework. In this article I will show how to use it with Angular.
WHAT IS FLUX
There are four core components in Flux.

- Views render data and trigger actions.
- The dispatcher propagates actions from views to stores.
- Stores contain application logic and state.
A typical Flux interaction looks as follows:
- The view triggers an action.
- The action layer, using the data provided by the view, constructrs an action object.
- The dispatcher receives the action, puts it in its action queue, and later notifies the store about it.
- The store updates its state based on the action and broadcasts a change event.
- The view reads data from the store and updates itself.
REPLACEMENT FOR MVC?
MVC is a very old pattern. And as a result, it is hardly possible to find two people who would agree on what it really means. I see MVC as a component pattern, with the view and the controller forming the component. Defined like this, MVC does not conflict with Flux and can be used in the view layer.
APPLICATION
I won’t cover the pattern itself in great detail. So if you feel confused after reading the post, I highly recommend you to check outthis screencast series by Joe Maddalone, which is absolutely excellent. In the series Joe Maddalone builds a shopping cart application, a simplified version of which I am going to build in this post.

The application displays two tables: the catalog and the shopping cart, and supports two operations: addItem and removeItem.
ANGULAR & FLUX
VIEWS
Let’s start with the catalog table.
<table ng-controller="CatalogCtrl as catalog">
<tr ng-repeat="item in catalog.catalogItems">
<td>{{item.title}}</td>
<td>{{item.cost}}</td>
<td>
<button ng-click="catalog.addToCart(item)">Add to Cart</button>
</td>
</tr>
</table>
Where CatalogCtrl is defined as follows:
var m = angular.module('cart', []);
class CatalogCtrl {
constructor(catalogItems, cartActions) {
this.cartActions = cartActions;
this.catalogItems = catalogItems;
}
addToCart(catalogItem) {
this.cartActions.addItem(catalogItem);
}
}
m.controller("CatalogCtrl", CatalogCtrl);
To make it simple, let’s hardcode catalogItems.
m.value("catalogItems", [
{id: 1, title: 'Item #1', cost: 1},
{id: 2, title: 'Item #2', cost: 2},
{id: 3, title: 'Item #3', cost: 3}
]);
The controller does not do much. It has no application logic: it just triggers the addItem action. It is worth noting that not everything should be handled as an action and go through the pipeline. Local view concerns (e.g., UI state) should be kept in the view layer.
ACTIONS
The action layer is nothing but a collection of helper functions, which, at least in theory, should have a helper for every action your application can perform.
var ADD_ITEM = "ADD_ITEM";
m.factory("cartActions", function (dispatcher) {
return {
addItem(item) {
dispatcher.emit({
actionType: ADD_ITEM,
item: item
})
}
};
});
DISPATCHER
The dispatcher is a message-bus service that propagates events from views to stores. It has no application logic. A real project should probably use the Dispatcher library from Facebook, or implement it using RxJS. In this article I will use this simpleEventEmitter class, just to illustrate the interaction.
class EventEmitter {
constructor() {
this.listeners = [];
}
emit(event) {
this.listeners.forEach((listener) => {
listener(event);
});
}
addListener(listener) {
this.listeners.push(listener);
return this.listeners.length - 1;
}
}
m.service("dispatcher", EventEmitter);
Since all actions go through the dispatcher, it is a great place to add such things as logging.
STORE
A store contains application logic and state for a particular domain. It registers itself with the dispatcher to be notified about the events flowing through the system. The store then updates itself based on those events. And what is really important, there is no other way to update it. Views and other stores can read data from the store, but not update it directly. The store is observable, so it emits events that views can listen for.
This’s one way to implement the cart store.
class CartStore extends EventEmitter {
constructor() {
super();
this.cartItems = [];
}
addItem(catalogItem) {
var items = this.cartItems.filter((i) => i.catalogItem == catalogItem);
if (items.length == 0) {
this.cartItems.push({qty: 1, catalogItem: catalogItem});
} else {
items[0].qty += 1;
}
}
removeItem(cartItem) {
var index = this.cartItems.indexOf(cartItem);
this.cartItems.splice(index, 1);
}
emitChange() {
this.emit("change");
}
}
Next, we need to register it.
m.factory("cartStore", function (dispatcher) {
var cartStore = new CartStore();
dispatcher.addListener(function (action) {
switch(action.actionType){
case ADD_ITEM:
cartStore.addItem(action.item);
cartStore.emitChange();
break;
case REMOVE_ITEM:
cartStore.removeItem(action.item);
cartStore.emitChange();
break;
}
});
//expose only the public interface
return {
addListener: (l) => cartStore.addListener(l),
cartItems: () => cartStore.cartItems
};
});
Since stores is where all of the application logic is contained, they can become very complicated very quickly. In this exampleCartStore is implemented using the transaction script pattern, which works fairly well for small applications, but breaks apart when dealing with complex domains. In addition, stores are responsible for both the domain and application logic. This, once again, works well for small applications, but not so much for larger one. So you may consider separating the two.
CLOSING THE LOOP
We are almost done. There is only one piece left to see the whole interaction working: we need to implement the cart table.
<h1>Cart</h1>
<table ng-controller="CartCtrl as cart">
<tr ng-repeat="item in cart.items track by $id(item)">
<td>{{item.catalogItem.title}}</td>
<td>{{item.qty}}</td>
<td>
<button ng-click="cart.removeItem(item)">x</button>
</td>
</tr>
</table>
class CartCtrl {
constructor(cartStore, cartActions) {
this.cartStore = cartStore;;
this.cartActions = cartActions;
this.resetItems();
cartStore.addListener(() => this.resetItems());
}
resetItems() {
this.items = this.cartStore.cartItems();
}
removeItem(item) {
//to be implemented
}
}
m.controller("CartCtrl", CartCtrl);
The controller listens to the store, and when that changes, resets the list of cart items.
INTERACTION

SUMMING UP
Flux is an architectural pattern based on unidirectional data flow. It can be used with any non-opinionated framework. In this blog post I have shown how it can be used with Angular.
READ MORE
- The example from this post, including the remove item functionality.
- Flux Architecture Overview
- Flux Arhitecture by Joe Maddalone
- Read up on CQRS. CQRS is a well-established pattern for building server-side applications. Flux is built around the same set of ideas. There is a lot of great material on CQRS: pros and cons, how to structure actions, and so on. Most of it can be applied to Flux.
BUILDING ANGULAR APPS USING FLUX ARCHITECTURE的更多相关文章
- Building Web Apps with SignalR, Part 1
Building Web Apps with SignalR, Part 1 In the first installment of app-building with SignalR, learn ...
- Building Android Apps 30条建议
Building Android Apps — 30 things that experience made me learn the hard way There are two kinds of ...
- Building the Unstructured Data Warehouse: Architecture, Analysis, and Design
Building the Unstructured Data Warehouse: Architecture, Analysis, and Design earn essential techniqu ...
- Flux architecture
[Flux architecture] Flux is a pattern for managing data flow in your application. The most important ...
- [AngularJS] Sane, scalable Angular apps are tricky, but not impossible.
Read fromhttps://medium.com/@bluepnume/sane-scalable-angular-apps-are-tricky-but-not-impossible-less ...
- HoloLens开发手记 - 构建2D应用 Building 2D apps
HoloLens可以让我们在真实世界中看到全息图像内容.但是它本质上还是一台Windows 10设备,这意味着HoloLens可以以2D应用形式运行Windows Store里的大部分UWP应用. 目 ...
- [Angular] Debug Angular apps in production without revealing source maps
Source: https://blog.angularindepth.com/debug-angular-apps-in-production-without-revealing-source-ma ...
- AngularJS + RequireJS
http://www.startersquad.com/blog/AngularJS-requirejs/ While delivering software projects for startup ...
- Kivy: Building GUI and Mobile apps with Python
Intro Python library for building gui apps (think qt, gdk,processing) build from ground up for lates ...
随机推荐
- [TopCoder14647]HiddenRabbits
vjudge description 有一棵\(n\)个节点的树和\(m\)只兔子,每只兔子要住在一个点上(可以多只兔子住在同一个点上).有\(q\)组要求,每组形如"当以\(r\)为根时, ...
- developerWorks 中国 技术主题 Java technology 文档库 Java 性能测试的四项原则
转-https://www.ibm.com/developerworks/cn/java/j-lo-java-performance-testing/?cm_mmc=dwchina-_-homepa ...
- Oracle SEQUENCE 具体说明
ORACLE SEQUENCE ORACLE没有自增数据类型,如需生成业务无关的主键列或惟一约束列,能够用sequence序列实现. CREATE SEQUENCE语句及參数介绍: 创建序 ...
- EXCEL中如何删除透视表的多余汇总
EXCEL中如何删除透视表的多余汇总 1)如下图,选中字段列,单击鼠标右键,在弹出的菜单中选择[字段设置]选项. 2)弹出[字段设置]对话框. 3)选择“分类汇总和筛选”选项卡,然后勾选“无”选项,单 ...
- hadoop之 安全模式及SafeModeException
问题: hadoop启动的时候报错 HTTP ERROR 500 Problem accessing /nn_browsedfscontent.jsp. Reason: Cannot issue de ...
- HTTP请求中的form data,request payload,query string parameters以及在node服务器中如何接收这些参数
http://www.cnblogs.com/hsp-blog/p/5919877.html 今天,在工作(倒腾微信小程序)的时候,发现发送post请求到node后台服务器接收不到前端传来的参数.其实 ...
- Spring MVC @ResponseBody和@RequestBody使用
@ResponseBody用法: 作用:该注解用于将Controller的方法返回的对象,根据HTTP Request Header的Accept的内容,通过适当的HttpMessageConvert ...
- WebFrom基础
ASP.NET WebForm C/S B/S客人 - 用户 要土豆丝 - 给IIS发送请求 ,IIS就相当于是服务员 通知厨房 - IIS把用户要想看到的ASPX告知.NET框架 厨房炒菜 - .n ...
- Autofac框架详解
一.组件 创建出来的对象需要从组件中来获取,组件的创建有如下4种(延续第一篇的Demo,仅仅变动所贴出的代码)方式: 1.类型创建RegisterType AutoFac能够通过反射检查一个类型,选择 ...
- Linux gdb调试器用法全面解析
GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具,GDB主要可帮助工程师完成下面4个方面的功能: 启动程序,可以按照工程师自定义的要求随心所欲的运行程序. 让被调试的程序在工程师指定的断 ...