好男人天堂网,久久精品国产这里是免费,国产精品成人一区二区,男人天堂网2021,男人的天堂在线观看,丁香六月综合激情

當前位置:首頁 > 網站舊欄目 > 學習園地 > 設計軟件教程 > domain model的延伸討論

domain model的延伸討論
2010-01-14 22:36:39  作者:  來源:
domain model,又稱為領域模型,是Java企業應用討論的一個熱門話題,JavaEye也曾經多次圍繞這個話題討論,我們來看個簡單的例子:

引用

一個簡單的公司工時管理系統,記錄員工的個人信息,每個員工的工作任務分配,以及工作所屬類別(例如開發,還是測試,還是培訓等等),其中每個員工有n個任務,員工和任務是一對多關系,每個員工也分別隸屬于多個不同的工作類別,員工和類型是多對多關聯關系,而每個任務也分別隸屬于唯一的工作類別,任務和類別是多對一關系。另外系統不要求對部門信息進行維護,不需要department表。因此,在這個系統中使用四張數據庫表:

users表保存員工信息,有name, password, gender, department, salary
tasks表保存工作任務信息,有name,start_time, end_time
kinds表保存工作所屬類別,有name
kinds_users表是一張關聯表,保存users表和kinds表的多對多關聯外鍵的

系統的功能需求如下:
1、某部門錄用一名新員工
2、某部門員工總薪水總和
3、某員工已經開始但尚未結束的任務
4、給某員工分配一項任務
5、所有用戶當前已經開始但尚未結束的任務
6、對某一類別,給所有和此一類別相關的員工,批量新增一批任務
7、針對任務的統計功能,給定某類別,統計當月總的任務數,已完成任務數,未完成任務數
 


我們先看看用ruby如何實現系統的領域模型:

Ruby代碼 復制代碼
  1. class User < ActiveRecord::Base   
  2.   has_and_belongs_to_many :kinds  
  3.      
  4.   has_many :tasks:dependent => :destroy do  
  5.     def processing_tasks   
  6.       find :all:conditions => ["start_time <= ? AND end_time is null"Time.now]   
  7.     end  
  8.   end  
  9.      
  10.   def apply_task(task_name)   
  11.     self.tasks << Task.new(:name => task_name, :start_time => Date.today)      
  12.   end      
  13.        
  14.   def self.all_processing_tasks   
  15.     Task.find :all:conditions => ["start_time <= ? AND end_time is null AND user_id is not null",Time.now]   
  16.   end  
  17. end  
  18.   
  19. class Task < ActiveRecord::Base   
  20.   belongs_to : owner, :class_name => 'User':foreign_key => 'user_id'  
  21.   belongs_to :kind  
  22.      
  23.   def self.current_month_tasks(kind)   
  24.     kind.tasks.current_month_tasks    
  25.   end  
  26. end  
  27.   
  28. class Kind < ActiveRecord::Base   
  29.   has_and_belongs_to_many :users  
  30.      
  31.   has_many :tasks do  
  32.     def current_month_tasks   
  33.       month_begin = Date.today - Date.today.mday + 1   
  34.       month_end = Date.today - Date.today.mday + 30   
  35.       processing_tasks = find :all:conditions => ["start_time <= ? AND end_time is null ", month_begin]   
  36.       processed_tasks = find :all:conditions => ["end_time >= ? AND end_time <= ? ", month_begin, month_end]   
  37.       all_tasks = processing_tasks.clone   
  38.       all_tasks << processed_tasks unless processed_tasks.size == 0   
  39.       return all_tasks, processed_tasks, processing_tasks   
  40.     end  
  41.   end  
  42.      
  43.   def add_batch_task_to_users(task_name)   
  44.     self.users.each do |user|   
  45.       task = Task.new(:name => task_name, :start_time => Date.today)    
  46.       user.tasks << task   
  47.       self.tasks << task   
  48.     end     
  49.   end  
  50. end  
  51.   
  52. class Department   
  53.   def self.employee(username, department)      
  54.     User.create(:name => username, :department => department)      
  55.   end     
  56.      
  57.   def self.total_salary(department)   
  58.     User.sum :salary:conditions => ["department = ?", department]   
  59.   end  
  60. end  


1、某部門錄用一名新員工
Ruby代碼 復制代碼
  1. Department.employee("robbin","開發部")  

2、某部門員工總薪水總和
Ruby代碼 復制代碼
  1. Department.total_salary("開發部")  

3、某員工已經開始但尚未結束的任務
Ruby代碼 復制代碼
  1. user.tasks.processing_tasks  

4、給某員工分配一項任務
Ruby代碼 復制代碼
  1. user.apply_task("學習Java")  

5、所有用戶當前已經開始但尚未結束的任務
Ruby代碼 復制代碼
  1. User.all_processing_tasks  

6、對某一類別,給所有和此一類別相關的員工,批量新增一批任務
Ruby代碼 復制代碼
  1. kind.add_batch_task_to_users("學習單元測試")  

7、針對任務的統計功能,給定某類別,統計當月總的任務數,已完成任務數,未完成任務數
Ruby代碼 復制代碼
  1. Task.current_month_tasks(kind)  


這里值得注意的是,RoR可以很方便的采用充血的領域模型,所有的業務邏輯都可以放在相關的domain model里面。這里的user,task和kind都是對應于數據庫表的領域模型,而department是不對應數據庫的純業務邏輯的domain model。總共4個ruby文件,4個domain model,55行代碼,所有要寫的代碼都在這里了,代碼量確實非常少,每個domain model的顆粒度都比較大。

然后我們再看看如何用Java:
Java代碼 復制代碼
  1. public class User {   
  2.     private Long id;   
  3.     private String name;   
  4.     private String password;   
  5.     private String gender;   
  6.     private String department;   
  7.     private int salary = 0;   
  8.     private List<Task> tasks = new ArrayList<Task>();   
  9.     # omit getter/setter methods ......   
  10. }   
  11.   
  12. # omit User's ORM Mapping file   
  13.   
  14. public class Task {   
  15.     private Long id;   
  16.     private String name;   
  17.     private int duration = 0;   
  18.     private User owner;   
  19.     # omit getter/setter methods ......   
  20. }   
  21.   
  22. # omit Task's ORM Mapping file   
  23.   
  24. public class Kind {    
  25.     ......   
  26. }   
  27.   
  28. # omit Kind's ORM Mapping file   
  29.   
  30. public interface UserDao {   
  31.     public void addUser(User user);   
  32.     public loadUserById(Long id);   
  33.     # omit CRUD and other persistent methods ......   
  34.     public List<User> findByDeparment(String department);   
  35. }   
  36.   
  37. public interface TaskDao {   
  38.     # omit CRUD and other persistent methods ......   
  39. }   
  40.   
  41. public class UserDaoImpl {   
  42.     # omit implementations ......   
  43. }   
  44.   
  45. public class TaskDaoImpl {   
  46.     # omit implementations ......   
  47. }   
  48.   
  49.   
  50. public class UserService {   
  51.     private UserDao userDao;   
  52.     public setUserDao(UserDao userDao) { this.userDao = userDao; }   
  53.     public int workload(User user) {   
  54.         int totalDuration = 0;   
  55.         for (Task task : user.getTasks()) {   
  56.             totalDuration += task.duration;   
  57.         }   
  58.         return totalDuration;   
  59.     }   
  60.     public employee(String username, String department) {   
  61.         User user = new User();   
  62.         user.setName(username);   
  63.         user.setDepartment(department);   
  64.         userDao.addUser(user);   
  65.     }   
  66. }   
  67.   
  68. public class TaskService {   
  69.     private TaskDao taskDao;   
  70.     public void setTaskDao(TaskDao taskDao) { this.taskDao = taskDao }   
  71.     public applyTask(String taskName, User user) {   
  72.         Task task = new Task();   
  73.         task.setName(taskName);   
  74.         task.setUser(user);   
  75.         taskDao.addTask(task);   
  76.     }   
  77. }   
  78.   
  79. public class DepartmentService {   
  80.     private UserDao userDao;   
  81.     public void setUserDao(UserDao userDao) { this.userDao = userDao; }   
  82.     private UserService userService;   
  83.     public void setUserService(UserService userService) { this.userService = userService; }   
  84.     public int totalSalary(String department) {   
  85.         ......   
  86.     }   
  87.     ......     
  88. }    
  89.   
  90. # omit IoC Container weaving configuration's file  


Java版本的實現代碼大家都比較熟悉,因此絕大部分代碼都省略了。Java版本需要3個持久對象,3個映射XML文件,3個DAO接口和實現類,4個Service和實現類,和一個IoC的bean組裝文件,總共21個文件,全部邏輯寫完整,代碼行數至少上千行。

通過對比,我們可以看到Java比較流行的實現是貧血的模型,按照面向對象的基本原則,對象的狀態應該和它的行為封裝在一起,因此Java多出來的這些XXXService是一些從純理論角度而言應該放入其相應的持久對象中去。但是Java實現充血模型從技術上有一定的難度,如何Service方法挪入到持久對象中呢?如何解決Dao的注入問題?如何解決domain logic方法的事務封裝問題?前者可以通過AspectJ的靜態織入來解決,后者也許可以通過織入或者annotation聲明來解決。但不管怎么說,Java從技術上很難實現充血模型,而且即使實現充血模型,也會導致一個Java類好幾百行代碼的狀況,其代碼的可閱讀性,模塊解藕能力都會變得很差,因此我們認為Java不適合充血模型,在表達復雜的業務邏輯的能力上,Java要比ruby差很多:

結論:
對于Java來說,更加適合采用貧血的模型,Java比較適合于把一個復雜的業務邏輯分離到n個小對象中去,每個小對象描述單一的職責,n個對象互相協作來表達一個復雜的業務邏輯,這n個對象之間的依賴和協作需要通過外部的容器例如IoC來顯式的管理。但對于每個具體的對象來說,他們毫無疑問是貧血的。

這種貧血的模型好處是:
1、每個貧血對象職責單一,所以模塊解藕程度很高,有利于錯誤的隔離。
2、非常重要的是,這種模型非常適合于軟件外包和大規模軟件團隊的協作。每個編程個體只需要負責單一職責的小對象模塊編寫,不會互相影響。

貧血模型的壞處是:
1、由于對象狀態和行為分離,所以一個完整的業務邏輯的描述不能夠在一個類當中完成,而是一組互相協作的類共同完成的。因此可復用的顆粒度比較小,代碼量膨脹的很厲害,最重要的是業務邏輯的描述能力比較差,一個稍微復雜的業務邏輯,就需要太多類和太多代碼去表達(針對我們假定的這個簡單的工時管理系統的業務邏輯實現,ruby使用了50行代碼,但Java至少要上千行代碼)。
2、對象協作依賴于外部容器的組裝,因此裸寫代碼是不可能的了,必須借助于外部的IoC容器。

對于Ruby來說,更加適合充血模型。因為ruby語言的表達能力非常強大,現在用ruby做企業應用的DSL是一個很熱門的領域,DSL說白了就是用來描述某個行業業務邏輯的專用語言。

充血模型的好處是:
1、對象自洽程度很高,表達能力很強,因此非常適合于復雜的企業業務邏輯的實現,以及可復用程度比較高。
2、不必依賴外部容器的組裝,所以RoR沒有IoC的概念。

充血模型的壞處是:
1、對象高度自洽的結果是不利于大規模團隊分工協作。一個編程個體至少要完成一個完整業務邏輯的功能。對于單個完整業務邏輯,無法再細分下去了。
2、隨著業務邏輯的變動,領域模型可能會處于比較頻繁的變動狀態中,領域模型不夠穩定也會帶來web層代碼頻繁變動。

附件是完整的RoR版本的項目示例代碼。要運行它,需要安裝MySQL數據庫(InnoDB表類型),Ruby和Ruby on rails環境。在MySQL數據庫中分別創建demo數據庫和demo_test數據庫,修改demo\config\database.yml中的MySQL數據庫配置,改成你的數據庫密碼。然后在項目跟目錄下面執行:
rake db:migrate
rake db:test:clone_structure
rake test
即創建開發環境數據庫,創建測試環境數據庫,和執行所有的單元測試。領域模型代碼位于demo\app\models目錄下面;單元測試代碼位于demo\test\units目錄下面

安徽新華電腦學校專業職業規劃師為你提供更多幫助【在線咨詢
久久成人综合网| 韩国三级视频网站| 99热精品一区| 亚洲 国产精品 日韩| 亚洲精品中文字幕久久久久久| 韩国三级一区| 韩国三级香港三级日本三级la | 亚洲 激情| 欧美激情伊人| 欧美国产日韩久久久| 毛片的网站| 欧美1区| 亚飞与亚基在线观看| 韩国三级视频网站| 亚洲精品影院一区二区| 欧美激情一区二区三区在线| 九九久久国产精品| 精品国产一区二区三区国产馆| 麻豆污视频| 日本在线不卡视频| 国产麻豆精品| 国产亚洲免费观看| 午夜在线亚洲| 国产成人精品综合在线| 成人免费一级纶理片| 日本免费乱理伦片在线观看2018| 欧美另类videosbestsex视频| 成人高清免费| 欧美大片aaaa一级毛片| 国产网站免费| 亚洲 国产精品 日韩| 你懂的国产精品| 沈樵在线观看福利| 精品毛片视频| 欧美另类videosbestsex视频| 青青久久国产成人免费网站| 久久国产精品自由自在| 一级片片| 美女免费毛片| 国产a视频精品免费观看| 欧美另类videosbestsex视频| 久草免费在线色站| 999久久66久6只有精品| 久久久成人影院| a级毛片免费全部播放| 久久精品欧美一区二区| 国产精品1024永久免费视频| 欧美日本免费| 日韩中文字幕一区| 亚洲第一页色| 精品国产亚一区二区三区| 精品国产亚一区二区三区| 国产网站在线| 国产成a人片在线观看视频| 日韩一级黄色片| 天天色色网| 日韩免费在线| 99热精品一区| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 精品国产一级毛片| 亚洲精品久久玖玖玖玖| 亚飞与亚基在线观看| 欧美α片无限看在线观看免费| 久草免费在线色站| 黄色免费三级| 国产麻豆精品| 国产精品1024永久免费视频| 色综合久久天天综合绕观看| 黄视频网站在线看| 99久久精品国产高清一区二区| 天天做日日爱| a级毛片免费全部播放| 国产91素人搭讪系列天堂| 麻豆系列 在线视频| 亚洲精品影院| 成人免费高清视频| 日日夜夜婷婷| 日韩综合| 午夜欧美成人香蕉剧场| 九九热国产视频| 尤物视频网站在线观看| 国产不卡福利| 国产不卡在线看| 亚洲 欧美 成人日韩| 欧美一级视频免费| 99久久精品国产高清一区二区| 九九精品久久久久久久久| 久久精品成人一区二区三区| 精品视频在线观看一区二区| 天天做人人爱夜夜爽2020毛片| 亚洲女人国产香蕉久久精品| 美女免费毛片| 欧美a级大片| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 国产成人啪精品| 欧美18性精品| 国产成人精品一区二区视频| 成人免费高清视频| 日韩中文字幕一区| 四虎影视库| 91麻豆国产福利精品| 天天色色网| 日本特黄特色aaa大片免费| 亚洲天堂在线播放| 香蕉视频久久| 91麻豆精品国产自产在线| 久久久久久久网| 成人影视在线观看| 亚洲精品中文字幕久久久久久| 日韩一级黄色| 国产91精品一区| 日韩一级黄色| 成人免费网站视频ww| 成人高清视频免费观看| 尤物视频网站在线观看| 久久国产一区二区| 国产麻豆精品| 久久久久久久男人的天堂| 夜夜操网| 精品毛片视频| 国产91丝袜高跟系列| 午夜精品国产自在现线拍| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 成人免费观看男女羞羞视频| 日本免费乱理伦片在线观看2018| 久久99中文字幕| 999久久狠狠免费精品| 99热视热频这里只有精品| 欧美大片毛片aaa免费看| 麻豆午夜视频| 精品视频在线观看免费| 天天做日日爱夜夜爽| 尤物视频网站在线| 精品国产亚一区二区三区| 国产高清在线精品一区二区| 日韩一级黄色| 一级女性全黄久久生活片| 99色视频在线观看| 天天做日日干| 国产视频一区在线| 国产精品自拍在线| 沈樵在线观看福利| 午夜欧美成人香蕉剧场| 免费的黄视频| 久久国产精品自由自在| 美女免费毛片| 国产一区二区高清视频| 亚洲第一页色| 成人影院一区二区三区| 国产a视频| 精品视频在线观看视频免费视频| 高清一级淫片a级中文字幕| 欧美a级片免费看| 成人高清视频免费观看| 国产成人精品综合| 国产韩国精品一区二区三区| 日韩av片免费播放| 高清一级片| 一级女性全黄生活片免费| 欧美a免费| 亚洲第一页色| 精品视频在线观看免费| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 99久久网站| 99热精品一区| 欧美激情一区二区三区在线| 久久99这里只有精品国产| 精品视频在线看| 成人免费观看的视频黄页| 欧美激情一区二区三区视频| 九九精品久久久久久久久| 中文字幕97| 国产一区二区精品| 色综合久久天天综合观看| 黄色免费三级| 日韩av成人| a级黄色毛片免费播放视频| 日日夜夜婷婷| 国产不卡在线看| 色综合久久天天综合绕观看| 日日夜夜婷婷| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 精品在线免费播放| 国产精品1024永久免费视频| 国产91精品系列在线观看| 国产网站免费视频| 欧美激情一区二区三区在线| 国产成人啪精品视频免费软件| 色综合久久天天综线观看| 91麻豆精品国产综合久久久| 国产极品白嫩美女在线观看看| 欧美大片毛片aaa免费看| 黄色福利| 中文字幕一区二区三区 精品| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 国产一区二区精品久久| 国产麻豆精品免费视频| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 久久精品成人一区二区三区| 二级片在线观看|