3

经过几个小时的工作来移植这个程序,它似乎终于处于工作状态。但是,我想知道是否有人知道将 Java servlet 移植到 Python 中的更好或更完整的方法。Python 脚本的开头包含大量支持代码,可以更轻松地将程序逐行直接移植到 Python 中。有谁知道更好的方法来解决这个问题?


爪哇

// --------------------------------------------------------
// 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 中以应对挑战和学习经验。移植 servlet 带来了额外的困难。

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 question

反正 ...

理想情况下,我认为您应该了解您要翻译的内容,这样您就可以知道要翻译的内容和不需要翻译的内容。您需要对这两种语言有一个合理的理解,以了解 Java 中的构造如何与 Python 中的类似构造相关联,以及库如何在翻译中发挥作用。

语言之间的直接翻译通常不是一个好方法。我之所以提到这一点,只是因为您问题中翻译的 Python 不是很 Pythonic。也因为相当多的 Python 与 HTTP Server 方面有关。Java 代码并非如此。所以为了产生一个“公平”的翻译,我会依赖一个 Python 框架,它为 Python 做 Servlet 容器为 Java 做的事情。也许是龙卷风/姜戈。我的观点是,Java Servlet 包含在一组重要的标准中,因此 Python 版本遵循类似的路线是有道理的,这至少可能意味着WSGI

于 2010-01-16T22:54:41.837 回答
2

不是一个直接的答案,但有充分的理由不使用Webware之类的东西,它提供了其他功能(参见概述):

  • 小服务程序。与 Java servlet 类似,它们为构建 Web 应用程序提供了熟悉的基础。
于 2010-01-16T22:42:48.497 回答
2

以这种方式创建 python webapp 是令人困惑且非 python 的。用 Django 重写你的应用程序可能比从 Java 中逐行移植它更成功

于 2010-01-16T23:15:18.063 回答