`

在SimpleFormController中用“SessionForm”概念构建修改信息页面

    博客分类:
  • Java
阅读更多
在通常的修改信息页面中都会先显示一张表单包含原有信息,用户修改原有信息表单然后提交完成信息的修改。

通过继承SimpleFormController复写部分方法来实现这一功能:
第一: 复写“protected ModelAndView showForm(HttpServletRequest request, HttpServletResponse response, BindException errors, Map controlModel) ”方法。
该方法用于显示包含原有信息的表单,在该方法中实现如下功能:
  1. 提取Command对象事例。 Spring Framework在调用该方法前已经创建了配置文件中所指定的Command对象,可以通过“errors.getBindingResult().getTarget()”来获取Command对象
  2. 提取被修改信息的ID,该ID信息可以通过页面上下文放在URL query string中。
  3. 调用对应的Service方法根据ID获取原始数据信息并加入到Command对象中。
  4. 将Command对象加入到session中“session.setAttribute(cmdNameInSession, cmd);”, 该session变量的名称需要通过AbstractFormController的“getFormSessionAttributeName (HttpServletRequest request)”方法获得。
  5. 设置AbstractFormController的“setSessionForm(true)”将该FormController设置为 SessionForm,这样在用户修改完Form信息再次提交时Spring Framework不会再创建一个新的Command对象,而是从Session中取出上一次保存在session中的command对象并将用户新提交 的值付给原有的Command对象。在AbstractFromController的“protected final Object getCommand(HttpServletRequest request)”方法中可以看到如上所述的逻辑
第二:复写doSubmitAction方法。 在该方法中调用对应的Service来完成数据的修改。

在这里有一个SessionForm的概念,通过Session来传递Command对象,用户修改表单并提交后,保存在Session中的 Command对象也会被修改,这样只要从Session中获取修改后的Command对象并交由对应的Service就可以完成数据的修改工作。类似与 “AbstractWizardFormController”中一页页传递Command对象的做法。

Spring Framework是在什么时候修改Session中的这个Command对象的还需要进一步研究。
分享到:
评论
11 楼 superzn 2007-12-13  
学习一下,不错~
10 楼 diyun 2007-07-31  
dynamo2 写道

是的,我的意思是不设置SessionForm为True,直接重载formBackingObject方法,就是你提到过的方式。然后在formBackingObject方法中根据请求类型对原始数据进行Cache存取操作。

一个简单的Cache定义:
public final class SessionCommandUtil {
	private static Map<String, Object> sessionCmd = 
							new HashMap<String, Object>();
	private static long subId = 0;

	public static synchronized String addCommand(Object cmd) {
		
		String key = Long.toString(subId++);
		sessionCmd.put(key, cmd);
		if (subId == Long.MAX_VALUE)
			subId = 0;

		return key;
	}

	public static synchronized Object getCommand(String id) {
		
		return sessionCmd.get(id);
	}

}


你的Cache理论上没有任何问题,只是实际程序中可能需要加以改进。一个好的Cache可能需要很多时间和精力去实现,所以我认为最好的办法是用现成的。实际应用中,我们可以把Cache交给Hibernate去处理。
9 楼 dynamo2 2007-07-20  
diyun 写道

当访问 product.form?id=1 时,session里保存的是产品1的信息,然后访问 product.form?id=2 这时,session里的信息变成了产品2,如果你切换到产品1,更新产品1,应该会出现意外结果。


啊~,知道你的意思了,这确是个问题。这样看来放在Session中是肯定会碰到问题的了。不过通过自定义的Cache也是可以解决数据库两次查询的问题。只是Cache中原始数据的key是不能放在session中的了,是要放在表单的某个隐藏值中待POST回来时通过他定位原始数据。

diyun 写道

另外,在 AbstractFormController 中,使用sessionForm 时,整个对象都放在session中,你没有选择。


是的,我的意思是不设置SessionForm为True,直接重载formBackingObject方法,就是你提到过的方式。然后在formBackingObject方法中根据请求类型对原始数据进行Cache存取操作。

一个简单的Cache定义:
public final class SessionCommandUtil {
	private static Map<String, Object> sessionCmd = 
							new HashMap<String, Object>();
	private static long subId = 0;

	public static synchronized String addCommand(Object cmd) {
		
		String key = Long.toString(subId++);
		sessionCmd.put(key, cmd);
		if (subId == Long.MAX_VALUE)
			subId = 0;

		return key;
	}

	public static synchronized Object getCommand(String id) {
		
		return sessionCmd.get(id);
	}

}
8 楼 diyun 2007-07-20  
dynamo2 写道

喔,如果用户开了两个窗口他们记录在session中的对象是两个不同的实例,这个冲突是存在数据的写写冲突。这个冲突即便不用Session存放原始数据也会存在。

对于Session对资源的占用问题我的想法是可以将原始的完成数据保存在一个全局Cache中,以一个唯一ID作为cache entry的key,根据这个key来存取数据对象。Cache的操作流程如下:

    1. 在Get请求时,生成唯一ID,将原始数据放入Cache中。
    2. 将唯一ID计入Session中
    3. 在POST请求时在Session中取出唯一ID
    4. 删除Session中记录此唯一ID的属性
    5. 根据ID取出原始数据
    6. 删除Cache中的原始数据
    7. 更新数据对象
    8. 更新数据库


这样是把数据换成数据的ID,也就是说资源的存放位置换了个地方。


先澄清一点,我们这里是讨论HttpSession。

假设我们有一个ProductFormController,可以修改产品信息。

同一个用户可以打开多个窗口,譬如

product.form?id=1
product.form?id=2

而session的属性名是一样的,下面获得属性名的代码。clas名和command名都是一样的。

	protected String getFormSessionAttributeName() {
		return getClass().getName() + ".FORM." + getCommandName();
	}


当访问 product.form?id=1 时,session里保存的是产品1的信息,然后访问 product.form?id=2 这时,session里的信息变成了产品2,如果你切换到产品1,更新产品1,应该会出现意外结果。

另外,在 AbstractFormController 中,使用sessionForm 时,整个对象都放在session中,你没有选择。
7 楼 dynamo2 2007-07-20  
diyun 写道


我觉得额外的数据库查询是需要的,当你更新一条记录的时候,form表单里可能没有所有的字段,所以对于这些字段,应该保留原有值。另外,和其他表的关系可能也不在表单里。所以修改记录时,先取出原来的值,然后和用户的输入合并(bind),可能是比较好的办法。



嗯,这个和取出原始数据的方式有关系。我是将被修改对象的所有相关数据都取出来的,所以有些个别字段是用户不可修改的比如表的ID,关联表的ID等等,但是这些数据又是在修改数据时所必须的。所以我的最初想法就是在取出数据后需要有个地方保留这些完整的原始数据对象,在表单提交后更新这些原始的数据对象,最后更新到数据库中。这种做法的好处是不需要两次查询数据库,不过如果有多个人同时更新同一个表的同一条记录,是会出现脏数据的问题。

所幸的是,当前我考虑的应用修改的是个人数据,就是说登录用户自己的数据,逻辑上不允许出现多人个人同时修改同一条记录。

diyun 写道

我的理解是:
如果同一个人同时用这个FormController不止一次(譬如同时打开多个tab),由于session的属性名字是一样的,从一个tab切换到另一个tab就会出现冲突。

而且session是全局的,估计会占用系统资源,session多了之后,可能增加调试的难度。如果你不小心在request中设置同样的属性,我记得session的属性优先,也就是说你在页面拿不到request的值。

仅供参考。


喔,如果用户开了两个窗口他们记录在session中的对象是两个不同的实例,这个冲突是存在数据的写写冲突。这个冲突即便不用Session存放原始数据也会存在。

对于Session对资源的占用问题我的想法是可以将原始的完成数据保存在一个全局Cache中,以一个唯一ID作为cache entry的key,根据这个key来存取数据对象。Cache的操作流程如下:

    1. 在Get请求时,生成唯一ID,将原始数据放入Cache中。
    2. 将唯一ID计入Session中
    3. 在POST请求时在Session中取出唯一ID
    4. 删除Session中记录此唯一ID的属性
    5. 根据ID取出原始数据
    6. 删除Cache中的原始数据
    7. 更新数据对象
    8. 更新数据库


这样是把数据换成数据的ID,也就是说资源的存放位置换了个地方。
6 楼 diyun 2007-07-20  
dynamo2 写道


如果在你的代码中没有设置SessionForm为True的话,在POST请求过来时,“getCommand”会再次调用“formBackingObject”来创建新的Object,如果在你覆写的“formBackingObject”方法中包含了数据库查询操作的话,就以位置会再次查询数据库然后返回一个新的Command对象,之后在Spring Framework中回把表单中提交的信息设置到Command对象中。结果就是和设置SessionForm一样,但不同只出就是需要再次查询数据库。

所以如果不用Session,又不想额外的查询数据库的话最好在覆写的“formBackingObject”中根据request的类型来决定是否需要再次查询数据库代码类似于:
protected Object formBackingObject(HttpServletRequest request) throws Exception {
	if (isFormSubmission(request)) {
		//1. Return saved command object
	}
	else {
		//1. Query Database
		//2. Save result to Command object
	}
}



我觉得额外的数据库查询是需要的,当你更新一条记录的时候,form表单里可能没有所有的字段,所以对于这些字段,应该保留原有值。另外,和其他表的关系可能也不在表单里。所以修改记录时,先取出原来的值,然后和用户的输入合并(bind),可能是比较好的办法。

dynamo2 写道
可以具体说明使用Session为什么不好吗?


我的理解是:
如果同一个人同时用这个FormController不止一次(譬如同时打开多个tab),由于session的属性名字是一样的,从一个tab切换到另一个tab就会出现冲突。

而且session是全局的,估计会占用系统资源,session多了之后,可能增加调试的难度。如果你不小心在request中设置同样的属性,我记得session的属性优先,也就是说你在页面拿不到request的值。

仅供参考。
5 楼 dynamo2 2007-07-19  
diyun 写道
sessionForm的默认值是false,在我的FormController中也没有设为true,所以目前我的FormController应该没有用sessionForm。


如果在你的代码中没有设置SessionForm为True的话,在POST请求过来时,“getCommand”会再次调用“formBackingObject”来创建新的Object,如果在你覆写的“formBackingObject”方法中包含了数据库查询操作的话,就以位置会再次查询数据库然后返回一个新的Command对象,之后在Spring Framework中回把表单中提交的信息设置到Command对象中。结果就是和设置SessionForm一样,但不同只出就是需要再次查询数据库。

所以如果不用Session,又不想额外的查询数据库的话最好在覆写的“formBackingObject”中根据request的类型来决定是否需要再次查询数据库代码类似于:
protected Object formBackingObject(HttpServletRequest request) throws Exception {
	if (isFormSubmission(request)) {
		//1. Return saved command object
	}
	else {
		//1. Query Database
		//2. Save result to Command object
	}
}


diyun 写道
正常情况下,如果没有必要,还是不用session的好。


可以具体说明使用Session为什么不好吗?


现在大概知道Command对象是如何被修改的了:

protected ModelAndView handleRequestInternal(HttpServletRequest request, HttpServletResponse response)
		throws Exception {

	// Form submission or new form to show?
	if (isFormSubmission(request)) {
		// Fetch form object from HTTP session, bind, validate, process submission.
		try {
			Object command = getCommand(request);
			ServletRequestDataBinder binder = bindAndValidate(request, command);
			BindException errors = new BindException(binder.getBindingResult());
			return processFormSubmission(request, response, command, errors);
		}
		catch (HttpSessionRequiredException ex) {
			// Cannot submit a session form if no form object is in the session.
			if (logger.isDebugEnabled()) {
				logger.debug("Invalid submit detected: " + ex.getMessage());
			}
			return handleInvalidSubmit(request, response);
		}
	}

	else {
		// New form to show: render form view.
		return showNewForm(request, response);
	}
}


如上述代码,在调用完"getCommand"后我们得到了Command对象(该Command对象可能是新建的也可能是从session中取出的),接下来会调用“bindAndValidate”,就是在这个bind and validate过程中将Form中提交的数据更新到Command对象中。
4 楼 diyun 2007-07-19  
不好意思,一下发了俩个,还不能删除。没有灌水的意思。
3 楼 diyun 2007-07-19  
sessionForm的默认值是false,在我的FormController中也没有设为true,所以目前我的FormController应该没有用sessionForm。
而且,正常情况下,如果没有必要,还是不用session的好。
2 楼 dynamo2 2007-07-18  
是的,覆写formBackingObject方法就是利用Session Form。使用Session Form还必须将Form设置为Session Form。如下代码末端用红色序号标注的地方表示了通过formBackingObject在“GET"修改页面时将Command Object加入到Session的过程。用绿色序号标注的代码表示了如何在”POST“请求后从Session中取出Command对象。<br/>
<br/>
此段代码拷贝自Spring的AbstractFormController类。<br/>
<br/>
<br/>
<div class='code_title'>java 代码</div>
<div class='dp-highlighter'>
<div class='bar'> </div>
<ol class='dp-j' start='1'>
    <li class='alt'><span><span class='keyword'>protected</span><span> ModelAndView handleRequestInternal(HttpServletRequest request, HttpServletResponse response)  </span></span></li>
    <li class=''><span>        <span class='keyword'>throws</span><span> Exception {  </span></span></li>
    <li class='alt'><span>    <span class='comment'>// Form submission or new form to show?</span><span>  </span></span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (isFormSubmission(request)) {  </span></span></li>
    <li class='alt'><span>        <span class='comment'>// Fetch form object from HTTP session, bind, validate, process submission.</span><span>  </span></span></li>
    <li class=''><span>        <span class='keyword'>try</span><span> {  </span></span></li>
    <li class='alt'><span>            <span style='background-color: rgb(255, 255, 153);'>Object command = getCommand(request);  <font size='5'><strong><font color='#339966'>1</font></strong></font><br/>
    </span></span></li>
    <li class=''><span>            ...  </span></li>
    <li class='alt'><span>        }  </span></li>
    <li class=''><span>        <span class='keyword'>catch</span><span> (HttpSessionRequiredException ex) {  </span></span></li>
    <li class='alt'><span>            ...  </span></li>
    <li class=''><span>        }  </span></li>
    <li class='alt'><span>    }  </span></li>
    <li class=''><span>  </span></li>
    <li class='alt'><span>    <span class='keyword'>else</span><span> {  </span></span></li>
    <li class=''><span>        <span class='comment'>// New form to show: render form view.</span><span>  </span></span></li>
    <li class='alt'><span>        <span class='keyword' style='background-color: rgb(0, 204, 255);'>return</span><span><span style='background-color: rgb(0, 204, 255);'> showNewForm(request, response);</span> <strong><font color='#ff0000'><font size='5'>1</font> </font></strong></span></span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>}  </span></li>
    <li class=''><span>  </span></li>
    <li class='alt'><span><span class='keyword'>protected</span><span> </span><span class='keyword'>final</span><span> ModelAndView showNewForm(HttpServletRequest request, HttpServletResponse response)  </span></span></li>
    <li class=''><span>        <span class='keyword'>throws</span><span> Exception {  </span></span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    logger.debug(<span class='string'>"Displaying new form"</span><span>);  </span></span></li>
    <li class='alt'><span>    <span class='keyword'>return</span><span> showForm(request, response, <span style='background-color: rgb(0, 204, 255);'>getErrorsForNewForm(request)</span>);<font size='5'> <strong><font color='#ff0000'>2</font></strong> </font></span></span></li>
    <li class=''><span>}  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span><span class='keyword'>protected</span><span> </span><span class='keyword'>final</span><span> BindException getErrorsForNewForm(HttpServletRequest request) </span><span class='keyword'>throws</span><span> Exception {  </span></span></li>
    <li class='alt'><span>        <span class='comment'>// Create form-backing object for new form.</span><span>  </span></span></li>
    <li class=''><span>    Object command = <span style='background-color: rgb(0, 204, 255);'>formBackingObject(request);</span> <font size='5'><strong><font color='#ff0000'>3</font></strong> </font></span></li>
    <li class='alt'><span>    <span class='keyword'>if</span><span> (command == </span><span class='keyword'>null</span><span>) {  </span></span></li>
    <li class=''><span>        <span class='keyword'>throw</span><span> </span><span class='keyword'>new</span><span> ServletException(</span><span class='string'>"Form object returned by formBackingObject() must not be null"</span><span>);  </span></span></li>
    <li class='alt'><span>    }  </span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (!checkCommand(command)) {  </span></span></li>
    <li class='alt'><span>        <span class='keyword'>throw</span><span> </span><span class='keyword'>new</span><span> ServletException(</span><span class='string'>"Form object returned by formBackingObject() must match commandClass"</span><span>);  </span></span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    <span class='comment'>// Bind without validation, to allow for prepopulating a form, and for</span><span>  </span></span></li>
    <li class='alt'><span>    <span class='comment'>// convenient error evaluation in views (on both first attempt and resubmit).</span><span>  </span></span></li>
    <li class=''><span>    ServletRequestDataBinder binder = createBinder(request, command);  </span></li>
    <li class='alt'><span>    BindException errors = <span class='keyword'>new</span><span> BindException(binder.getBindingResult());  </span></span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (isBindOnNewForm()) {  </span></span></li>
    <li class='alt'><span>        logger.debug(<span class='string'>"Binding to new form"</span><span>);  </span></span></li>
    <li class=''><span>        binder.bind(request);  </span></li>
    <li class='alt'><span>        onBindOnNewForm(request, command, errors);  </span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    <span class='comment'>// Return BindException object that resulted from binding.</span><span>  </span></span></li>
    <li class='alt'><span>    <span class='keyword'>return</span><span> errors;  </span></span></li>
    <li class=''><span>}  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span><span class='keyword'>protected</span><span> </span><span class='keyword'>final</span><span> Object getCommand(HttpServletRequest request) </span><span class='keyword'>throws</span><span> Exception {  </span></span></li>
    <li class='alt'><span>    <span class='comment'>// If not in session-form mode, create a new form-backing object.</span><span>  </span></span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (<span style='background-color: rgb(255, 255, 153);'>!isSessionForm()</span>) {  <font size='5'><strong><font color='#339966'>2</font></strong></font></span></span></li>
    <li class='alt'><span>        <span class='keyword'>return</span><span> formBackingObject(request);  </span></span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    <span class='comment'>// Session-form mode: retrieve form object from HTTP session attribute.</span><span>  </span></span></li>
    <li class='alt'><span>    HttpSession session = request.getSession(<span class='keyword'>false</span><span>);  </span></span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (session == </span><span class='keyword'>null</span><span>) {  </span></span></li>
    <li class='alt'><span>        <span class='keyword'>throw</span><span> </span><span class='keyword'>new</span><span> HttpSessionRequiredException(</span><span class='string'>"Must have session when trying to bind (in session-form mode)"</span><span>);  </span></span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    String formAttrName = getFormSessionAttributeName(request);  </span></li>
    <li class='alt'><span>    <span style='background-color: rgb(255, 255, 153);'>Object sessionFormObject = session.getAttribute(formAttrName); </span> <font size='5'><strong><font color='#339966'>3</font></strong></font></span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (sessionFormObject == </span><span class='keyword'>null</span><span>) {  </span></span></li>
    <li class='alt'><span>        <span class='keyword'>throw</span><span> </span><span class='keyword'>new</span><span> HttpSessionRequiredException(</span><span class='string'>"Form object not found in session (in session-form mode)"</span><span>);  </span></span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    <span class='comment'>// Remove form object from HTTP session: we might finish the form workflow</span><span>  </span></span></li>
    <li class='alt'><span>    <span class='comment'>// in this request. If it turns out that we need to show the form view again,</span><span>  </span></span></li>
    <li class=''><span>    <span class='comment'>// we'll re-bind the form object to the HTTP session.</span><span>  </span></span></li>
    <li class='alt'><span>    <span class='keyword'>if</span><span> (logger.isDebugEnabled()) {  </span></span></li>
    <li class=''><span>        logger.debug(<span class='string'>"Removing form session attribute ["</span><span> + formAttrName + </span><span class='string'>"]"</span><span>);  </span></span></li>
    <li class='alt'><span>    }  </span></li>
    <li class=''><span>    session.removeAttribute(formAttrName);  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    <span class='keyword'>return</span><span> currentFormObject(request, sessionFormObject);  </span></span></li>
    <li class='alt'><span>}  </span></li>
    <li class=''><span>  </span></li>
    <li class='alt'><span><span class='keyword'>protected</span><span> </span><span class='keyword'>final</span><span> ModelAndView showForm(  </span></span></li>
    <li class=''><span>    HttpServletRequest request, BindException errors, String viewName, Map controlModel)  </span></li>
    <li class='alt'><span>    <span class='keyword'>throws</span><span> Exception {  </span></span></li>
    <li class=''><span>  </span></li>
    <li class='alt'><span>    <span class='comment'>// In session form mode, re-expose form object as HTTP session attribute.</span><span>  </span></span></li>
    <li class=''><span>    <span class='comment'>// Re-binding is necessary for proper state handling in a cluster,</span><span>  </span></span></li>
    <li class='alt'><span>    <span class='comment'>// to notify other nodes of changes in the form object.</span><span>  </span></span></li>
    <li class=''><span>    <span class='keyword'>if</span><span> (<span style='background-color: rgb(0, 204, 255);'>isSessionForm()</span>) { <font size='5'><strong><font color='#ff0000'>5</font></strong></font> </span></span></li>
    <li class='alt'><span>        String formAttrName = getFormSessionAttributeName(request);  </span></li>
    <li class=''><span>        <span class='keyword'>if</span><span> (logger.isDebugEnabled()) {  </span></span></li>
    <li class='alt'><span>            logger.debug(<span class='string'>"Setting form session attribute ["</span><span> + formAttrName + </span><span class='string'>"] to: "</span><span> + errors.getTarget());  </span></span></li>
    <li class=''><span>        }  </span></li>
    <li class='alt'><span>        <span style='background-color: rgb(0, 204, 255);'>request.getSession().setAttribute(formAttrName, errors.getTarget());</span>  <font size='5'><strong><font color='#ff0000'>6</font></strong></font></span></li>
    <li class=''><span>    }  </span></li>
    <li class='alt'><span>  </span></li>
    <li class=''><span>    ...  </span></li>
    <li class='alt'><span>}  </span></li>
    <li class=''><span>  </span></li>
    <li class='alt'><span><span class='keyword'>protected</span><span> Object formBackingObject(HttpServletRequest request) </span><span class='keyword'>throws</span><span> Exception {  </span></span></li>
    <li class=''><span>    <span class='keyword'>return</span><span> <span style='background-color: rgb(0, 204, 255);'>createCommand();</span> <font size='5'><strong><font color='#ff0000'>4</font></strong></font> </span></span></li>
    <li class='alt'><span>}  </span></li>
</ol>
</div>
<br/>
如此看来覆写formBackingObject方法才是使用Session Form的正道。不过就是好像必须在form初始化时手动设置为Session Form。<br/>
现在还没有看出来Session中的Command对象是什么时候被修改的,这还是个问题。
1 楼 diyun 2007-07-18  
我的理解有点小小的不同。

第一:显示原有信息

override protected Object formBackingObject(HttpServletRequest request)

从url中取得id,调用service取得原有信息,返回。

第二:修改信息

override protected void doSubmitAction(Object command)

这一步好像和你的理解一样。

我不知道command的信息是否放在session里,但是至少在继承的程序里没有体现。

相关推荐

    【NLP 66、实践 ⑰ 基于Agent + Prompt优化进行文章优化】

    【NLP 66、实践 ⑰ 基于Agent + Prompt优化进行文章优化】

    梦限大mewtype成员 藤都子RVC模型

    考虑微网新能源经济消纳的共享储能优化配置附Matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。

    tokenizers-0.30.0.jar中文文档.zip

    # 【tokenizers-***.jar***文档.zip】 中包含: ***文档:【tokenizers-***-javadoc-API文档-中文(简体)版.zip】 jar包下载地址:【tokenizers-***.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【tokenizers-***.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【tokenizers-***.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【tokenizers-***-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: tokenizers-***.jar***文档.zip,java,tokenizers-***.jar,ai.djl.huggingface,tokenizers,***,ai.djl.engine.rust,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,djl,huggingface,中文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压 【tokenizers-***.jar***文档.zip】,再解压其中的 【tokenizers-***-javadoc-API文档-中文(简体)版.zip】,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件; # Maven依赖: ``` <dependency> <groupId>ai.djl.huggingface</groupId> <artifactId>tokenizers</artifactId> <version>***</version> </dependency> ``` # Gradle依赖: ``` Gradle: implementation group: 'ai.djl.huggingface', name: 'tokenizers', version: '***' Gradle (Short): implementation 'ai.djl.huggingface:tokenizers:***' Gradle (Kotlin): implementation("ai.djl.huggingface:tokenizers:***") ``` # 含有的 Java package(包): ``` ai.djl.engine.rust ai.djl.engine.rust.zoo ai.djl.huggingface.tokenizers ai.djl.huggingface.tokenizers.jni ai.djl.huggingface.translator ai.djl.huggingface.zoo ``` # 含有的 Java class(类): ``` ai.djl.engine.rust.RsEngine ai.djl.engine.rust.RsEngineProvider ai.djl.engine.rust.RsModel ai.djl.engine.rust.RsNDArray ai.djl.engine.rust.RsNDArrayEx ai.djl.engine.rust.RsNDArrayIndexer ai.djl.engine.rust.RsNDManager ai.djl.engine.rust.RsSymbolBlock ai.djl.engine.rust.RustLibrary ai.djl.engine.rust.zoo.RsModelZoo ai.djl.engine.rust.zoo.RsZooProvider ai.djl.huggingface.tokenizers.Encoding ai.djl.huggingface.tokenizers.HuggingFaceTokenizer ai.djl.huggingface.tokenizers.HuggingFaceTokenizer.Builder ai.djl.hu

    人形机器人是当今世界科技领域最具潜力和前景的产业之一 随着科技的不断进步和人工智能技术的快速发展,人形机器人作为未来产业的新赛道和经济增长的新引擎,将深刻变革人类生产生活方式,重塑全球产业发展格局

    人形机器人产业的发展需要人工智能、高端制造、新材料等先进技术的协同创新和突破。

    【状态估计】用于非标量系统估计的最优卡尔曼滤波附Matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。

    开关电源的尖峰干扰及其抑制.zip

    开关电源的尖峰干扰及其抑制.zip

    房地产培训 -新进业务员压马路市调培训.ppt

    房地产培训 -新进业务员压马路市调培训.ppt

    MATLAB实现计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度

    内容概要:本文探讨了基于MATLAB平台的虚拟电厂优化调度方法,特别关注电转气(P2G)协同、碳捕集技术和垃圾焚烧的应用。文中介绍了虚拟电厂的概念及其重要性,详细解释了碳捕集、需求响应和电转气协同调度的关键技术,并展示了如何使用MATLAB和CPLEX求解器进行优化调度的具体步骤。通过定义决策变量、构建目标函数和设定约束条件,最终实现了多目标优化,即经济性最优和碳排放最低。此外,还讨论了一些常见的代码实现技巧和潜在的问题解决方案。 适合人群:从事能源管理和优化调度研究的专业人士,尤其是那些熟悉MATLAB编程和优化算法的人士。 使用场景及目标:适用于希望深入了解虚拟电厂运作机制和技术实现的研究人员和工程师。主要目标是通过优化调度提高能源利用效率,减少碳排放,降低成本。 其他说明:文章提供了详细的代码片段和理论分析,有助于读者更好地理解和复现实验结果。同时,强调了在实际应用中需要注意的一些细节问题,如约束条件的平衡、求解器配置等。

    在网格化数据集上轻松执行 2D 高通、低通、带通或带阻滤波器研究附Matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。

    spring-ai-pinecone-store-1.0.0-M7.jar中文-英文对照文档.zip

    # 【spring-ai-pinecone-store-1.0.0-M7.jar中文-英文对照文档.zip】 中包含: 中文-英文对照文档:【spring-ai-pinecone-store-1.0.0-M7-javadoc-API文档-中文(简体)-英语-对照版.zip】 jar包下载地址:【spring-ai-pinecone-store-1.0.0-M7.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【spring-ai-pinecone-store-1.0.0-M7.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【spring-ai-pinecone-store-1.0.0-M7.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【spring-ai-pinecone-store-1.0.0-M7-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: spring-ai-pinecone-store-1.0.0-M7.jar中文-英文对照文档.zip,java,spring-ai-pinecone-store-1.0.0-M7.jar,org.springframework.ai,spring-ai-pinecone-store,1.0.0-M7,org.springframework.ai.vectorstore.pinecone,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,springframework,spring,ai,pinecone,store,中文-英文对照API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压 【spring-ai-pinecone

    基于MATLAB混合整数规划的微网电池储能容量优化配置

    内容概要:本文详细介绍了如何使用MATLAB及其优化工具箱,通过混合整数规划(MILP)方法对微网电池储能系统的容量进行优化配置。主要内容包括定义目标函数(如最小化运行成本),设置约束条件(如充放电功率限制、能量平衡约束),并引入决策变量(如电池容量、充放电功率和状态)。文中提供了具体的MATLAB代码示例,演示了如何将实际问题转化为数学模型并求解。此外,还讨论了一些实用技巧,如避免充放电互斥冲突、考虑电池寿命损耗等。 适用人群:从事微电网设计与运维的技术人员,尤其是那些希望通过优化算法提高系统性能和经济效益的专业人士。 使用场景及目标:适用于需要确定最佳电池储能容量的微电网项目,旨在降低总体运行成本,提高系统的稳定性和可靠性。具体应用场景包括工业园区、商业建筑或其他分布式能源系统。 其他说明:文章强调了模型的实际应用价值,并指出通过精确控制充放电策略可以显著减少不必要的容量闲置,从而节省大量资金。同时提醒读者注意模型的时间粒度选择、电池退化成本等因素的影响。

    langchain4j-ollama-1.0.0-beta1.jar中文文档.zip

    # 压缩文件中包含: 中文文档 jar包下载地址 Maven依赖 Gradle依赖 源代码下载地址 # 本文件关键字: jar中文文档.zip,java,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,中文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压最外层zip,再解压其中的zip包,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件;

    光伏离网并网逆变器设计:基于TMS320F28335的数字控制与SPWM技术详解

    内容概要:本文详细介绍了基于TMS320F28335的光伏离网并网逆变器设计方案,涵盖了从硬件架构到软件控制的各个方面。首先,文章阐述了TMS320F28335作为高性能DSP的优势及其初始化配置方法。其次,探讨了逆变器的数字控制策略,如双闭环控制(电压外环和电流内环)的具体实现方式。然后,深入讲解了SPWM(正弦脉宽调制)技术,包括SPWM波的生成方法和相关代码示例。此外,还讨论了硬件保护逻辑、过流检测、死区时间配置等实际应用中的注意事项。最后,提供了调试经验和学习资源建议。 适合人群:从事光伏逆变器设计、嵌入式系统开发的技术人员,尤其是有一定DSP编程基础的研发人员。 使用场景及目标:适用于需要深入了解光伏逆变器设计原理和技术实现的研究人员和工程师。主要目标是掌握基于TMS320F28335的逆变器控制系统设计,包括数字控制策略和SPWM技术的应用。 其他说明:文中提供的代码示例和实践经验有助于读者更好地理解和应用于实际项目中。建议读者结合TI官方提供的学习资料进行进一步学习和实践。

    【医疗影像分析】深度学习技术在医疗影像分析中的应用优势及未来发展方向:自动特征学习、高精度高效处理、多模态数据融合、个性化治疗与预测、实时远程支持

    内容概要:深度学习在医疗影像分析中展现出显著的优势,主要体现在自动特征学习、高准确性和效率、多模态数据融合与综合分析、个性化治疗与预测、减少主观性、处理复杂和高维数据、实时分析与远程医疗支持、数据挖掘与科研突破以及可扩展性与持续优化九个方面。通过卷积神经网络(CNN)、U-Net等模型,深度学习能够自动从影像中提取多层次特征,无需手动干预,在分类、分割任务中表现出色,处理速度远超人工。此外,它还能够整合多源数据,提供全面的诊断依据,实现个性化治疗建议,减少误诊和漏诊,支持实时分析和远程医疗,挖掘病理模式并加速研究,同时具有可扩展性和持续优化的能力。; 适合人群:医疗行业从业者、科研人员、计算机视觉和深度学习领域的研究人员。; 使用场景及目标:①用于医疗影像的自动特征提取和分类,如乳腺癌筛查、皮肤癌诊断等;②整合多模态数据,如CT、MRI等,提高诊断准确性;③提供个性化治疗建议,优化治疗方案;④支持实时分析和远程医疗,尤其适用于偏远地区的急诊场景;⑤挖掘病理模式,加速疾病机制的研究。; 其他说明:深度学习正逐渐成为医疗影像分析的核心诊断伙伴,未来发展方向包括增强可解释性、保护数据隐私和轻量化部署,旨在进一步提升医疗效率和患者护理质量。

    深度学习机器学习子领域关键技术解析:神经网络基础、常见架构及应用场景综述

    内容概要:深度学习是机器学习的一个子领域,通过构建多层次的“深度神经网络”来模拟人脑结构,从而学习和提取数据的复杂特征。文章介绍了深度学习的核心概念,包括神经元、多层感知机、深度神经网络(DNN)、卷积神经网络(CNN)、循环神经网络(RNN)和Transformer等常见网络结构。同时,详细讲解了激活函数、损失函数与优化器的作用。此外,还探讨了深度学习的关键突破,如大数据与算力的支持、正则化技术和迁移学习的应用。文中列举了深度学习在计算机视觉、自然语言处理、语音与音频以及强化学习等领域的应用场景,并指出了其面临的挑战,如数据依赖、计算成本和可解释性问题。最后提供了使用PyTorch和TensorFlow/Keras框架的经典代码示例,涵盖图像分类、文本生成和迁移学习等内容。; 适合人群:对机器学习有一定了解,希望深入学习深度学习理论和技术的研究人员、工程师及学生。; 使用场景及目标:①理解深度学习的基本原理和核心概念;②掌握常见深度学习框架的使用方法,如PyTorch和TensorFlow;③能够根据具体应用场景选择合适的网络结构和算法进行实践。; 其他说明:本文不仅提供了理论知识,还附带了详细的代码示例,便于读者动手实践。建议读者结合理论与实践,逐步深入理解深度学习的各个方面。

    深度学习答辩PPT案例展示

    适用于理工专业的毕业生,毕业答辩时可供参考,叙述详细准确,可以作为自己答辩PPT的参考

    tokenizers-0.22.1.jar中文-英文对照文档.zip

    # 【tokenizers-***.jar***文档.zip】 中包含: ***文档:【tokenizers-***-javadoc-API文档-中文(简体)版.zip】 jar包下载地址:【tokenizers-***.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【tokenizers-***.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【tokenizers-***.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【tokenizers-***-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: tokenizers-***.jar***文档.zip,java,tokenizers-***.jar,ai.djl.huggingface,tokenizers,***,ai.djl.engine.rust,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,djl,huggingface,中文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压 【tokenizers-***.jar***文档.zip】,再解压其中的 【tokenizers-***-javadoc-API文档-中文(简体)版.zip】,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件; # Maven依赖: ``` <dependency> <groupId>ai.djl.huggingface</groupId> <artifactId>tokenizers</artifactId> <version>***</version> </dependency> ``` # Gradle依赖: ``` Gradle: implementation group: 'ai.djl.huggingface', name: 'tokenizers', version: '***' Gradle (Short): implementation 'ai.djl.huggingface:tokenizers:***' Gradle (Kotlin): implementation("ai.djl.huggingface:tokenizers:***") ``` # 含有的 Java package(包): ``` ai.djl.engine.rust ai.djl.engine.rust.zoo ai.djl.huggingface.tokenizers ai.djl.huggingface.tokenizers.jni ai.djl.huggingface.translator ai.djl.huggingface.zoo ``` # 含有的 Java class(类): ``` ai.djl.engine.rust.RsEngine ai.djl.engine.rust.RsEngineProvider ai.djl.engine.rust.RsModel ai.djl.engine.rust.RsNDArray ai.djl.engine.rust.RsNDArrayEx ai.djl.engine.rust.RsNDArrayIndexer ai.djl.engine.rust.RsNDManager ai.djl.engine.rust.RsSymbolBlock ai.djl.engine.rust.RustLibrary ai.djl.engine.rust.zoo.RsModelZoo ai.djl.engine.rust.zoo.RsZooProvider ai.djl.huggingface.tokenizers.Encoding ai.djl.huggingface.tokenizers.HuggingFaceTokenizer ai.djl.huggingface.tokenizers.HuggingFaceTokenizer.Builder ai.djl.hu

    能源领域:基于MATLAB的阶梯式碳交易与供需灵活双响应综合能源系统优化调度

    内容概要:本文详细介绍了考虑阶梯式碳交易与供需灵活双响应的综合能源系统优化调度方法。在供给侧,引入了有机朗肯循环(ORC)实现热电联产机组的灵活响应;在需求侧,提出电、热、气负荷之间的可替代性,以提高能源利用效率。构建了以最小化碳排放成本、购能成本、弃风成本和需求响应成本为目标的优化调度模型,并采用MATLAB和CPLEX进行了模型构建和求解。文中提供了具体的代码示例,展示了如何处理热电耦合、负荷替代和阶梯式碳交易等问题。 适合人群:从事能源系统优化、电力系统调度、碳交易等相关领域的研究人员和技术人员。 使用场景及目标:适用于需要优化能源系统调度、降低成本并减少碳排放的实际应用场景。目标是帮助读者理解和掌握如何通过先进的技术和算法实现更加灵活和高效的能源调度。 其他说明:文章提供了完整的代码实现和服务支持,包括12种典型场景的数据集和预设模型,方便读者快速上手实践。

    spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar中文文档.zip

    # 【spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar中文文档.zip】 中包含: 中文文档:【spring-ai-autoconfigure-mcp-client-1.0.0-M7-javadoc-API文档-中文(简体)版.zip】 jar包下载地址:【spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【spring-ai-autoconfigure-mcp-client-1.0.0-M7-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar中文文档.zip,java,spring-ai-autoconfigure-mcp-client-1.0.0-M7.jar,org.springframework.ai,spring-ai-autoconfigure-mcp-client,1.0.0-M7,org.springframework.ai.mcp.client.autoconfigure,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,springframework,spring,ai,autoc

Global site tag (gtag.js) - Google Analytics