【Eclipse】動的Webプロジェクトでのプロパティファイル読込

動的Webプロジェクトのプロパティファイル読込で
以下のように処理を書いてみました。

prop.load(new FileInputStream("hoge.properties"));

プロパティファイルをプロジェクト直下、クラスパス直下、パッケージ直下においてみましたが、
いずれも実行時にエラーになりました。
解決策として以下の通りにしました。

プロパティファイルは「WEB-INF/src」直下に配置し、
「WEB-INF/class」に自動コピーされるので、
以下のように処理を変更しました。

prop.load(this.getClass().getResourceAsStream("/hoge.properties"));

【Eclipse】classpathに書き込めませんでしたというエラーの原因

新規でプロジェクトを作成せず、
インポートしたプロジェクトに対して、外部JARファイルを追加した場合、
以下のようなエラーが出ることがあります。

ビルド・パスの設定(.classpathに書き込めませんでした)

原因は「.classpath」ファイルが隠しファイルになっているから。
「.classpath」ファイルを右クリックし、隠しファイルのチェックを外せば
エラーが解消します。

【Eclipse】動的Webプロジェクトのコピー

Eclipseで動的Webプロジェクトをコピーしたい場合、以下を行う。

  • プロジェクト名のリファクタリング
    プロジェクトで右クリックし、リファクタリングを選択する。
    変更後のプロジェクトを入力する。
  • プロジェクトのコンテキスト名変更
    プロジェクトで右クリックし、「プロパティ」を選択する。
    Webプロジェクトの設定を選択し、コンテキスト・ルートを
    コピー後のプロジェクトに合わせて変更する。
  • ローカルのTomcat設定変更
    以下ファイルの「Context docBase」のプロジェクト名を変更する。

    /workspace/Servers/ローカル・ホスト の Tomcat v7.0 サーバー-config/server.xml
    

【Java】jarファイル付のJavaコンパイル

基本的なことをメモ。

  • 前提条件
    /home/testにjarファイル、Javaファイルがある。
    Javaファイルは処理内でjarファイル内のメソッドを利用する。
  • コンパイルと実行

    cd /home/test
    javac -cp .:./jarファイル javaファイル
    java -cp .:./jarファイル クラス名
    
  • クラスパスの区切り文字
    Linuxはコロンになる。
    Windowsはセミコロンになる。
  • 実行時のルートパス
    実行時に-cp指定がないとJAVA_HOMEがルートになる。
    .を指定することで現在いるディレクトリがルートになる。

【SpringMVC】GoogleChartsの利用

画面でボタン押下時にモーダルダイアログを表示し、
GoogleChartsを利用し、グラフを表示します。

  • 画面の処理
    ボタンにIDを追加する。

    <button id="bsButton" type="button" class="btn btn-danger">グラフ表示</button>
    

    モーダルダイアログの表示領域を設定する。

    <!-- Chart Create Modal -->
    <div th:fragment="chart-create-modal">
      <div class="modal fade bs-example-modal-lg" id="myModal" tabindex="-1">
        <div class="modal-dialog modal-lg">
          <div class="modal-content">
            <div class="modal-header">
              <button type="button" class="close" data-dismiss="modal">&times;</button>
              <h3 class="modal-title" id="myModalLabel">タイトル</h3>
            </div>
            <div class="modal-body">
              <div id="chart_div" style="width: 400px; height: 400px;"></div>
            </div>
          </div>
        </div>
      </div>
    </div>
    

    JavaScriptでサーバー側からデータ取得処理を呼び出し
    GoogleのAPIにデータを渡します。

    <script type="text/javascript" src="http://www.google.com/jsapi"></script>
    <script type="text/javascript">
    google.load("visualization", "1", {
      packages : [ "corechart" ]
    });
    <!--
    $(document).ready(function() {
      $("#bsButton").click(showChart);
    });
    
    function showChart() {
      $.post(
        '/sample/showChart', // URL
        {},                  // Userデータ
        function(result) {
          $("#myModalLabel").html("テストグラフ");
          $('#myModal').modal('show');
          var data = google.visualization.arrayToDataTable(result);
          var options = {
            title : 'テストグラフタイトル',
            vAxis : {title : "縦座標"},
            hAxis : {title : "横座標"},
            seriesType : "bars",
            series : {5 : {type : "line"}}
          };
          var chart = new google.visualization.ComboChart(document.getElementById('chart_div'));
          chart.draw(data, options);
        },                   // 成功時のコールバック
        "json"               // JSON形式
      );
    }
    
  • サーバの処理
    サーバー側からデータを取得しJSON形式で返します。

    @RequestMapping(value = "/showChart", method = RequestMethod.POST)
    @ResponseBody
    public Object[][] showChart(WebRequest request) {
        int rows = 3;
        int cols = 6;
        Object[][] result = new Object[rows][cols];
        int row=0, col=0;
        // result配列にヘッダとデータを設定
        return result;
    }
    

【SpringMVC】バリデーションチェック

SpringMVCでバリデーションチェックを実施します。
例として、作業時間入力用のクラスがあるとして、それを複数所持するListがあるとし、
JSPでそのListをループし、作業時間入力レコードを複数表示およびチェックします。

  • バリデーションチェックのjarファイル追加
    「【SpringMVC設定】ファイルその4」を参考にpom.xmlにjarファイルを追加する。
  • アノテーション追加
    作業時間入力用のクラスは全て必須項目とするため、「@NotNull」を追加する。

    public class DayTimeWork implements Serializable {
        private static final long serialVersionUID = 1L;
    
        @NotNull
        private String ankenNo;
        @NotNull
        private String startTime;
        @NotNull
        private String endTime;
    
        アクセサメソッドは省略。
    }
    

    作業時間入力用のクラスを保持するListには「@Valid」を追加する。
    「@Valid」はネストしたクラスも検証対象とする場合、追加します。

    public class DayTime implements Serializable {
        @Valid
        private List<DayTimeWork> dayTimeWork;
    
        アクセサメソッドは省略。
    }
    
  • コントローラーへの設定追加
    Javaコントローラーにバリデーションチェックを実施するための設定を追加する。
    「@Validated」の次の項目にBindingResultを記載しないと
    実行時にValidationExceptionが発生する。

    
    @Controller
    public class InputController {
    
        @InitBinder
        public void initBinder(WebDataBinder binder) {
            // bind empty strings as null
            binder.registerCustomEditor(String.class, new StringTrimmerEditor(true));
        }
    
        @RequestMapping(value = "/updateDayTime", method = RequestMethod.POST)
        public String updateDayTime(
                @ModelAttribute(value = "dayTime") DayTime dayTime, 
                final @Validated DayTime inputDayTime, 
                BindingResult result, 
                Model model) {
            // 処理省略
        }
    }
    
  • JSPへの設定追加
    JSPでバリデーションチェック結果を取得する。
    Thymeleafのタグを利用し、全項目内でエラーが1つでもあれば、
    変数errとしてエラーを取得し、表示しています。

    <form class="form-horizontal" action="#" th:action="@{/updateDayTime}" th:object="${dayTime}" method="post">
    <fieldset>
      <ul th:if="${#fields.hasErrors('*')}">
        <li th:each="err : ${#fields.errors('*')}" th:text="${err}">Input is incorrect</li>
      </ul>
    </fieldset>
    </form>
    

    Thymeleafのタグを利用し、エラー項目のテキストボックスの枠を赤くしたり、
    入力がない場合、placeholderで「案件番号」という文字を初期表示することも可能。

    <form class="form-horizontal" action="#" th:action="@{/updateDayTime}" th:object="${dayTime}" method="post">
      <table class="table">
        <!-- 処理省略 -->
        <tr th:each="vc, row : *{dayTimeWork}" >
          <td th:class="${#fields.hasErrors('dayTimeWork[__${row.index}__].ankenNo')}? has-error"  align="center">
          <input type="text" th:field="*{dayTimeWork[__${row.index}__].ankenNo}" placeholder="案件番号" class="form-control" />
          </td>
        </tr>
      </table>
    </form>
    

【SpringMVC】でThymeleafを利用

JSPではサーブレット側の処理を記載していないと
値が渡せないので、画面側のデザインが値ありで確認しにくかったりしていましたが、
Thymeleafでは値がある場合、ない場合の動作を切り分けられる点、
また便利なタグが多い点から利用してみます。

  • src/main/webapp/WEB-INF/spring/appServlet/servlet-context.xml

    • コントローラーの処理終了後に呼び出される画面用ファイルの指定
      この部分を変更します。

      【変更前】

      <beans:bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <beans:property name="prefix" value="/WEB-INF/views/" />
        <beans:property name="suffix" value=".jsp" />
      </beans:bean>
      

      【変更後】

      <beans:bean id="templateResolver" class="org.thymeleaf.templateresolver.ServletContextTemplateResolver">
        <beans:property name="prefix" value="/WEB-INF/templates/" />
        <beans:property name="suffix" value=".html" />
        <beans:property name="templateMode" value="HTML5" />
      </beans:bean>
      
      <beans:bean id="templateEngine" class="org.thymeleaf.spring3.SpringTemplateEngine">
        <beans:property name="templateResolver" ref="templateResolver" />
      </beans:bean>
      
      <beans:bean class="org.thymeleaf.spring3.view.ThymeleafViewResolver">
        <beans:property name="characterEncoding" value="UTF-8" />
        <beans:property name="templateEngine" ref="templateEngine" />
      </beans:bean>
      
  • pom.xml
    「dependencies」タグの中に下記内容を追加します。

    • Thymeleafライブラリー

      <!-- Thymeleaf -->
      <dependency>
        <groupId>org.thymeleaf</groupId>
        <artifactId>thymeleaf-spring3</artifactId>
        <version>2.1.2.RELEASE</version>
      </dependency>
      

【SpringMVC】設定ファイルその4

  • pom.xml
    拡張設定の内容を記載します。
    前回の設定と同様に「dependencies」タグの中に下記内容を記載していきます。

    • バリデーションチェックライブラリー

      <!-- Validator -->
      <dependency>
        <groupId>javax.validation</groupId>
        <artifactId>validation-api</artifactId>
        <version>1.0.0.GA</version>
      </dependency>
      <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-validator</artifactId>
        <version>4.3.1.Final</version>
      </dependency>
      
    • JavaオブジェクトをJSON形式に変換しGoogleAPIに渡すライブラリー

      <!-- Jacson -->
      <dependency>
        <groupId>org.codehaus.jackson</groupId>
        <artifactId>jackson-core-asl</artifactId>
        <version>1.6.4</version>
      </dependency>
      <dependency>
        <groupId>org.codehaus.jackson</groupId>
        <artifactId>jackson-mapper-asl</artifactId>
        <version>1.6.4</version>
      </dependency>
      <!-- GoogleAPI -->
      <dependency>
        <groupId>com.googlecode.json-simple</groupId>
        <artifactId>json-simple</artifactId>
        <version>1.1</version>
      </dependency>
      
    • JUnitライブラリー

      <!-- JUnit -->
      <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.7</version>
        <scope>test</scope>
      </dependency>
      

【SpringMVC】設定ファイルその3

  • pom.xml
    mavenのプロジェクト定義ファイルで
    ビルドする際に依存する外部jarファイルの設定やJDKのバージョンを定義します。

    • 基本設定

      <?xml version="1.0" encoding="UTF-8"?>
      <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
        <!-- モデルバージョン -->
        <modelVersion>4.0.0</modelVersion>
        <!-- パッケージ名 -->
        <groupId>jp.co</groupId>
        <!-- コンテキスト名 -->
        <artifactId>sample</artifactId>
        <!-- プロジェクト名 -->
        <name>Sample</name>
        <!-- 展開用拡張子 -->
        <packaging>war</packaging>
        <!-- バージョン -->
        <version>1.0.0-BUILD-SNAPSHOT</version>
        <!-- バージョン情報外だし -->
        <properties>
          <java-version>1.6</java-version>
          <org.springframework-version>3.1.1.RELEASE</org.springframework-version>
          <org.aspectj-version>1.6.10</org.aspectj-version>
          <org.slf4j-version>1.6.6</org.slf4j-version>
        </properties>
        <!-- 各種設定 -->
        <dependencies>後述の各種設定をタグ内に記載</dependencies>
        <!-- ビルド -->
        <build>省略</build>
      </project>
      

    「dependencies」タグの中に下記内容を記載していきます。

    • Web開発に必要なライブラリー

      <!-- Servlet -->
      <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>servlet-api</artifactId>
        <version>2.5</version>
        <scope>provided</scope>
      </dependency>
      <dependency>
        <groupId>javax.servlet.jsp</groupId>
        <artifactId>jsp-api</artifactId>
        <version>2.1</version>
        <scope>provided</scope>
      </dependency>
      
      <!-- Tag Library -->
      <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>jstl</artifactId>
        <version>1.2</version>
      </dependency>
      
      <!-- Logging -->
      <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-api</artifactId>
        <version>${org.slf4j-version}</version>
      </dependency>
      <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>jcl-over-slf4j</artifactId>
        <version>${org.slf4j-version}</version>
        <scope>runtime</scope>
      </dependency>
      <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-log4j12</artifactId>
        <version>${org.slf4j-version}</version>
        <scope>runtime</scope>
      </dependency>
      <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>1.2.15</version>
        <exclusions>
          <exclusion>
            <groupId>javax.mail</groupId>
            <artifactId>mail</artifactId>
          </exclusion>
          <exclusion>
            <groupId>javax.jms</groupId>
            <artifactId>jms</artifactId>
          </exclusion>
          <exclusion>
            <groupId>com.sun.jdmk</groupId>
            <artifactId>jmxtools</artifactId>
          </exclusion>
          <exclusion>
            <groupId>com.sun.jmx</groupId>
            <artifactId>jmxri</artifactId>
          </exclusion>
        </exclusions>
        <scope>runtime</scope>
      </dependency>
      
    • SpringFrameWorkのライブラリー

      <!-- Spring -->
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>${org.springframework-version}</version>
        <exclusions>
          <!-- Exclude Commons Logging in favor of SLF4j -->
          <exclusion>
            <groupId>commons-logging</groupId>
            <artifactId>commons-logging</artifactId>
           </exclusion>
        </exclusions>
      </dependency>
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>${org.springframework-version}</version>
      </dependency>
      <!-- AOP関連(AspectJ) -->
      <dependency>
        <groupId>org.aspectj</groupId>
        <artifactId>aspectjrt</artifactId>
        <version>${org.aspectj-version}</version>
      </dependency>
      <!-- AOP関連(cglib) -->
      <dependency>
        <groupId>cglib</groupId>
        <artifactId>cglib</artifactId>
        <version>2.2.2</version>
      </dependency>
      

【SpringMVC】設定ファイルその2

  • src/main/webapp/WEB-INF/spring/root-context.xml
    アプリケーションの共通設定として、
    ファンクション層(F層:Service)、データ層(D層:Dao)の定義を行う。

    • 「@Autowire」「@Component」「@Service」のアノテーションを有効にする設定

      <context:annotation-config />
      
    • アノテーションでScan対象とするパッケージを指定

      <context:component-scan base-package="jp.co.sample.service" />
      <context:component-scan base-package="jp.co.sample.dao" />
      

      上記パッケージにJavaファイルを配置すると、
      JavaファイルにSpringの管理を示す「S」マークがつきます。

      このファイルは共通設定なので、パッケージにcontrollerが
      格納されている場所を指定すれば「@Controller」も利用可能ですが、
      SpringMVCの考え方として、通常は「@Controller」が属するパッケージは
      後述のservlet-context.xmlに定義します。

    • 共通設定

      <beans:bean id="commonProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
        <beans:property name="locations" value="classpath:common.properties" />
      </beans:bean>
      

      上記記述を行うことでJavaファイル内でcommon.propertiesに
      「commonProperties」という名前でアクセスできるようになります。

      @Autowired
      private Properties commonProperties;
      
  • src/main/webapp/WEB-INF/spring/appServlet/servlet-context.xml
    プレゼンテーション層(P層:Controller)の定義を行う

    • 「@Controller」のアノテーションを有効にする設定

      <annotation-driven />
      
    • アノテーションでScan対象とするパッケージを指定

      <context:component-scan base-package="jp.co.sample.controller" />
      

      上記パッケージにJavaファイルを配置すると、
      JavaファイルにSpringの管理を示す「S」マークがつきます。

    • 静的ファイルにHTTP GETリクエストが来た場合の動作定義

      <resources mapping="/resources/**" location="/resources/" />
      
    • コントローラーの処理終了後に呼び出される画面用ファイルの指定

      <beans:bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <beans:property name="prefix" value="/WEB-INF/views/" />
        <beans:property name="suffix" value=".jsp" />
      </beans:bean>