为了账号安全,请及时绑定邮箱和手机立即绑定

连接 Java 和 Python Flask

连接 Java 和 Python Flask

至尊宝的传说 2023-08-09 16:16:31
我有一个简单的 Flask API:from flask import Flask, jsonifyapp = Flask(__name__)@app.route('/')def hello_world():    return 'Hello World!'@app.route('/add/<params>', methods = ['GET'])def add_numbers(params):    #params is expected to be a dictionary: {'x': 1, 'y':2}    params = eval(params)    return jsonify({'sum': params['x'] + params['y']})if __name__ == '__main__':    app.run(debug=True)现在,我想从 Java 调用这个方法并提取结果。我尝试过使用java.net.URL和java.net.HttpURLConnection;import java.io.*;import java.net.HttpURLConnection;import java.net.MalformedURLException;import java.net.URL;public class MyClass {    public static void main(String[] args) {        try {            URL url = new URL("http://127.0.0.1:5000/add/{'x':100, 'y':1}");            HttpURLConnection conn = (HttpURLConnection) url.openConnection();            conn.setRequestMethod("GET");            conn.setRequestProperty("Accept", "application/json");            if (conn.getResponseCode() != 200) {                throw new RuntimeException("Failed : HTTP error code : "                        + conn.getResponseCode());            }            BufferedReader br = new BufferedReader(new InputStreamReader(                    (conn.getInputStream())));            String output;            System.out.println("Output from Server .... \n");            while ((output = br.readLine()) != null) {                System.out.println(output);            }            conn.disconnect();        } catch (MalformedURLException e) {            e.printStackTrace();        }catch (IOException e){e.printStackTrace();        }    }}但这不起作用。在烧瓶服务器中,我收到一条错误消息:代码 400,消息请求语法错误(“GET /add/{'x':100, 'y':1} HTTP/1.1”)“GET /add/{'x':100, 'y':1} HTTP/1.1” HTTPStatus.BAD_REQUEST -在 Java 代码中,我收到错误:线程“main”中的异常 java.lang.RuntimeException:失败:HTTP 错误代码:-1 在 MyClass.main(MyClass.java:17)我究竟做错了什么?我的最终目标是将字典对象传递给我的Python函数并将函数的响应返回给java。该词典可以包含超过一千个单词的文本值。我怎样才能实现这个目标?
查看完整描述

2 回答

?
鸿蒙传说

TA贡献1865条经验 获得超7个赞

正如上面 @TallChuck 的评论所述,您需要替换或删除 URL 中的空格


URL url = new URL("http://127.0.0.1:5000/add?x=100&y=12&text='Test'");

我建议使用请求对象来检索 GET 调用中的参数。


请求对象

要访问 Flask 中的传入数据,您必须使用请求对象。请求对象保存来自请求的所有传入数据,其中包括 mimetype、referrer、IP 地址、原始数据、HTTP 方法和标头等。尽管请求对象保存的所有信息都可能有用,但我们只关注通常由端点调用者直接提供的数据。


正如在发布大量参数和数据的评论中提到的,此任务的更合适的实现可能是使用 POST 方法。


以下是后端 POST 相同实现的示例:


from flask import Flask, jsonify, request

import json


app = Flask(__name__)


@app.route('/')

def hello_world():

    return 'Hello World!'


@app.route('/add/', methods = ['POST'])

def add_numbers():

    if request.method == 'POST':

        decoded_data = request.data.decode('utf-8')

        params = json.loads(decoded_data)

        return jsonify({'sum': params['x'] + params['y']})


if __name__ == '__main__':

    app.run(debug=True)

这是使用 cURL 测试 POST 后端的简单方法:


 curl -d '{"x":5, "y":10}' -H "Content-Type: application/json" -X POST http://localhost:5000/add

使用Java发布请求:


import java.io.*;

import java.net.HttpURLConnection;

import java.net.MalformedURLException;

import java.net.URL;

import java.nio.charset.StandardCharsets;


public class PostClass {

    public static void main(String args[]){

        HttpURLConnection conn = null;

        DataOutputStream os = null;

        try{

            URL url = new URL("http://127.0.0.1:5000/add/"); //important to add the trailing slash after add

            String[] inputData = {"{\"x\": 5, \"y\": 8, \"text\":\"random text\"}",

                    "{\"x\":5, \"y\":14, \"text\":\"testing\"}"};

            for(String input: inputData){

                byte[] postData = input.getBytes(StandardCharsets.UTF_8);

                conn = (HttpURLConnection) url.openConnection();

                conn.setDoOutput(true);

                conn.setRequestMethod("POST");

                conn.setRequestProperty("Content-Type", "application/json");

                conn.setRequestProperty( "charset", "utf-8");

                conn.setRequestProperty("Content-Length", Integer.toString(input.length()));

                os = new DataOutputStream(conn.getOutputStream());

                os.write(postData);

                os.flush();


                if (conn.getResponseCode() != 200) {

                    throw new RuntimeException("Failed : HTTP error code : "

                            + conn.getResponseCode());

                }


                BufferedReader br = new BufferedReader(new InputStreamReader(

                        (conn.getInputStream())));


                String output;

                System.out.println("Output from Server .... \n");

                while ((output = br.readLine()) != null) {

                    System.out.println(output);

                }

                conn.disconnect();

            }

    } catch (MalformedURLException e) {

        e.printStackTrace();

    }catch (IOException e){

        e.printStackTrace();

    }finally

        {

            if(conn != null)

            {

                conn.disconnect();

            }

        }

    }

}


查看完整回答
反对 回复 2023-08-09
?
函数式编程

TA贡献1807条经验 获得超9个赞

你的 python 代码有一个严重的设计缺陷,这会产生一个非常危险的安全缺陷,并且(幸运的是,考虑到安全缺陷的存在)是你的代码无法工作的原因。

在 URL 中的简单字符串旁边放置任何内容都是不好的做法,因为:

  1. URL 应该是地址,从语义上讲,将它们用作数据载体没有什么意义

  2. 它通常需要杂乱的代码来生成和读取(在您的示例中,您被迫使用eval来解析请求,这是极其危险的)

  3. URL 的规则要求对字符进行编码(读起来很糟糕%20等等)

如果您期望参数数量固定,则应该使用查询参数,否则最好使用请求正文。考虑到你的逻辑是什么,我认为使用查询参数在语义上会更好(所以你的请求看起来像/add?x=100&y=1)。

一般来说,eval是你的敌人,而不是你的朋友,并且eval通过网络发送给你的东西是你的敌人。

查看完整回答
反对 回复 2023-08-09
  • 2 回答
  • 0 关注
  • 120 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信