Использование как Thymeleaf, так и JSP

Я использовал JSP + JSTL, но мне скучно c: if, c: choose,...

Итак, я хочу, чтобы мои страницы JSP отображались как с JSP, так и с Thymeleaf (я как можно скорее удалю все JSTL). Я использую структуру Spring MVC:

<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <property name="prefix" value="/WEB-INF/pages/" />
    <property name="suffix" value=".jsp" />
    <property name="order" value="1" />
</bean>
<!-- Thymeleaf -->
<bean id="templateResolver" class="org.thymeleaf.templateresolver.ServletContextTemplateResolver">
    <property name="prefix" value="/WEB-INF/pages/" />
    <property name="suffix" value=".html" />
    <property name="templateMode" value="HTML5" />
</bean>
<bean id="templateEngine" class="org.thymeleaf.spring4.SpringTemplateEngine">
    <property name="templateResolver" ref="templateResolver" />
</bean>
<bean class="org.thymeleaf.spring4.view.ThymeleafViewResolver">
    <property name="templateEngine" ref="templateEngine" />
    <property name="order" value="2" />
</bean> 

В моем контроллере я просто возвращаю jsp без расширения.

return "folder/page";

Могут ли мои страницы JSP быть первыми с помощью распознавателя JSP, а затем с помощью распознавателя Thymeleaf? Если да, то как?

Похоже, что очень сложно связать JSP и Thymeleaf. Таким образом, я хочу использовать Внутренний резольвер для JSP файлов и Thymeleaf template resolver для файлов HTML. Как я могу это сделать?

Ответ 1

Вот ответ, основанный на ответе @Igd

<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <property name="prefix" value="/WEB-INF/pages/" />
    <property name="viewNames" value="*.jsp" />
</bean>
<!-- Thymeleaf -->
<bean id="templateResolver" class="org.thymeleaf.templateresolver.ServletContextTemplateResolver">
    <property name="prefix" value="/WEB-INF/pages/" />
    <property name="templateMode" value="HTML5" />
</bean>
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <property name="viewNames" value="redirect*" />
</bean>
<bean id="templateEngine" class="org.thymeleaf.spring4.SpringTemplateEngine">
    <property name="templateResolver" ref="templateResolver" />
</bean>
<bean class="org.thymeleaf.spring4.view.ThymeleafViewResolver">
    <property name="templateEngine" ref="templateEngine" />
    <property name="viewNames" value="*.html" />
</bean> 

И я использую это для отображения:

@RequestMapping("/view1")
public String thymeleafView(){
    return "mythymeleafview.html";
}

@RequestMapping("/view2")
public String jspView(){
    return "myjspview.jsp";
}

Ответ 2

В соответствии с этим сообщением на форуме Thymeleaf у вас есть два решения.

Первое решение:

Удалите свойство суффикса в объявлении bean (<property name="suffix" value=".html" /> и <property name="suffix" value=".jsp" />) и передайте суффикс в возвращаемом значении ваших контроллеров, например.

@RequestMapping("/view1")
public String thymeleafView(){
    return "mythymeleafview.html";
}

@RequestMapping("/view2")
public String jspView(){
    return "myjspview.html";
}

Второе решение:

Добавьте свойство viewNames в преобразователи. Значение - это имя папки, которая содержит представления в зависимости от их расширения. Таким образом, у вас будет одна папка для JSP файлов, а другая - для файлов HTML (thymeleaf), например.

Конфигурация

<bean id="templateResolver" class="org.thymeleaf.templateresolver.ServletContextTemplateResolver">
     <property name="prefix" value="/WEB-INF/views/" />
     <property name="suffix" value=".html" />
     <property name="viewNames" value="thymeleaf/*" />
     <property name="templateMode" value="HTML5" />
</bean>

<bean id="jspViewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
     <property name="viewClass" value="org.springframework.web.servlet.view.JstlView" />
     <property name="prefix" value="/WEB-INF/views/" />
     <property name="viewNames" value="jsp/*" />
     <property name="suffix" value=".jsp" />
</bean>

контроллер

@RequestMapping("/view1") 
public String thymeleafView() { 
     return "thymeleaf/mythymeleafview"; 
} 

@RequestMapping("/view2") 
public String jspView() { 
     return "jsp/myjspview"; 
}

Папка проекта

WEB-INF/views/jsp/myjspview.jsp
WEB-INF/views/thymeleaf/mythymeleafview.jsp

Оба решения работают, но имеют некоторые недостатки. Вы должны указать так или иначе, хотите ли вы разрешить работу с JSP или Thymeleaf.

"Идеальное" решение для цепочек JSP и Thymeleaf, которое состояло бы в попытке разрешить представление с JSP, когда оно не может быть разрешено с помощью Thymeleaf или наоборот, невозможно, и Дэниел Фернандес (команда Тимелеафа) объяснил, почему в этот же пост:

Thymeleaf позволяет вам создать какую бы то ни было реализацию ITemplateResolver, в том числе некоторые, которые могут не позволить определить, существует ли шаблон или нет, прежде чем читать его. [...] Таким образом, Тимелеаф не может быть уверен, будет ли шаблон разрешимым или нет, прежде чем пытаться обработать шаблон. И вот почему ThymeleafViewResolver должен прибегать к свойству "viewNames".

Ответ 3

В качестве альтернативы два сервлета работают нормально. Ключ должен поддерживать минимальную конфигурацию сервлета и включать appConfig.xml для базы данных и других сервисов (это позволяет избежать большого дублирования конфигурации).

web.xml:

<web-app id="WebApp_ID" version="2.4"
   xmlns="http://java.sun.com/xml/ns/j2ee" 
   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
   xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee 
   http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">

   <display-name>Spring MVC Application</display-name>

   <servlet>
      <servlet-name>AssessmentAdmin</servlet-name>
      <servlet-class>
         org.springframework.web.servlet.DispatcherServlet
      </servlet-class>
      <load-on-startup>1</load-on-startup>
   </servlet>
 <servlet-mapping>
      <servlet-name>AssessmentAdmin</servlet-name>
      <url-pattern>/xz/*</url-pattern>
   </servlet-mapping>
   
   
<servlet>
      <servlet-name>AssessmentAdminTL</servlet-name>
      <servlet-class>
         org.springframework.web.servlet.DispatcherServlet
      </servlet-class>
      <load-on-startup>1</load-on-startup>
   </servlet>   
   <servlet-mapping>
      <servlet-name>AssessmentAdminTL</servlet-name>
      <url-pattern>/xztl/*</url-pattern>
   </servlet-mapping>
   ........

Ответ 4

в соответствии с ответом @Athanor, у нас может быть другой выбор.

мы используем свойство "viewNames" для управления тем, какой преобразователь выбирает шаблон

<!-- jsp -->
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <property name="prefix" value="/" />
    <property name="suffix" value=".jsp" />
    <property name="order" value="1" />
    <property name="viewNames" value="*admin/*,*packer/*,*courier/*,/" />
</bean>
<!-- thymeleaf -->
<bean id="templateResolver" class="org.thymeleaf.templateresolver.ServletContextTemplateResolver">
    <property name="prefix" value="/" />
    <property name="suffix" value=".html" />
    <property name="templateMode" value="HTML5" />
    <property name="cacheable" value="false"/>
</bean>
<bean id="templateEngine" class="org.thymeleaf.spring3.SpringTemplateEngine">
    <property name="templateResolver" ref="templateResolver" />
</bean>
<bean class="org.thymeleaf.spring3.view.ThymeleafViewResolver">
    <property name="characterEncoding" value="UTF-8"/>
    <property name="templateEngine" ref="templateEngine" />
    <property name="viewNames" value="*thymeleaf/*" />
    <property name="order" value="2" />
</bean>

и контроллер

@RequestMapping(value="/test")
public ModelAndView dboxPrint(Model model){
    ModelAndView modelAndView = new ModelAndView("thymeleaf/dbox_print");

    return modelAndView;
}