3

このプログラムの移植に数時間取り組んだ後、最終的には動作状態になっているようです。しかし、JavaサーブレットをPythonに移植するためのより良い方法またはより完全な方法を誰かが知っているかどうか疑問に思いました。Pythonスクリプトの冒頭には、プログラムを1行ずつPythonに直接移植しやすくするための多くのサポートコードが含まれています。誰かがこれについて行くためのより良い方法を知っていますか?


Java

// --------------------------------------------------------
// File: Hello.java
// Description: A simple "Hello World" servlet
// --------------------------------------------------------

import javax.servlet.*;
import javax.servlet.http.*;
import java.io.*;
import java.util.*;
import sun.servlet.http.HttpServer;

public class Hello extends HttpServlet {

  int count;

  public void service(HttpServletRequest request, 
         HttpServletResponse response) throws ServletException, IOException {

    // setup response

    response.setContentType("text/html");
    PrintWriter out = response.getWriter();

    // send response

    out.println("<html><body>");
    out.println("<h5>Stephen Paul Chappell (SPC)</h5>");
    out.println("<h5>:) (ZTD) Zero The Dragon :(</h5>");
    String name = request.getParameter("NAME");
    if (name != null && !name.equals("")) {
      out.println("<h2>Hello, " + name + "</h2>");
    } else {
      out.println();
      if (name != null && name.equals("")) {
        out.println("  <h2>You didn't enter your name. Please enter your name. </h2>");
      } else {
        out.println("  <h2>What's your name? </h2>");
      }
      out.println("  <hr>");
      out.println("  <form action=\"http://127.0.0.1:8080/servlet/Hello\">");
      out.println();
      out.println("    Enter your name: <input type=\"text\" name=\"NAME\" value=\"Fred\"><br>");
      out.println("    <input type=\"submit\" value=\"Click for greeting\">");
      out.println();
      out.println("  </form>");
    }
    String color = request.getParameter("FAVCOLOR");
    if (color != null) {
      out.println("<h2>Why, " + color + " is my favorite color too!</h2>");
    }

    count++;
    out.println("This page has been hit " + count + " time(s).");

    out.print("</body></html>");
  }

  // start web server
  public static void main(String[] args) throws Exception {
    HttpServer.main(args);
  }
}

Python

import urllib.parse
import http.server
import cgitb
import sys
import io

################################################################################

class HttpServlet(http.server.BaseHTTPRequestHandler):

    __debug = False

    @staticmethod
    def debug(value):
        HttpServlet.__debug = value

    def do_GET(self):
        if self.path == '/favicon.ico':
            self.send_error(404)
            return
        request = HttpServletRequest(self.path)
        response = HttpServletResponse()
        try:
            self.service(request, response)
        except Exception:
            if HttpServlet.__debug:
                self.send_response(500)
                self.send_header('Content-type', 'text/html')
                self.end_headers()
                klass, value, trace = sys.exc_info()
                html = cgitb.html((klass, value, trace.tb_next))
                self.wfile.write(html.encode())
            else:
                self.send_error(500)
        else:
            self.send_response(200)
            self.send_header('Content-type', response._type)
            self.end_headers()
            self.wfile.write(response._value)

    def service(self, request, response):
        raise NotImplementedError()

################################################################################

class HttpServletRequest:

    def __init__(self, path):
        query = urllib.parse.urlparse(path).query
        self.__dict = urllib.parse.parse_qs(query, True)

    def getParameter(self, name):
        return self.__dict.get(name, [None])[0]

################################################################################

class HttpServletResponse:

    def __init__(self):
        self.__content_type = 'text/plain'
        self.__print_writer = PrintWriter()

    def setContentType(self, content_type):
        self.__content_type = content_type

    def getWriter(self):
        return self.__print_writer

    @property
    def _type(self):
        return self.__content_type

    @property
    def _value(self):
        return self.__print_writer.getvalue().encode()

################################################################################

class PrintWriter(io.StringIO):

    print = io.StringIO.write

    def println(self, string):
        self.write(string + '\r\n')

################################################################################

class HttpServer(http.server.HTTPServer):

    @staticmethod
    def main(RequestHandlerClass, port=80):
        server = HttpServer(('', port), RequestHandlerClass)
        socket = server.socket.getsockname()
        print('Serving HTTP on', socket[0], 'port', socket[1], '...')
        try:
            server.serve_forever()
        except KeyboardInterrupt:
            print('Keyboard interrupt received: EXITING')
            server.server_close()

################################################################################
################################################################################

## ---------------------------------------------------------
## File: Hello.py
## Description: A simple "Hello World" servlet
## ---------------------------------------------------------

class Hello(HttpServlet):

    count = 0

    def service(self, request, response):

        # Setup for a response.
        response.setContentType('text/html')
        out = response.getWriter()

        # Send an appropriate response.
        out.println('''\
<html>
    <head>
        <title>Java to Python servlet</title>
    </head>
    <body>
        <h5>Stephen Paul Chappell (SPC)</h5>
        <h5>:) (ZTD) Zero The Dragon :(</h5>''')
        name = request.getParameter('NAME')
        if name:
            out.println('''\
        <h2>Hello, {}!</h2>'''.format(name))
        else:
            if name == '':
                out.println('''\
        <h2>You did not enter your name.</h2>
        <h3>Please enter your name.</h3>''')
            else:
                out.println('''\
        <h2>What is your name?</h2>''')
            out.println('''\
        <form>
            <fieldset>
                <legend>About Yourself</legend>
                <label for='NAME'>Enter your name:</label>
                <input id='NAME' name='NAME' type='text' value='John Doe' />
                <br />
                <input type='submit' value='Click me!' />
            </fieldset>
        </form>''')
        color = request.getParameter('FAVCOLOR')
        if color:
            out.println('''\
        <h2>Why, {} is my favorite color too!</h2>'''.format(color))

        Hello.count += 1
        out.println('''\
        This page has been hit {} times.'''.format(Hello.count))

        out.print('''\
    </body>
</html>''')

# Start the web server.
def main():
    HttpServlet.debug(True)
    HttpServer.main(Hello)

################################################################################

if __name__ == '__main__':
    main()

このプロジェクトの目的が何か疑問に思われる方のために、オリジナルのJavaプログラムは、新入生レベルのコンピュータサイエンスコースの入門演習として提供されました。それ以来、私がJavaを使用して古いプログラムを実行し、チャレンジと学習体験のためにそれらをPythonに移植してから、約6年になります。サーブレットの移植には、さらに困難が伴います。

4

4 に答える 4

7

私の最善の提案は、行ごとにポートを記述しないことです。PythonはJavaではないので、Python言語でJavaスタイルのコードを書いているだけなら、自分を困惑させていることになります。

一般的な違い/間違いのいくつかを認識するためのスターターは次のとおりです。http://dirtsimple.org/2004/12/python-is-not-java.html

幸運にも単体テストを含むコードベースで作業できる場合、これを実行するための優れた方法は、単体テストをJavaからPythonに移植してから、単体テストに一致する新しいPythonを作成することです。それがあなたが持っているものではない場合は、ソフトウェアがJavaで何をするかを認識し、同じことを行うPythonコードを書くことがはるかに重要です。少なくとも行ごとのレベルでは、ほぼ確実に同じ方法ではありません。

于 2010-01-16T22:29:45.413 に答える
3

これは翻訳ではなく、移植ではありませんか?:)

私はちょうどグーグルで検索し、java2python同様のSOの質問を見つけました。

ともかく ...

理想的には、何を翻訳するのか、何を翻訳する必要がないのかを理解できるように、何を翻訳しているのかを理解する必要があると思います。Javaの構成がPythonの類似とどのように関連しているか、およびライブラリが翻訳でどのように役割を果たすかを知るには、両方の言語を合理的に理解する必要があります。

言語間の直接翻訳は、多くの場合、適切なアプローチではありません。あなたの質問で翻訳されたPythonがあまりPythonicではないという理由だけで、これについて言及します。また、かなりの量のPythonがHTTPサーバーの側面に関係しているためです。これは、Javaコードには当てはまりません。したがって、「公正な」翻訳を作成するには、サーブレットコンテナがJavaに対して行うことをPythonに対して行うPythonフレームワークに依存していました。多分竜巻/ジャンゴ。私の言いたいことは、Javaサーブレットは重要な一連の標準にラップされているので、Pythonバージョンが同様のルートに従うことは理にかなっています。これは、少なくともWSGIを意味します。

于 2010-01-16T22:54:41.837 に答える
2

この方法でPythonWebアプリを作成することは、混乱を招き、Pythonではありません。Javaから1行ずつ移植するよりも、Djangoでアプリを書き直す方がうまくいく可能性があります

于 2010-01-16T23:15:18.063 に答える
2

直接的な答えではありませんが、他の機能の中でも特にWebwareのようなものを使用しない正当な理由(概要を参照):

  • サーブレット。Javaサーブレットと同様に、これらはWebアプリケーションを構築するための使い慣れた基盤を提供します。
于 2010-01-16T22:42:48.497 に答える