`

基于泛型DAO的Facade设计模式.

阅读更多
在做管理系统时。通常基于Facade模式的系统持久化层要写许多Dao。这些dao里面的方法又是重复的,那么有没有什么好的方法来统一利用一个公共的Dao。

答案是可以的。这里我们接触到JDK5.0里面的一个新特性:泛型。

关于泛型的含义我这里就不再解释了。

下面我们以一个对用户管理和新闻管理的来示范。

首先是2个POJO。我这里只列出User  POJO。
(基于注释的Pojo)
  
package com.oa;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;

@Entity
@Table(name = "tb_user")
public class User {

	@Id
	@GeneratedValue(strategy = GenerationType.IDENTITY)
	private int id;

	@Column(name = "username", length = 15)
	private String username;

	@Column(name = "password", length = 15)
	private String password;

	public int getId() {
		return id;
	}

	public void setId(int id) {
		this.id = id;
	}

	public String getUsername() {
		return username;
	}

	public void setUsername(String username) {
		this.username = username;
	}

	public String getPassword() {
		return password;
	}

	public void setPassword(String password) {
		this.password = password;
	}

}

 

如果按照常规的Facade模式来设计,我们的思路是:
先创建一个UserDao的接口。

package com.oa.dao;

import java.util.List;

import com.oa.User;

public interface UserDao {
	public void save(User user);

	public void delete(int id);

	public void update(User user);

	public List<User> query();

	public User get(int id);

}

 


然后实现这个接口:UserDaoImpl
package com.oa.dao.impl;

import java.util.List;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Repository;

import com.oa.User;
import com.oa.dao.MyHibernateDaoSupport;
import com.oa.dao.UserDao;



/**
 * 从Spring 2.0开始,引入了@Repository注解,
 *  用它来标记充当储存库(又称 Data Access Object或DAO)角色或典型的类
 */


/**
 * Spring 2.5引入了更多典型化注解(stereotype annotations): @Component、@Service和 @Controller。 
 * @Component是所有受Spring管理组件的通用形式; 而@Repository、@Service和 @Controller则是@Component的细化, 
 *                   用来表示更具体的用例(例如,分别对应了持久化层、 服务层  和   表现层)。
 */
//@Scope("singlton")
@Repository("userDao")//声明此类为数据持久层的类
public class UserDaoImpl extends MyHibernateDaoSupport implements UserDao {

	public void delete(int id) {
		super.getHibernateTemplate().delete(
				super.getHibernateTemplate().load(User.class, id));

	}

	public User get(int id) {

		return (User) super.getHibernateTemplate().get("from  User", id);
	}

	@SuppressWarnings("unchecked")
	public List<User> query() {
		return super.getHibernateTemplate().find("from User");

	}

	public void save(User user) {
		super.getHibernateTemplate().save(user);

	}

	public void update(User user) {
		super.getHibernateTemplate().update(user);

	}

}

 


持久化层完毕。


接下来的是事务层



先创建一个UserService的接口。


package com.oa.service;

import com.oa.User;

public interface UserService {
	
	public void save(User user);

	public void update(User user);

}
 


然后实现这个接口:UserServiceImpl。
在UserServiceImpl里引用UserDao来实现业务逻辑。


package com.oa.service.impl;

import com.oa.User;
import com.oa.service.UserService;
import com.oa.dao.UserDao;


import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;



/**
 * 声明此类为业务逻辑层的类
 * 默认bean名称生成器会返回小写开头的非限定(non-qualified)类名
 * @Service
 * userServiceImpl
 */
@Service("userService")
public class UserServiceImpl implements UserService {

	
	/**
	 * @Autowired
	 * 
	 * @Autowired 注解可以用于"传统的"setter 方法,如下例:
	 * public void setUserDao(UserDAO userDao)
	{
		this.userDao = userDao;
	}
	 */
	
	/**
	 * @Resource有一个'name'属性,缺省时,Spring 将这个值解释为要注射的 bean 的名字。
	 *   @Resource(name="userDao") 
	 */
	@Autowired //  or  @Resource(name="userDao")
    private UserDao userDao;

	public void save(User user) {
		userDao.save(user);

	}

	public void update(User user) {
		userDao.update(user);

	}

}

 

按照上面的模式:新闻管理也这么写一遍。

重复的工作使得我们觉得好烦。


这个时候是泛型出场的时候了。

基于Facade的设计模式,dao和service还是要的。 这里我们就要设计一个公共的Dao..  我们称之为:GenericDao
package com.oa.dao;

import java.io.Serializable;
import java.util.*;

/**
 * *
 * 
 * @param <T>
 *            泛型,指实体类  type
 * @param <PK>
 *            泛型,指实体类主键的数据类型,如Integer,Long
 */
public interface GenericDao<T, PK> {
	
	/**  
     * 保存指定实体类  
     *   
     * @param entityobj  
     *            实体类  
     */ 
	public  void save(T entity);
	
	
	  /**  
     * 删除指定实体  
     *   
     * @param entityobj  
     *            实体类  
     */  
	public void delete(T entity);
	
	
	 /** *
     * 删除实体
     * @param entityClass 实体类名
     * @param id 实体的ID
     */
    public void deleteById(Class<T> entityClass,PK id);
	
	
	/**  
    * 更新或保存指定实体  
    *   
    * @param entity 实体类  
    */ 
	public void saveorupdate(T entity);
	
	
	 /** *
     * 更新实体
     * 可用于添加、修改、删除操作
     * @param hql 更新的HQL语句
     * @param params 参数,可有项目或多项目,代替Hql中的"?"号
     */
    public void update(final String hql,final Object[] params);
    
    

    /**  
     * 模糊查询指定条件对象集合 <br>  
     * 用法:可以实例化一个空的T对象,需要查询某个字段,就set该字段的条件然后调用本方法<br>  
     * 缺点:目前测试貌似只能支持String的模糊查询,虽然有办法重写,但没必要,其他用HQL<br>  
     *   
     * @param entity  
     *            条件实体  
     * @return 结合  
     */  
    public List<T> findByExample(T entity);
    
    
    /**  
     * 获取所有实体集合  
     *   
     * @param entityClass  
     *            实体  
     * @return 集合  
     */  
    public List<T> findAll(Class<T> entityClass);
    
    public List<T> findAll(Class<T> entityClass,String hql,Object[] params,int start, int limit);
	/**  
     * 查找指定PK实体类对象  
     *   
     * @param entityClass  
     *            实体Class  
     * @param id  
     *            实体PK  
     * @return 实体对象  
     */   
    public T findById(Class<T> entityClass, PK id);
    
    /** *
     * 按HQL条件查询列表
     * @param hql 查询语句,支持连接查询和多条件查询
     * @param params 参数数组,代替hql中的"?"号
     * @return 结果集List
     */
   
    public List<T> findByHql(String hql,Object[]  params);
    
    /**  
     * 查找指定属性的实体集合  
     *   
     * @param entityClass  
     *            实体  
     * @param propertyName  
     *            属性名  
     * @param value  
     *            条件  
     * @return 实体集合  
     */  
    public List<T> findByProperty(Class<T> entityClass, String propertyName,Object value);
    
    
    /**  
     * 查询指定HQL语句的分页数据集合  
     *   
     * @param hsql  
     *            HQL语句  
     * @param start  
     *            开始记录号  
     * @param limit  
     *            最大记录号  
     * @return 分页数据集合  
     * @throws Exception  
     *             抛出异常  
     */  
    public List<T> findByPage(Class<T> entityClass,int start,int limit) ;
    
    
    
    /**
     * 获得总记录数
     */
    public T getTotalCount(Class<T> entityClass);
    
    public T getPageCount(String hql,Object[] params);

}
 



看到,我们不再是具体的User , News
。。而是用 T  来取代实体。


因为我这个是基于 注解的,所以附上MyHibernateDaoSupport的代码。


package com.oa.dao;

import javax.annotation.Resource;
import org.hibernate.SessionFactory;
import org.springframework.orm.hibernate3.support.HibernateDaoSupport;

/**
 * 我们之所以要改写
 * HibernateDaoSupport,是因我为,我们要为DAO层的类注入SessionFactory这个属性。
 * 以后,我们开发的DAO类,就可以直接重用这个MyHibernateDaoSupport了。
 * 其实,这样做是相当于配置文件方式的代码:
 * <bean id="userDao" class="com.oa.dao.UserDaoImpl"> 
 * <property
 * name="sessionFactory" ref="sessionFactory"/> 
 * </bean>
 * 
 * @author Administrator
 * 
 */
public class MyHibernateDaoSupport extends HibernateDaoSupport {
    
    @Resource(name="sessionFactory")    //为父类HibernateDaoSupport注入sessionFactory的值
    public void setSuperSessionFactory(SessionFactory sessionFactory){
        super.setSessionFactory(sessionFactory);
    }
    

}

 


到现在位置genericdao的接口有了,也就是我们要做什么。。现在就是实现它,就是怎么做。
GenericDaoImpl 代码:

package com.oa.dao.impl;

import java.io.Serializable;
import java.util.List;

import org.hibernate.Query;
import org.springframework.stereotype.Repository;

import com.oa.dao.GenericDao;
import com.oa.dao.MyHibernateDaoSupport;
@SuppressWarnings("unchecked")
@Repository("genericDao")   //声明此类为数据持久层的类
public class GenericDaoImpl<T, PK extends Serializable> extends
                                    MyHibernateDaoSupport implements GenericDao<T, PK> {

	public void delete(T entity) {
		super.getHibernateTemplate().delete(entity);
	}


	public void deleteById(Class entityClass, PK id) {
		super.getHibernateTemplate().delete(findById(entityClass, id));

	}
	
	public void save(T entity) {
		super.getHibernateTemplate().save(entity);

	}

	public void saveorupdate(T entity) {
		super.getHibernateTemplate().saveOrUpdate(entity);

	}

	public void update(String hql, Object[] params) {
		     Query query = super.getSession().createQuery(hql);
                for(int i=0; i<params.length; i++){
                    query.setParameter(i, params[i]);
                }
                query.executeUpdate();
     }
	

	public List<T> findAll(Class entityClass) {
		
		return super.getHibernateTemplate().loadAll(entityClass);
	}

	public List<T> findAll(Class entityClass, String hql, Object[] params,int start, int limit) {
		Query query = super.getSession().createQuery(hql);
		if(params!=null&&params.length>0){
			for(int i = 0;i<params.length;i++){
				query.setParameter(i, params[i]);
			}
		}
		if(start!=0&&limit!=0){
			query.setFirstResult(start).setMaxResults(limit);
		}
		return query.list();
	}

	public List<T> findByExample(T entity) {
		return super.getHibernateTemplate().findByExample(entity);
	}

	public List<T> findByHql(String hql, Object[] params) {
		Query query = super.getSession().createQuery(hql);
		if(null!= params && params.length>0){
			for(int i = 0; i<params.length;i++){
				query.setParameter(i, params[i]);
			}
		}
		return query.list();
	}

	public T findById(Class entityClass, PK id) {
		return (T)super.getHibernateTemplate().get(entityClass, id);
	}
	
	public List<T> findByProperty(Class entityClass, String propertyName,Object value) {
		String queryString = "from "+entityClass.getName()+ " as model where model." + propertyName + "=?";   
		return super.getHibernateTemplate().find(queryString, value);
	}


	
	//分页使用
	public List<T> findByPage(Class<T> entityClass,int start,int limit) {
		  Query query=super.getSession().createQuery("select o from "+entityClass.getName()+" o");
		  query.setFirstResult(start).setMaxResults(limit);
		return query.list();
	}

	
	public T getTotalCount(Class entityClass) {
		
		return (T)super.getSession().createQuery("select count(o) from "+entityClass.getName()+" o").uniqueResult();
	}

	public T getPageCount(String hql, Object[] params) {
		Query query = super.getSession().createQuery(hql);
		if(null!= params && params.length>0){
			for(int i = 0; i<params.length;i++){
				query.setParameter(i, params[i]);
			}
		}
		return (T)query.list();
	}

	

}


至此 泛型就告一个段落。


接下来日子就好过了。

我们不是有user  news   等等一系列的curd管理。


以User为例子;
定义一个user的接口,
UserDao.Java
package com.oa.dao;

import com.oa.User;

public interface UserDao extends GenericDao<User, Integer> {

public   int   login(User user);
//其他的方法的
}


然后就是实现它 UserDaoImpl

package com.oa.dao.impl;



import com.oa.User;
import com.oa.dao.UserDao;

public class UserDaoImpl extends GenericDaoImpl<User, Integer> implements UserDao {

	public  int  login(User  user){
//登陆判断的方法

return   XX;
};


//其他的方法的实现


}

 

持久化层就是这么多了。

下面进入业务逻辑层,依然是先定义一个接口。

package com.oa.service;

import com.oa.User;

public interface UserService {
	
	public void save(User user);

	public void update(User user);

   public  int  login(User  user);

//其他的方法

}
 


接下来是实现


package com.oa.service.impl;

import com.oa.User;
import com.oa.dao. UserDao;
import com.oa.service.TestUserService;

public class UserService implements UserService {

	private  UserDao  UserDao;
	public void save(User user) {
		UserDao.save(user);

	}

	public void updasaveorupdatete(User user) {
		UserDao.saveorupdate(user);

	}

	public int   login(User user) {
		return   UserDao.login(user);

	}

//其他的方法。。。。

}

 


Ok。。到现在我们就利用泛型dao来设计就完毕了

两者相对比,发现dao层的代码可以复用,少了不少。

对于大型管理系统,效果更明显。


Junit  Test方法我这里就不提供了。

参考:rongxh7
吼吼。谢谢!  

欢迎大家提出意见,但是拒绝人身攻击。
分享到:
评论
14 楼 gdufsbobo 2010-01-14  
我见过有一种,基本的增,删,查,改都不用写一个独产的DAO,有一个共用的SERVICE提供这些服务,更方便。对一般的java bean操作直接使用里面的方法即可。但似乎不够严谨。
13 楼 nurenok 2010-01-06  
貌似有点缺陷
12 楼 zhmiao 2009-07-02  
强大的泛型  解决的办法都是大同小异 期待亮点
11 楼 yangtao309 2009-07-02  
看下springside的DAO设计吧....
10 楼 liujianst 2009-07-01  
star022 写道
这绝不是泛型的最佳实践!

最佳实践是什么
9 楼 star022 2009-06-24  
这绝不是泛型的最佳实践!
8 楼 zhiblin 2009-06-23  
这个思路深入人心了
7 楼 crabboy 2009-06-22  
g81997842 写道
首先,这种方式确实不错,但是,说真话,这种设计在JAVAEYE里都能搜索到很多,泛滥了!都是一个意思!

支持这个。大同小异而已。
6 楼 aaronluo 2009-06-21  
我觉得这也算楼主开发经验的一个升华吧。我个人也有过开发此类泛型Dao,当时就是为了简化Dao层开发的目的,参考的是Spring自带的PetStore里面JPA的实现。
5 楼 tiyi 2009-06-21  
g81997842 写道
首先,这种方式确实不错,但是,说真话,这种设计在JAVAEYE里都能搜索到很多,泛滥了!都是一个意思!


je上确实是蛮多了。现在就是看大家标新立异能出点什么新东西了。。。
4 楼 g81997842 2009-06-21  
首先,这种方式确实不错,但是,说真话,这种设计在JAVAEYE里都能搜索到很多,泛滥了!都是一个意思!
3 楼 raymond2006k 2009-06-20  
泛型在减少dao,service类或代码量上很有用,很强大, 楼主的设计思路不错。
2 楼 treblesoftware 2009-06-08  
支持LZ一下,跟我那个泛型DAO设计的差不多,不过感觉总是缺少点什么。等我把我的项目做起来,在做总结吧。
1 楼 ftj20003 2009-06-06  
谢谢,值得借鉴和学习。getTotalCount()与getPageCount()返回的不是数值?

相关推荐

    java开发文档.pdf

    文档中列举了大量的代码片段和类名,这些信息对于理解Java开发中各个层次的职责、设计模式、异常处理机制、服务与DAO层的操作方法等方面是十分重要的。 根据以上知识点,可以推测文档是Java项目的开发文档,可能...

    用OpenGL开发的机械臂运动仿真程序,并且实现机械手臂向四个方向的旋转.rar

    OpenGL是一种强大的图形库,用于创建2D和3D图形,广泛应用于游戏开发、科学可视化、工程设计等领域。在这个项目中,我们看到一个基于OpenGL的机械臂运动仿真程序,它能够实现机械臂在四个方向上的旋转。这样的模拟对于理解机械臂的工作原理、机器人控制算法以及进行虚拟环境中的机械臂运动测试具有重要意义。 我们需要了解OpenGL的基础知识。OpenGL是一个跨语言、跨平台的编程接口,用于渲染2D和3D矢量图形。它提供了大量的函数来处理图形的绘制,包括几何形状的定义、颜色设置、光照处理、纹理映射等。开发者通过OpenGL库调用这些函数,构建出复杂的图形场景。 在这个机械臂仿真程序中,C#被用来作为编程语言。C#通常与Windows平台上的.NET Framework配合使用,提供了一种面向对象的、类型安全的语言,支持现代编程特性如LINQ、异步编程等。结合OpenGL,C#可以构建高性能的图形应用。 机械臂的运动仿真涉及到几个关键的计算和控制概念: 1. **关节角度**:机械臂的每个部分(或关节)都有一个或多个自由度,表示为关节角度。这些角度决定了机械臂各部分的位置和方向。 2. **正向运动学**:根据关节角度计算机械臂末端执行器(如抓手)在空间中的位置和方向。这涉及将各个关节的角度转换为欧拉角或四元数,然后转化为笛卡尔坐标系的X、Y、Z位置和旋转。 3. **反向运动学**:给定末端执行器的目标位置和方向,计算出各关节所需的理想角度。这是一个逆向问题,通常需要解决非线性方程组。 4. **运动规划**:确定从当前状态到目标状态的路径,确保机械臂在运动过程中避免碰撞和其他约束。 5. **OpenGL的使用**:在OpenGL中,我们首先创建几何模型来表示机械臂的各个部分。然后,使用矩阵变换(如旋转、平移和缩放)来更新关节角度对模型的影响。这些变换组合起来,形成机械臂的动态运动。 6. **四向旋转**:机械臂可能有四个独立的旋转轴,允许它在X、Y、Z三个轴上旋转,以及额外的绕自身轴线的旋转。每个轴的旋转都由对应的关节角度控制。 7. **交互控制**:用户可能可以通过输入设备(如鼠标或键盘)调整关节角度,实时观察机械臂的运动。这需要将用户输入转换为关节角度,并应用到运动学模型中。 8. **图形渲染**:OpenGL提供了多种渲染技术,如深度测试、光照模型、纹理映射等,可以用于提高机械臂模拟的真实感。例如,可以添加材质和纹理来模拟金属表面,或者使用光照来增强立体感。 这个项目结合了OpenGL的图形渲染能力与C#的编程灵活性,构建了一个可以直观展示机械臂运动的仿真环境。通过理解并实现这些关键概念,开发者不仅能够学习到图形编程技巧,还能深入理解机器人学的基本原理。

    android11 udpate-engine 系统升级模块源码

    android11 udpate-engine 系统升级模块源码下载

    MATLAB环境下SVM二分类算法的实现与参数优化

    内容概要:本文详细介绍了如何在MATLAB环境中实现SVM二分类算法,涵盖数据预处理、参数寻优及结果可视化的全过程。首先进行数据归一化处理,确保各特征在同一量纲下参与模型训练。接着采用网格搜索法对SVM的关键参数c(惩罚系数)和g(核参数)进行自动化寻优,利用5折交叉验证评估每组参数的表现。最后通过等高线图和3D曲面图直观展示参数与准确率之间的关系,并完成最终模型的训练与预测。 适合人群:具有一定MATLAB编程基础的研究人员和技术爱好者,尤其是从事机器学习、数据分析领域的从业者。 使用场景及目标:适用于需要快速搭建SVM二分类模型并进行参数调优的项目。主要目标是在短时间内获得较高准确度的分类结果,同时掌握SVM的工作原理及其在MATLAB中的具体应用方法。 其他说明:文中提供了完整的代码示例,便于读者直接上手实践。此外还提到了一些常见的注意事项,如数据格式要求、类别不平衡处理以及特征工程的重要性等。

    ffmpeg liblame pcm转mp3教程.zip

    ffmpeg

    江科大CAN入门教程,万字长文理解

    江科大CAN入门教程,万字长文理解

    移动电源设计方案详解:基于新唐N79E814单片机的双路输出PCB设计与实现

    内容概要:本文详细介绍了基于新唐N79E814单片机的移动电源设计方案,涵盖硬件架构、PCB原理图、电路设计、代码实现等方面。移动电源主要由电池、充电电路和输出电路构成,文中重点讲解了5V1A和5V2.1A两路输出的设计思路,包括同步整流、PWM控制、充电管理等关键技术。同时,文章还探讨了PCB布局、烧录注意事项、效率优化等内容,并提供了具体的代码示例和调试建议。 适合人群:具有一定电子技术和单片机开发基础的工程师和技术爱好者。 使用场景及目标:适用于希望深入了解移动电源设计原理和实现方法的人群,旨在帮助读者掌握从原理图绘制到实际产品制作的全过程,提升电路设计和调试能力。 其他说明:文章不仅提供了理论知识,还包括大量实践经验分享,如常见的调试陷阱和解决方法,有助于读者在实践中少走弯路。

    动漫角色分割-基于深度学习实现的高精度动漫角色分割算法-附项目源码-优质项目实战.zip

    动漫角色分割_基于深度学习实现的高精度动漫角色分割算法_附项目源码_优质项目实战

    一款Java通过javacv实现的支持各种音视频播放的播放器项目源码

    javacv实现的支持多种音视频播放的播放器,比如MP4、avi、mkv、flv、MP3、ogg、wav、au等多种音视频格式,非常好用。

    浏览器插件+JS+自动登录+demo

    开发调试demo,简单的自动登录功能,插件开发入门参考

    【计算机管理与注册表编辑】Windows系统管理员工具与用户账户管理:本地用户和组、远程桌面配置及SAM权限设置,具体教学与批处理可私信我

    内容概要:本文详细介绍了通过修改Windows注册表来启用和配置被禁用的用户账户(如WDAGUtilityAccount)的过程。首先,通过计算机管理界面查看被禁用的用户账户,并进入注册表编辑器定位到HKEY_LOCAL_MACHINE\SAM\SAM\Domains\Account\Users路径下的相应用户条目。接着,通过对特定用户的二进制数据进行编辑,包括复制和修改关键字段,实现对被禁用账户的克隆与重新启用。最后,验证账户状态的变化,并通过远程桌面连接测试新配置的有效性。 适合人群:具备一定Windows系统管理基础的技术人员,尤其是负责企业内部网络和用户账户管理的IT管理员。 使用场景及目标:①当需要恢复或重新配置被禁用的用户账户时;②在进行系统故障排除或安全审计时,了解如何通过注册表直接操作用户账户;③确保特定用户能够正常登录并访问远程桌面服务。 阅读建议:本文涉及较为底层的系统操作,建议读者在实际操作前充分备份系统和注册表,避免误操作导致系统不稳定。同时,对于不熟悉注册表编辑的用户,应先在测试环境中练习,确保掌握相关技能后再应用于生产环境。此外,建议结合官方文档或其他权威资料,加深对Windows用户账户管理机制的理解。

    新冠抗原自测平台 2025免费JAVA微信小程序毕设

    2025免费微信小程序毕业设计成品,包括源码+数据库+往届论文资料,附带启动教程和安装包。 启动教程:https://www.bilibili.com/video/BV1BfB2YYEnS 讲解视频:https://www.bilibili.com/video/BV1BVKMeZEYr 技术栈:Uniapp+Vue.js+SpringBoot+MySQL。 开发工具:Idea+VSCode+微信开发者工具。

    基于S7-1200 PLC的两部六层电梯控制系统设计与梯形图编程详解

    内容概要:本文详细介绍了基于西门子S7-1200 PLC的两部六层电梯控制系统的设计与实现。主要内容涵盖前期准备工作,如选择合适的PLC型号和配置硬件;核心逻辑部分深入讲解了梯形图编程的具体实现方法,包括楼层呼叫逻辑、电梯运行方向控制以及两部电梯之间的协同工作;此外,文章还探讨了仿真测试的方法及其重要性,提供了许多实用技巧和注意事项。通过具体实例展示了如何利用博途V15软件进行电梯系统的开发,并分享了一些实际操作中的经验和常见问题解决方案。 适合人群:从事工业自动化领域的工程师和技术人员,特别是那些对PLC编程有兴趣或者正在参与类似项目的从业者。 使用场景及目标:适用于需要理解和掌握S7-1200 PLC编程技能的人群,尤其是希望通过实际案例加深对梯形图编程理解的学习者。目标是在实践中提高编程能力,能够独立完成类似的工程项目。 其他说明:文中不仅包含了详细的理论解释,还有丰富的代码片段供读者参考。对于初学者而言,建议先从单部电梯开始练习,逐步过渡到复杂的双梯联调。同时,作者强调了仿真测试的重要性,指出这是验证程序正确性和优化性能的关键步骤。

    农产品自主供销系统 2025免费JAVA微信小程序毕设

    2025免费微信小程序毕业设计成品,包括源码+数据库+往届论文资料,附带启动教程和安装包。 启动教程:https://www.bilibili.com/video/BV1BfB2YYEnS 讲解视频:https://www.bilibili.com/video/BV1BVKMeZEYr 技术栈:Uniapp+Vue.js+SpringBoot+MySQL。 开发工具:Idea+VSCode+微信开发者工具。

    大学英语四级听力练习音频MP3

    内容概要:该资源为大学英语四级听力练习音频 MP3,包含丰富多样的听力素材。涵盖四级考试常见的各类场景,如校园生活(课程学习、社团活动等)、日常社交(聚会、聊天等)、工作求职(面试、职场事务等)、旅行交通(出行方式、景点介绍等)、饮食健康(餐厅点餐、健康养生等)。音频内容依照四级听力考试题型和难度精心录制,有短对话、长对话、短文听力等形式,且语速、口音等符合四级考试要求,助力考生熟悉考试形式与节奏。 适合人群:正在备考大学英语四级考试,希望提升听力水平的学生;英语基础中等,需要通过针对性练习来适应四级听力难度、提升听力理解能力的学习者;对英语听力学习有需求,想通过大量练习积累场景词汇、熟悉英语表达习惯的人群。 能学到什么:①熟悉四级听力考试的各类场景词汇,增强词汇储备并提升在听力语境中的反应速度;②掌握不同场景下的英语常用表达和句式,提升英语语言运用能力;③锻炼听力理解技巧,如抓取关键词、推断隐含意思、梳理篇章逻辑等;④适应四级听力考试的语速、口音和题型设置,增强应试能力和自信心。 阅读建议:制定系统的练习计划,定期定量进行听力练习,如每天安排 30 - 60 分钟;第一遍泛听,了解大致内容和主题;第二遍精听,逐句听写或分析不懂的词汇和句子;对照听力原文,明确错误和没听懂的地方,积累生词和表达;定期进行模拟测试,利用该音频模拟考试环境,检验学习效果并调整学习策略。

    2000-2017年各省天然气消费量数据

    2000-2017年各省天然气消费量数据 1、时间:2000-2017年 2、来源:国家统计j、能源nj 3、指标:行政区划代码、城市、年份、天然气消费量 4、范围:31省

    西门子PLC1200与库卡机器人协同控制系统的多设备集成方案

    内容概要:本文详细介绍了基于西门子PLC1200的自动化控制系统,涵盖了PLC与库卡机器人通过Profinet通讯、PTO模式控制松下伺服、36路模拟量处理(包括压力检测、位置检测及压力输出)、以及26个温控器通过485总线通讯的关键技术和实现方法。此外,还包括了昆仑通态触摸屏的人机交互界面设计,提供了详细的硬件组态、软件编程指导和设备操作说明。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是那些负责多设备协同控制项目的设计和实施的专业人士。 使用场景及目标:适用于需要整合多种设备(如PLC、机器人、伺服系统、温控器等)的复杂自动化生产线。主要目标是提高生产效率、增强系统的稳定性和可靠性,同时降低维护成本。 其他说明:文中不仅提供了具体的编程实例和硬件配置指南,还分享了许多实际调试过程中积累的经验教训,有助于读者在实际应用中少走弯路。

    汽车电子基于AUTOSAR的BSW层功能详解:服务、驱动、接口与管理器模块设计及应用了文档的主要内容

    内容概要:本文深入探讨了AUTOSAR BSW(Basic Software)层所提供的各类服务、驱动、接口和管理器模块及其功能。BSW提供的服务包括I/O、Memory、Crypto、Communication、Off-board Communication和System等,涵盖了标准化的访问方式以确保不同硬件和系统的兼容性与安全性。BSW里的驱动分为内部驱动和外部驱动,分别用于控制和访问微控制器内部和外部的设备,确保硬件功能的正常运作。BSW里的接口(xx_IF)对下层模块进行抽象和封装,提供标准API接口,使上层应用无需关注底层硬件细节。BSW里的管理器(xxxM)则为多个客户端提供特定服务,能够修改或适配调整一些数据,以满足复杂需求。此外,文中还简要介绍了AUTOSAR里的库文件,它们是无状态的函数集合,可被多个模块调用以实现特定功能。 适合人群:对汽车电子软件架构有一定了解,尤其是对AUTOSAR标准感兴趣的工程师和技术人员。 使用场景及目标:①理解BSW层提供的各类服务及其应用场景;②掌握BSW中驱动、接口和管理器模块的设计原理和功能;③了解库文件的作用及其在AUTOSAR架构中的位置。 其他说明:本文详细解释了BSW层各组件的功能和作用,帮助读者更好地理解AUTOSAR架构的设计思想和实现方法。建议读者结合实际项目经验,深入研究各模块的具体实现和应用场景。

    西门子1200伺服步进FB块程序:支持多轴调用的自动化控制解决方案

    内容概要:本文介绍了西门子1200系列PLC的伺服步进FB块程序,该程序由两个FB块组成,分别采用SCL语言和梯形图编写,支持PTO和PN模式,适用于多种伺服系统和步进电机。程序经过实际调试,稳定性高,兼容性强,能够灵活应用于单轴或多轴控制系统。文中提供了详细的代码示例和调试指南,帮助用户快速上手并解决常见问题。 适合人群:从事工业自动化控制领域的工程师和技术人员,尤其是使用西门子1200系列PLC进行项目开发的人员。 使用场景及目标:①用于单轴或多轴伺服步进系统的控制;②提高项目的开发效率和稳定性;③减少重复开发的工作量,加快项目进度。 其他说明:程序需要在TIA V14及以上版本打开,附带详细的文档说明和实际项目视频链接,方便用户理解和使用。

    基于SpringBoot+Vue的电子招投标系统源码+数据库.zip

    基于SpringBoot+Vue的电子招投标系统源码+数据库.zip 高分通过项目,已获导师指导。 本项目是一套基于Springboot的电子招投标系统,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的Java学习者。也可作为课程设计、期末大作业 包含:项目源码、数据库脚本、开发说明文档、演示视频等,该项目可以直接作为毕设使用。 项目都经过严格调试,确保可以运行! 基于SpringBoot+Vue的电子招投标系统源码+数据库.zip 高分通过项目,已获导师指导。 本项目是一套基于Springboot的电子招投标系统,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的Java学习者。也可作为课程设计、期末大作业 包含:项目源码、数据库脚本、开发说明文档、演示视频等,该项目可以直接作为毕设使用。 项目都经过严格调试,确保可以运行! 基于SpringBoot+Vue的电子招投标系统源码+数据库.zip 高分通过项目,已获导师指导。 本项目是一套基于Springboot的电子招投标系统,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的Java学习者。也可

Global site tag (gtag.js) - Google Analytics