文档章节

两天完成一个小型工程报价系统(三层架构)

 木宛城主
发布于 2015/03/02 19:42
字数 1480
阅读 6
收藏 1
点赞 0
评论 0

花了两天,整理了一下三层架构,顺便练了一个小型三层架构——工程报价系统。

功能很简单,完成基本的增删改查

  • 搭建项目三层结构

  • 界面的设计以及美化

  • 分层代码的设计与实现

  1. 模型层:由于表设计时存在主外键关系,故把表映射成对象时添加一个外键对象来保存外键那张表的相关数据。

 

// 外键实体
            private Product _product;
            private Project _project;

            public Project _Project
            {
                get { return _project; }
                set { _project = value; }
            }

            public Product _Product
            {
                get { return _product; }
                set { _product = value; }
            }

 

  2.数据访问层:把数据封装成Model对象的各层之间传递

  我个人不喜欢三层之间传递DataSet数据集,我的理解是尽量不在BLL层出现DataSet,所以我在DAL层把所有获取的数据封装成IEnumerable集合,然后返回。

protected IEnumerable<ProjectItem> ToModelsByFK(SqlDataReader reader)
         {
             var list = new List<ProjectItem>();
             while (reader.Read())
             {
                 list.Add(ToModelByFK(reader));
             }
             return list;
         }  

      
public ProjectItem ToModelByFK(SqlDataReader reader)
         {
             ProjectItem projectItem = new ProjectItem();
             projectItem._Product = new Product();
             projectItem._Project = new Project();
             projectItem.ID = (int)ToModelValue(reader, "ID");
             projectItem._Product.ID = (int)ToModelValue(reader, "ProductID");
             projectItem._Project.ID = (int)ToModelValue(reader,"projectID");
             projectItem._Product.Manufacturer = ToModelValue(reader, "Manufacturer").ToString();
             projectItem._Product.Parameters = ToModelValue(reader, "Parameters").ToString();
             projectItem._Product.Price = decimal.Parse( ToModelValue(reader, "Price").ToString());
             projectItem._Product.ProductName = ToModelValue(reader, "ProductName").ToString();
             projectItem._Product.Specification = ToModelValue(reader, "Specification").ToString();
             projectItem._Product.Unit = ToModelValue(reader, "Unit").ToString();
             projectItem.Count = (int)ToModelValue(reader, "Count");
             projectItem.TotalMoney = (decimal)ToModelValue(reader, "TotalMoney");
             projectItem.UnitPrice = (decimal)ToModelValue(reader, "UnitPrice"); 
             return projectItem;
         }
protected object ToModelValue(SqlDataReader reader,string columnName)
        {
            if(reader.IsDBNull(reader.GetOrdinal(columnName)))
            {
                return null;
            }
            else
            {
                return reader[columnName];
            }
        }

使用GetProjectByCondition方法返回一个封装成对象的 IEnumerable集合

public IEnumerable<Model.Project> GetProjectByCondition(string projectName, string customerName, string contract, string tel)
        {
            StringBuilder sqlWhere = new StringBuilder("select * from Project where 1=1");
            List<SqlParameter> listParameters = new List<SqlParameter>();
            if (!string.IsNullOrWhiteSpace(projectName))
            {
                sqlWhere.AppendLine(" and projectName like @projectName");
                listParameters.Add(new SqlParameter("projectName", "%" + projectName + "%"));
            }
            if (!string.IsNullOrWhiteSpace(contract))
            {
                sqlWhere.AppendLine(" and Contact like @Contact");
                listParameters.Add(new SqlParameter("Contact", "%" + contract + "%"));
            }
            if (!string.IsNullOrWhiteSpace(customerName))
            {
                sqlWhere.AppendLine(" and customer like @customer");
                listParameters.Add(new SqlParameter("customer", "%" + customerName + "%"));
            }
            if (!string.IsNullOrWhiteSpace(tel))
            {
                sqlWhere.AppendLine(" and tel like @tel");
                listParameters.Add(new SqlParameter("tel", "%" + tel + "%"));
            }
            using (SqlDataReader reader = SqlHelper.ExecuteDataReader(sqlWhere.ToString(), listParameters.ToArray()))
            {
                return ToModels(reader);
            }

        }

UpdatePassChecked方法接受一个对象,通过对象的属性来获取数据

public int UpdatePassChecked(ProjectItem projectItem)
         {
             string sql =
                 "UPDATE Project " +
                 "SET " +
             " ProjectName = @ProjectName"
                 + ", Customer = @Customer"
                 + ", Contact = @Contact"
                 + ", Tel = @Tel"
                 + ", DeliveryPlace = @DeliveryPlace"
                 + ", DeliveryTime = @DeliveryTime"
                 + ", TransportCosts = @TransportCosts"
                 + ", PaymentTerm = @PaymentTerm"
                 + ", Bak = @Bak"

             + " WHERE ID = @ID";


             SqlParameter[] para = new SqlParameter[]
            {
                new SqlParameter("@ID", projectItem._Project.ID)
                    ,new SqlParameter("@ProjectName", ToDBValue(projectItem._Project.ProjectName))
                    ,new SqlParameter("@Customer", ToDBValue(projectItem._Project.Customer))
                    ,new SqlParameter("@Contact", ToDBValue(projectItem._Project.Contact))
                    ,new SqlParameter("@Tel", ToDBValue(projectItem._Project.Tel))
                    ,new SqlParameter("@DeliveryPlace", ToDBValue(projectItem._Project.DeliveryPlace))
                    ,new SqlParameter("@DeliveryTime", ToDBValue(projectItem._Project.DeliveryTime))
                    ,new SqlParameter("@TransportCosts", ToDBValue(projectItem._Project.TransportCosts))
                    ,new SqlParameter("@PaymentTerm", ToDBValue(projectItem._Project.PaymentTerm))
                    ,new SqlParameter("@Bak", ToDBValue(projectItem._Project.Bak))
            };

             return SqlHelper.ExecuteNonQuery(sql, para);

         }

Add方法传入一个已经把数据封装好的对象,然后返回一个新的对象(包含新创建生成的ID)

public Product Add(Product product)
        {
                string sql ="INSERT INTO Product (ProductName, Specification, Manufacturer, Parameters, Price, Unit)  output inserted.ID VALUES (@ProductName, @Specification, @Manufacturer, @Parameters, @Price, @Unit)";
                SqlParameter[] para = new SqlParameter[]
                    {
                        new SqlParameter("@ProductName", ToDBValue(product.ProductName)),
                        new SqlParameter("@Specification", ToDBValue(product.Specification)),
                        new SqlParameter("@Manufacturer", ToDBValue(product.Manufacturer)),
                        new SqlParameter("@Parameters", ToDBValue(product.Parameters)),
                        new SqlParameter("@Price", ToDBValue(product.Price)),
                        new SqlParameter("@Unit", ToDBValue(product.Unit)),
                    };
                    
                int newId = (int)SqlHelper.ExecuteScalar(sql, para);
                return GetByID(newId);
        }

 

  3.业务逻辑层:业务逻辑层是项目的核心,业务逻辑上的代码通常在这儿实现(一言难尽)。

简单数据验证:

public bool UpdateThroughChecked(ProjectItem projectItem)
        {
            if (string.IsNullOrEmpty(projectItem._Project.ProjectName))
            {
                throw new Exception("项目名称不能为空");
            }
            //顾客姓名可以为空,但联系人不能为空,所以数据库里就应该设计好
            if (string.IsNullOrEmpty(projectItem._Project.Contact))
            {
                throw new Exception("联系人姓名不能为空");
            }
            if (string.IsNullOrEmpty(projectItem._Project.Tel))
            {
                throw new Exception("联系电话不能为空");
            }
            if (string.IsNullOrEmpty(projectItem._Project.DeliveryPlace))
            {
                throw new Exception("交货地点不能为空");
            }
            if (string.IsNullOrEmpty(projectItem._Project.DeliveryTime))
            {
                throw new Exception("交货时间不能为空");
            }
            if (projectItem._Project.TransportCosts<00)
            {
                throw new Exception("运输费用不能为负数");
            }
            return new DAL.ProjectItemService().UpdatePassChecked(projectItem) > 0;

        }

计算总金额:

public decimal GetProductTotalMoney(int projectID)
        {
            var list = new DAL.ProjectItemService().GetProductTotalMoney(projectID);
            decimal  totalMoney = 0.00M;
            foreach (var model in list)
            {
                totalMoney+=model.TotalMoney.Value;
            }
            return totalMoney;
        }

根据查询条件获取数据集合:

/// <summary>
        /// 根据查询条件获取数据集合
        /// </summary>
        /// <param name="condition"></param>
        /// <returns></returns>
        public IEnumerable<Model.Product> GetProductsByCondition(string condition)
        {
           return new DAL.ProductService().GetProductsByCondition(condition);
       
        }

根据ProjectItemID获取一条记录(封装成对象)

public Model.ProjectItem GetOneProjectItemByID(int ProjectItemID)
        {
            return new DAL.ProjectItemService().GetOneProjectItemByID(ProjectItemID);
        }

增加一条记录:

/// <summary>
        /// 新增一条记录 根据条件
        /// </summary>
        /// <param name="modelProject"></param>
        /// <returns></returns>
        public Model.Project AddPassCheckd(Model.Project modelProject)
        {

            if (string.IsNullOrEmpty(modelProject.ProjectName))
            {
                throw new Exception("项目名称不能为空");
            }
            if (string.IsNullOrEmpty(modelProject.Customer))
            {
                throw new Exception("顾客名称不能为空");
            }
            if (string.IsNullOrEmpty(modelProject.Tel))
            {
                throw new Exception("联系电话不能为空");
            }
            if (modelProject.TransportCosts <= 0)
            {
                throw new Exception("运输费用不能为负数:(");
            }
            return new DAL.ProjectService().Add(modelProject);
           
        }

删除一个项目:

/// <summary>
        /// 删除一个项目
        /// </summary>
        /// <param name="projectID"></param>
        /// <returns></returns>
        public bool DeleteProject(int projectID)
        {
             bool isSucceed=  new BLL.ProjectItemManager().DeleteProjectItemsByProjectID(projectID);
             isSucceed = new DAL.ProjectService().DeleteByID(projectID) > 0 && isSucceed;
             return isSucceed;
        }

4.表现层(UI)

简单的窗体里嵌套窗体:

public void LoadProjectListForm()
        {
            projectListForm = null;
            if (projectListForm==null)
            {
                projectListForm = new ProjectListForm();
            }
            //去掉边框
            projectListForm.FormBorderStyle = FormBorderStyle.None;
            projectListForm.TopLevel = false;//窗体是否是顶级窗体
            projectListForm.Dock = DockStyle.Fill;//填充
            projectListForm.Show();//不写看不到
            panelContainer.Controls.Clear();//清空
            panelContainer.Controls.Add(projectListForm);
        }

绑定数据:

我原先思路是直接在DataGridView里绑定对象的属性,和GridView一样使用,但一直无法绑上去,哪儿错了?

所以只能曲线救国了:

void LoadData()
        {
            var list = new BLL.ProjectItemManager().GetAllProducts(projectID);
            //dataGridViewProjectItems.DataSource = list;//当控件被绑定时无法向其添加Row
            dataGridViewProjectItems.Rows.Clear();
            foreach (var model in list)
            {
                int i = dataGridViewProjectItems.Rows.Add();
                dataGridViewProjectItems.Rows[i].Cells["idColumn"].Value = model.ID;
                dataGridViewProjectItems.Rows[i].Cells["specificationColumn"].Value = model._Product.Specification;
                dataGridViewProjectItems.Rows[i].Cells["productNameColumn"].Value = model._Product.ProductName;
                dataGridViewProjectItems.Rows[i].Cells["manufacturerColumn"].Value = model._Product.Manufacturer;
                dataGridViewProjectItems.Rows[i].Cells["parametersColumn"].Value = model._Product.Parameters;
                dataGridViewProjectItems.Rows[i].Cells["productIDColumn"].Value = model._Product.ID;
                dataGridViewProjectItems.Rows[i].Cells["totalMoneyColumn"].Value = model.TotalMoney;
                dataGridViewProjectItems.Rows[i].Cells["countColumn"].Value = model.Count;
                dataGridViewProjectItems.Rows[i].Cells["unitPriceColumn"].Value = model.UnitPrice;
                dataGridViewProjectItems.Rows[i].Cells["projectIDColumn"].Value = model._Project.ID;
            }

5.项目总结

花了一个周末修修补补完成了这个工程报价系统,算是复习了一把三层架构。

如果有需要这个项目的朋友可以留下邮箱,我发给你,不算很专业,凑合着参考下~~

没想到这么多朋友要我的源码,我很感动。附件我就不传了,我也怕误导新人。有需要的朋友可以留下邮箱,大家一起探讨,我的代码纯当入门阶段,是我个人对三层架构的理解,不足之处,也请多多包涵,提个意见,希望与君互勉。想要更深刻探索我推介《ASP.NET 设计模式》


 源代码下载:

© 著作权归作者所有

共有 人打赏支持
粉丝 2
博文 222
码字总数 199010
作品 0
黄浦
咱们在讨论的时候能不能先确定一个明确的范围呢?最好能把自己的经历简单说一下。

昨天晚上看到了三层,又见三层。别了,三层 有点想法,在这里说一下。和周六活动的讨论没有什么关系,特此声明! 讨论是好的,但是范围不明确的话,就很容易跑题,也容易带出来火药味儿。想这...

金色海洋 ⋅ 2017/12/24 ⋅ 0

广告公关公司如何选择项目管理软件

广告公关行业属于大传播行业其中的一种,广告公关公司如果选择管理软件,可以参照传播业管理软件。为什么叫传播业?我们一直在找一个什么样的词把这几个细分行业都囊括进来,发现唯有传播能更...

玄学酱 ⋅ 05/22 ⋅ 0

设计讨论:好设计长什么样?

怎样的设计算得上好设计? 就如一百个人眼中有一百个哈姆雷特一样,这个问题多半也不会有标准答案。何况,脱离了具体和细节的业务需求,设计、架构之类的讨论也只剩套路了。 不过,前两天想了...

winters1224 ⋅ 2017/01/03 ⋅ 0

不在乎y/RTXCPluginForDelphiSDK

======= #RTXCPluginForDelphiSDK ##Delphi开发RTX插件 (Delphi XE6) 当初开发只是好奇, 所以去研究了下, 花了两天的时间翻译了原C++ SDK到Delphi 目录结构及祥细情况 Demo NotetipsGroup 一...

不在乎y ⋅ 2014/11/23 ⋅ 0

Google SketchUp 7——简单而不简单

自从在VeryCD下了个Google SketchUp 7后,这两天就一直在研究这个。 话说我在3D领域是一个骨灰级菜鸟,3Ds MAX摸了无数次也没入过门,而在SketchUp中却瞬间找到了感觉,很是Happy^^。 看看咱...

老朱教授 ⋅ 2017/11/07 ⋅ 0

【blade04】用面向对象的方法写javascript坦克大战

前言 javascript与程序的语言比如C#或者java不一样,他并没有“类”的概念,虽然最新的ECMAScript提出了Class的概念,我们却没有怎么用 就单以C#与Java来说,要到真正理解面向对象的程度也是...

范大脚脚 ⋅ 2017/11/16 ⋅ 0

#最新动态#下周工作计划

昨日完成了【攻克要塞软考高级论文公开课】课程的在线直播,上课讲义已转成PDF上传。 本周末在山东进行系统架构设计师两天面授冲刺培训,课程结束后将开始: 在51CTO发布论文批改服务包。此服...

liuyiok ⋅ 2015/10/14 ⋅ 0

VB源码升级后的几幅截图-VBIDE嵌入窗体、代码资源自动加入

用了整整两天的时间终于完成了功能上的升级,但多国语言版和数据库的加密还不知什么时间能做完,所以一时还不能发布,不过作为“内部人员”倒是可以先用为快:) 一、把原图标进行了修改,可...

刘洪峰iot ⋅ 2005/08/23 ⋅ 0

【技术专栏】传播业强身三步走:控成本、管创意、提效率

“听了很多道理,不如业者最实际的分享。” 本周起,《商旅专家》于微信公众平台开辟全新【技术专栏】板块。经验丰富的业界大佬将基于自身在行业的所得所感,通过此平台陆续与我们的读者开展...

玄学酱 ⋅ 05/25 ⋅ 0

rest只适合做api吗?

这两天一直在纠结一件事,要开发一个新的web项目,其中后台管理系统部分不知道适不适合用restful? 而我查找rest相关资料都是说做api的,而后台这块不打算做api+重前端方式(人手不足)。。 比较...

吾爱 ⋅ 2015/01/03 ⋅ 9

没有更多内容

加载失败,请刷新页面

加载更多

下一页

MySQL 数据库设计总结

规则1:一般情况可以选择MyISAM存储引擎,如果需要事务支持必须使用InnoDB存储引擎。 注意:MyISAM存储引擎 B-tree索引有一个很大的限制:参与一个索引的所有字段的长度之和不能超过1000字节...

OSC_cnhwTY ⋅ 58分钟前 ⋅ 0

多线程(四)

线程池和Exector框架 什么是线程池? 降低资源的消耗 提高响应速度,任务:T1创建线程时间,T2任务执行时间,T3线程销毁时间,线程池没有或者减少T1和T3 提高线程的可管理性。 线程池要做些什...

这很耳东先生 ⋅ 今天 ⋅ 0

使用SpringMVC的@Validated注解验证

1、SpringMVC验证@Validated的使用 第一步:编写国际化消息资源文件 编写国际化消息资源ValidatedMessage.properties文件主要是用来显示错误的消息定制 [java] view plain copy edit.userna...

瑟青豆 ⋅ 今天 ⋅ 0

19.压缩工具gzip bzip2 xz

6月22日任务 6.1 压缩打包介绍 6.2 gzip压缩工具 6.3 bzip2压缩工具 6.4 xz压缩工具 6.1 压缩打包介绍: linux中常见的一些压缩文件 .zip .gz .bz2 .xz .tar .gz .tar .bz2 .tar.xz 建立一些文...

王鑫linux ⋅ 今天 ⋅ 0

6. Shell 函数 和 定向输出

Shell 常用函数 简洁:目前没怎么在Shell 脚本中使用过函数,哈哈,不过,以后可能会用。就像java8的函数式编程,以后获取会用吧,行吧,那咱们简单的看一下具体的使用 Shell函数格式 linux ...

AHUSKY ⋅ 今天 ⋅ 0

单片机软件定时器

之前写了一个软件定时器,发现不够优化,和友好,现在重写了 soft_timer.h #ifndef _SOFT_TIMER_H_#define _SOFT_TIMER_H_#include "sys.h"typedef void (*timer_callback_function)(vo...

猎人嘻嘻哈哈的 ⋅ 今天 ⋅ 0

好的资料搜说引擎

鸠摩搜书 简介:鸠摩搜书是一个电子书搜索引擎。它汇集了多个网盘和电子书平台的资源,真所谓大而全。而且它还支持筛选txt,pdf,mobi,epub、azw3格式文件。还显示来自不同网站的资源。对了,...

乔三爷 ⋅ 今天 ⋅ 0

Debian下安装PostgreSQL的表分区插件pg_pathman

先安装基础的编译环境 apt-get install build-essential libssl1.0-dev libkrb5-dev 将pg的bin目录加入环境变量,主要是要使用 pg_config export PATH=$PATH:/usr/lib/postgresql/10/bin 进......

玛雅牛 ⋅ 今天 ⋅ 0

inno安装

#define MyAppName "HoldChipEngin" #define MyAppVersion "1.0" #define MyAppPublisher "Hold Chip, Inc." #define MyAppURL "http://www.holdchip.com/" #define MyAppExeName "HoldChipE......

backtrackx ⋅ 今天 ⋅ 0

Linux(CentOS)下配置php运行环境及nginx解析php

【part1:搭建php环境】 1.选在自己需要安装的安装包版本,wget命令下载到服务器响应目录 http://php.net/releases/ 2.解压安装包 tar zxf php-x.x.x 3.cd到解压目录执行如下操作 cd ../php-...

硅谷课堂 ⋅ 今天 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部