Android 开发 有道翻译

本文介绍如何使用Android抓取有道翻译API接口,包括逆向分析加密数据、设置网络权限、构建POST请求及解析JSON响应。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

爬取有道翻译api接口,定制专属于你的翻译官

1.首先是通过浏览器的开发者工具抓取有道翻译的数据包,解析request与response。

2.然后会发现这个是一个post请求,而且居然数据还加密,而这就需要通过js逆向来分析,这里面有MD5,时间戳,随机数,然后再糅合在一起。

3.通过正则解析响应回来的json数据

4.最后就是安卓的UI设计以及业务逻辑代码了。

抓取有道翻译数据

打开有道翻译网页版开始抓取数据包,可以发现salt,sign,lts,bv,这几个请求参数是加密了的,通过js逆向分析可以得出:

  1. i :待翻译的文本
  2. lts:时间戳
  3. salt:时间戳+10以内的随机数
  4. bv:UA请求头通过MD5信息摘要法产生的32位16进制数
  5. sign:固定字符串通过MD5产生的16进制数
  6. 接口就抓取下来了,留着等会用

Android 网络权限设置

问题:无法进行网络请求
原因:Android9.0以后只支持HTTPS请求了,所有的http请求都认为是不安全的,所以不能访问。
解决方法:
第一步:首先是最基本的网络权限 在AndroidManif.xml中添加

<!-- 获取网络权限 -->    		         
<uses-permissionandroid:name="android.permission.INTERNET" />**;

第二步:在res文件夹中新建xml文件夹添加network_security_config.xml文件,文件内容如下:

<?xml version="1.0" encoding="utf-8"?>	                <network-security-config>    	
	<base-config cleartextTrafficPermitted="true" />
</network-security-config>

第三步:在AndroidManif.xml文件中注册写命令文件:

	android:networkSecurityConfig=
                     "@xml/network_security_config"

Android 逻辑代码

  1. 创建全局变量
  2. 在onCreat方法中找到组件
  3. 重写onResume方法
  4. 添加按钮监听事件
  5. 发送post请求
  6. 解析翻译结果
  7. setText的值
//debug调试
    String TAG = "mytag";
    //声明组件component
    Button button0;
    TextView text0,text1,text2;

    //创建全局OkHttp对象
    private OkHttpClient okHttpClient;

    //全局翻译结果变量
    String you_dao ="";

    //更多翻译的词性
    String entries = "";



    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //找到组件
        button0 = findViewById(R.id.button0);
        text0 = findViewById(R.id.text0);
        text1 = findViewById(R.id.text1);
        text2 = findViewById(R.id.text2);
        //创建okhttp对象
        okHttpClient = new OkHttpClient();

    }

    //交互
    @Override
    protected void onResume() {
        super.onResume();
        //翻译按钮监听事件
        button0.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //获取文本框内容
                String string = text0.getText().toString();
                //请求有道翻译
                //interpret(string);
                youdao_interpret(string);
                //线程延时
                sleep(300);
                //通过全局变量在results显示翻译结果
                text1.setText(you_dao);
                text2.setText(entries);


            }
        });


post请求

上面已经说完逻辑了,现在就要实现okhttp向有道翻译发送post请求,并返回翻译结果。
我们刚刚已经解析出加密的数据了,而我们现在要做的就是产生加密的数据并发送给
有道翻译服务器。也就是post请求添加请求头和表单请求体了。

(1)产生加密数据(补充:Android进行网络请求是必须新开线程)
public void youdao_interpret(String arg){
        new Thread(){
            final String string = arg;
            @Override
            public void run(){
                //时间戳 r = ts
                String ts = String.valueOf(System.currentTimeMillis());
                //随机数 i = salt
                Random random = new Random();
                String salt = ts+String.valueOf(random.nextInt(10));
                //请求头
                String ua = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36";
                // md5 t = bv
                String bv = "";
                try {
                    bv = MD5(ua);
                } catch (Exception e) {
                    e.printStackTrace();
                }
                // x
                String x="fanyideskweb" + this.string + salt + "Tbh5E8=q6U3EXe+&L[4c@";
                // sign
                String sign = "";
                try {
                    sign=MD5(x);
                } catch (Exception e) {
                    e.printStackTrace();
                }


(2)封装MD5函数
    //MD5函数
    public static String MD5(String data) throws Exception {

        java.security.MessageDigest md = MessageDigest.getInstance("MD5");

        byte[] array = md.digest(data.getBytes("UTF-8"));

        StringBuilder sb = new StringBuilder();

        for (byte item : array) {

            sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));

        }

        return sb.toString().toUpperCase();

    }
(3)http请求添加请求体和请求头
//请求有道翻译的完整网址
                String url="http://fanyi.youdao.com/translate_o?smartresult=dict&smartresult=rule";
                //创建请求体表单 >>填入请求翻译的内容
                FormBody formBody = new FormBody.Builder()
                        .add("i",this.string)
                        .add("smartresult","dict")
                        .add("salt",salt)
                        .add("sign",sign)
                        .add("lts",ts)
                        .add("bv",bv)
                        .add("client","fanyideskweb")
                        .add("doctype","json")
                        .add("version","2.1")
                        .add("keyfrom","fanyi.web")
                        .build();
                formBody.contentType();
                //创建请求对象  >>并填入form表单
                Request request = new Request.Builder()
                        .url(url)
                        .addHeader("User-Agent",ua)    //请求头
                        .addHeader("Cookie","OUTFOX_SEARCH_USER_ID=-1876541902@10.169.0.84; OUTFOX_SEARCH_USER_ID_NCOO=2111552283.5701284; JSESSIONID=aaaNARG2_9FBJ-Ce81qBx; ___rl__test__cookies=1609821642788")    //请求头
                        .addHeader("Referer","http://fanyi.youdao.com/")    //请求头
                        .post(formBody).build();

                //准备请求对象   Client:客户
                Call call = okHttpClient.newCall(request);
(5)解析返回的json数据,通过正则表达式提取翻译结果
try{
                    Response response = call.execute();
                    //读取响应
                    String string = response.body().string();   //获取响应的字符串 还有byteStream方法,获取二进制数据
                    Log.d(TAG, "post请求成功");
                    //显示返回的数据
                    Log.d(TAG, "返回的全部:"+string);
                    //提取数据
                    Pattern compile =Pattern.compile("\"tgt\":\".*?\"");
                    Pattern compile1 =Pattern.compile("entries\":\\[\"\",\".*");
                    //匹配
                    Matcher matcher = compile.matcher(string);
                    while(matcher.find()){
                        //输出每个匹配的字符
                        you_dao = matcher.group().substring(7,matcher.group().length()-1);
                    }
                    //更多翻译匹配
                    Matcher matcher1 = compile1.matcher(string);
                    while(matcher1.find()){
                        //输出每个匹配的字符
                        entries = matcher1.group().substring(14,matcher1.group().length()-17).replace("\\r\\n\",\"","   ");
                        Log.d(TAG, "词根匹配的为"+entries);
                    }
                }catch (IOException e ){
                    String string = e.getMessage();
                    Log.d(TAG, "post失败");
                }
            }

调试,打包,签名,加固,发布。

1.build 构建apk
2.360加固签名

完整代码

package com.example.translate;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import java.io.IOException;
import java.security.MessageDigest;
import java.util.Date;
import java.util.Random;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import okhttp3.Call;
import okhttp3.FormBody;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.OkHttpClient;

public class MainActivity extends AppCompatActivity {
    //debug调试
    String TAG = "mytag";
    //声明组件component
    Button button0;
    TextView text0,text1,text2;

    //创建全局OkHttp对象
    private OkHttpClient okHttpClient;

    //全局翻译结果变量
    String you_dao ="";

    //更多翻译的词性
    String entries = "";



    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //找到组件
        button0 = findViewById(R.id.button0);
        text0 = findViewById(R.id.text0);
        text1 = findViewById(R.id.text1);
        text2 = findViewById(R.id.text2);
        //创建okhttp对象
        okHttpClient = new OkHttpClient();

    }

    //交互
    @Override
    protected void onResume() {
        super.onResume();
        //翻译按钮监听事件
        button0.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //获取文本框内容
                String string = text0.getText().toString();
                //请求有道翻译
                //interpret(string);
                youdao_interpret(string);
                //线程延时
                sleep(300);
                //通过全局变量在results显示翻译结果
                text1.setText(you_dao);
                text2.setText(entries);


            }
        });

    }
    //有道翻译函数
    public void interpret(String arg){
        new Thread(){
            final String string = arg;
            @Override
            public void run(){
                //请求有道翻译的简化网址
                String url="http://fanyi.youdao.com/translate?smartresult=dict&smartresult=rule";
                //创建请求体表单 >>填入请求翻译的内容
                FormBody formBody = new FormBody.Builder().add("i",this.string).add("doctype","json").build();
                formBody.contentType();
                //创建请求对象  >>并填入form表单
                Request request = new Request.Builder()
                        .url(url)
                        .addHeader("User-Agent","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36")    //请求头
                        .post(formBody).build();

                //准备请求对象   Client:客户
                Call call = okHttpClient.newCall(request);

                //response 是响应对象
                try{
                    Response response = call.execute();

                    //读取响应
                    String string = response.body().string();   //获取响应的字符串 还有byteStream方法,获取二进制数据
                    Log.d(TAG, "post请求");
                    //显示返回的数据
                    Log.d(TAG, string);
                    //提取数据
                    Pattern compile =Pattern.compile("\"tgt\":\".*?\"");
                    //匹配
                    Matcher matcher = compile.matcher(string);
                    while(matcher.find()){
                        //输出每个匹配的字符
                        you_dao = matcher.group().substring(7,matcher.group().length()-1);
                    }
                }catch (IOException e ){
                    String string = e.getMessage();
                    Log.d(TAG, "post失败");
                }
            }
        }.start();

    }

    //js逆向的完整请求函数
    public void youdao_interpret(String arg){
        new Thread(){
            final String string = arg;
            @Override
            public void run(){
                //时间戳 r = ts
                String ts = String.valueOf(System.currentTimeMillis());
                //随机数 i = salt
                Random random = new Random();
                String salt = ts+String.valueOf(random.nextInt(10));
                //请求头
                String ua = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36";
                // md5 t = bv
                String bv = "";
                try {
                    bv = MD5(ua);
                } catch (Exception e) {
                    e.printStackTrace();
                }
                // x
                String x="fanyideskweb" + this.string + salt + "Tbh5E8=q6U3EXe+&L[4c@";
                // sign
                String sign = "";
                try {
                    sign=MD5(x);
                } catch (Exception e) {
                    e.printStackTrace();
                }


                //请求有道翻译的完整网址
                String url="http://fanyi.youdao.com/translate_o?smartresult=dict&smartresult=rule";
                //创建请求体表单 >>填入请求翻译的内容
                FormBody formBody = new FormBody.Builder()
                        .add("i",this.string)
                        .add("smartresult","dict")
                        .add("salt",salt)
                        .add("sign",sign)
                        .add("lts",ts)
                        .add("bv",bv)
                        .add("client","fanyideskweb")
                        .add("doctype","json")
                        .add("version","2.1")
                        .add("keyfrom","fanyi.web")
                        .build();
                formBody.contentType();
                //创建请求对象  >>并填入form表单
                Request request = new Request.Builder()
                        .url(url)
                        .addHeader("User-Agent",ua)    //请求头
                        .addHeader("Cookie","OUTFOX_SEARCH_USER_ID=-1876541902@10.169.0.84; OUTFOX_SEARCH_USER_ID_NCOO=2111552283.5701284; JSESSIONID=aaaNARG2_9FBJ-Ce81qBx; ___rl__test__cookies=1609821642788")    //请求头
                        .addHeader("Referer","http://fanyi.youdao.com/")    //请求头
                        .post(formBody).build();

                //准备请求对象   Client:客户
                Call call = okHttpClient.newCall(request);
                //response 是响应对象
                try{
                    Response response = call.execute();
                    //读取响应
                    String string = response.body().string();   //获取响应的字符串 还有byteStream方法,获取二进制数据
                    Log.d(TAG, "post请求成功");
                    //显示返回的数据
                    Log.d(TAG, "返回的全部:"+string);
                    //提取数据
                    Pattern compile =Pattern.compile("\"tgt\":\".*?\"");
                    Pattern compile1 =Pattern.compile("entries\":\\[\"\",\".*");
                    //匹配
                    Matcher matcher = compile.matcher(string);
                    while(matcher.find()){
                        //输出每个匹配的字符
                        you_dao = matcher.group().substring(7,matcher.group().length()-1);
                    }
                    //更多翻译匹配
                    Matcher matcher1 = compile1.matcher(string);
                    while(matcher1.find()){
                        //输出每个匹配的字符
                        entries = matcher1.group().substring(14,matcher1.group().length()-17).replace("\\r\\n\",\"","   ");
                        Log.d(TAG, "词根匹配的为"+entries);
                    }
                }catch (IOException e ){
                    String string = e.getMessage();
                    Log.d(TAG, "post失败");
                }
            }
        }.start();

    }
    //线程阻塞method
    public void sleep(int ms){
        try{
            Thread.sleep(ms);
        }catch(InterruptedException e){
            e.getMessage();
        }
    }

    //MD5加密函数
    public static String MD5(String data) throws Exception {

        java.security.MessageDigest md = MessageDigest.getInstance("MD5");

        byte[] array = md.digest(data.getBytes("UTF-8"));

        StringBuilder sb = new StringBuilder();

        for (byte item : array) {

            sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));

        }

        return sb.toString().toUpperCase();

    }
}

效果图

在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值