当前位置: 首页 > news >正文

崆峒区城乡建设局网站seo查询爱站

崆峒区城乡建设局网站,seo查询爱站,涪陵网站设计,上海专业做网站排名背景:由于项目需要,需要将apk包加入服务端返回的静态资源文件到apk中,形成离线apk包供下载安装。经过调查研究,决定使用apktool实现。关于apktool的资料可以参考 https://blog.csdn.net/quantum7/article/details/124060620 htt…

背景:由于项目需要,需要将apk包加入服务端返回的静态资源文件到apk中,形成离线apk包供下载安装。经过调查研究,决定使用apktool实现。关于apktool的资料可以参考

https://blog.csdn.net/quantum7/article/details/124060620

https://blog.csdn.net/qq_20451879/article/details/117300056

windows版环境

1.JDK环境

2.下载apktool.jar

打包流程:

apktool下载地址:https://ibotpeaches.github.io/Apktool/
3.解压apk包
java -jar apktool_2.6.1.jar d app-release.apk
4删除签名文件
签名文件在解压文件后的\original\META-INF目录下
C:\Users***\Downloads\app-release1111\original\META-INF

5.添加要替换的文件到
C:\Users***\Downloads\app-release\assets\assets下

6.生成签名文件

.keystore 签名方式:

keytool -genkey -alias test.keystore -keyalg RSA -validity 20000 -keystore test.keystore

.jks方式:

keytool -genkey -v -keystore test.jks -alias test-keyalg RSA -keysize 2048 -validity 20000

keytool -importkeystore -srckeystore test.jks -destkeystore test.jks -deststoretype pkcs12

7.重新打包
java -jar apktool_2.6.1.jar b app-release

8.使用重新打包后的apk和签名文件打包

.keystore重新签名打包方式:

jarsigner -verbose -keystore test.keystore -signedjar app-release-1-0224.apk app-release-1.apk test.keystore

.jks重新签名打包方式:

jarsigner -verbose -keystore test.jks -signedjar 222.apk test.apk test

java环境

构建脚本

bulidApk.bat

@echo off
start cmd /k "cd C:\Users\aipingh\Downloads && java -jar C:\Users\aipingh\Downloads\apktool.jar b C:\Users\***\Downloads\app-release8"

rebuildKeystoreApk.bat

@echo off
start cmd /k "cd C:\Users\aipingh\Downloads && jarsigner -verbose -keystore tinnove.keystore -storepass 123456 -signedjar C:\Users\***\Downloads\app-release8\dist\app-release.apk C:\Users\aipingh\Downloads\app-release8\dist\app-release8.apk test.keystore"

代码:

import ch.qos.logback.core.util.FileUtil;
import cn.hutool.core.io.FileUtil;import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URLEncoder;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;public class ApkUtil {private String outPth;// windows版下载public void downloadWindowsOfflineApk(InfoReqVO reqVO, HttpServletResponse response) {try {//  apk解压包路径String apkOriginalPath = "C:\\Users\\***\\Downloads\\app-release8\\";// 下载离线js文件到目标apk的资源文件路径中String fullPath = apkOriginalPath + "original\\META-INF\\";downloadJsFile(reqVO);//删除签名文件File mkdir = FileUtil.mkdir(fullPath);//去掉签名FileUtils.deleteTempFiles(mkdir, fullPath);//重新打包try {String commandStr = "cmd /c C:\\Users\\***\\Downloads\\buildApk.bat";Runtime.getRuntime().exec(commandStr);} catch (IOException e) {}// 下载签名文件到dist目录中String packagePath = "dist";File packagePathFile = FileUtil.mkdir(apkOriginalPath + packagePath);String keystorePath = "https://***/apk/keystore/tinnove.keystore";ImageInfo appDesignDetailImageInfo = new ImageInfo();appDesignDetailImageInfo.setFilename("tinnove.keystore");appDesignDetailImageInfo.setPathUrl(keystorePath);downloadFile(packagePathFile, appDesignDetailImageInfo);// 加签名后打包APKtry {String commandStr = "cmd /c C:\\Users\\***\\Downloads\\rebuildKeystoreApk.bat";Runtime.getRuntime().exec(commandStr).waitFor(30, TimeUnit.MILLISECONDS);} catch (IOException e) {} catch (InterruptedException e) {e.printStackTrace();}// 重新构建后的apk文件地址String newApkName = "app-release.apk";String newApkPath = apkOriginalPath + "dist\\" + newApkName;// 将apk包返回给前端File file = new File(newApkPath);response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("app-release.apk", "UTF-8"));//获取文件的输入流InputStream fis = new FileInputStream(file);byte[] buffer = new byte[1024 * 5];int r;while ((r = fis.read(buffer)) != -1) {response.getOutputStream().write(buffer, 0, r);}// 删除build目录,方便下次打包FileUtils.deleteTempFiles(null, apkOriginalPath + "build");// 删除dist及目录下的apk包FileUtils.deleteTempFiles(null, apkOriginalPath + "dist");} catch (IOException e) {}}//liunx版下载public void apkShellDownload(InfoReqVO reqVO, HttpServletResponse response) {try {//下载原始apkFile designOfflineFile = cn.hutool.core.io.FileUtil.mkdir(outPth);String designOfflineApkPath = "https://***/apk/offline/app-release.apk";ImageInfo detailImageInfo = new ImageInfo();detailImageInfo.setFilename("app-release.apk");detailImageInfo.setPathUrl(designOfflineApkPath);downloadFile(designOfflineFile, detailImageInfo);//下载apktool.jar工具包String apktoolPath = "https://***/apk/offline/apktool.jar";ImageInfo imageInfo = new ImageInfo();imageInfo.setFilename("apktool.jar");imageInfo.setPathUrl(apktoolPath);downloadFile(designOfflineFile, imageInfo);//解压原始apkFileUtils.execSh("cd  " + outPth + " && " + "java -jar " + outPth + " apktool.jar  d  " + outPth + "app-release.apk");//  apk解压包路径String apkOriginalPath = outPth + "app-release/";// 下载离线js文件到目标apk的资源文件路径中String fullPath = apkOriginalPath + "original/META-INF/";downloadJsFile(reqVO);//删除签名文件 去掉签名cn.hutool.core.io.FileUtil.clean(fullPath);//重新打包FileUtils.execSh("cd " + outPth + " && " + " java -jar" + outPth + "apktool.jar b  " + outPth + "app-release", 5, TimeUnit.MILLISECONDS);// 下载签名文件到dist目录中String packagePath = "dist/";File packagePathFile = cn.hutool.core.io.FileUtil.mkdir(apkOriginalPath + packagePath);String keystorePath = "https://***/apk/keystore/test.keystore";ImageInfo appDesignDetailImageInfo = new ImageInfo();appDesignDetailImageInfo.setFilename("test.keystore");appDesignDetailImageInfo.setPathUrl(keystorePath);downloadFile(packagePathFile, appDesignDetailImageInfo);// 加签名后打包APKFileUtils.execSh("cd  " + outPth + " &&  " + "jarsigner -verbose -keystore test.keystore -storepass 123456 -signedjar  "+ apkOriginalPath + packagePath + "app-release-offline.apk   " + apkOriginalPath + packagePath + "app-release.apk test.keystore", 5, TimeUnit.MILLISECONDS);//            List<File> fileList = cn.hutool.core.io.FileUtil.loopFiles(outPth);// 重新构建后的apk文件地址String newApkName = "app-release-offline.apk";String newApkPath = apkOriginalPath + packagePath;// 将apk包返回给前端File file = cn.hutool.core.io.FileUtil.file(newApkPath, newApkName);if (file.exists()) {response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("app-release.apk", "UTF-8"));//获取文件的输入流InputStream fis = new FileInputStream(file);byte[] buffer = new byte[1024 * 5];int r;while ((r = fis.read(buffer)) != -1) {response.getOutputStream().write(buffer, 0, r);}}// 删除build目录,方便下次打包cn.hutool.core.io.FileUtil.clean(apkOriginalPath + "build");// 删除dist及目录下的apk包cn.hutool.core.io.FileUtil.clean(apkOriginalPath + "dist");} catch (IOException e) {}}private void downloadFile(File target, ImageInfo detailImageInfo) throws IOException {File file = org.apache.commons.io.FileUtils.getFile(target, detailImageInfo.getFilename());FileOutputStream outputStream = new FileOutputStream(file);//获取文件的网络输入流byte[] bytes = cn.hutool.http.HttpUtil.downloadBytes(detailImageInfo.getPathUrl());InputStream fis = new ByteArrayInputStream(bytes);byte[] buffer = new byte[1024 * 5];int r;while ((r = fis.read(buffer)) != -1) {outputStream.write(buffer, 0, r);}fis.close();outputStream.close();}private void downloadJsFile(InfoReqVO reqVO) {try {//apk包所在的服务器路径String fullPath = outPth + "/app-release/" + "/assets/app-data/";//本地路径
//            String fullPath = "C:\\Users\\xxx\\Downloads\\app-release\\assets\\app-data";
//        String fullPath = designOfflinePath + "/" + reqVO.getDesignId() + "/" + reqVO.getCount() + "/";File target = cn.hutool.core.io.FileUtil.mkdir(new File(fullPath));
//            File target = new File(fullPath + "preview");
//
//            // 返回图片文件夹List<ImageInfo> designDetailImages = new ArrayList<>();downloadFiles(designDetailImages, target);// 返回逻辑连线 json文件List<Object> designLogicWiring = new ArrayList<>();String logicWiringListJs = "let  logicWiringList  = " + cn.hutool.json.JSONUtil.toJsonStr(designLogicWiring);FileUtils.object2JsonFile(fullPath + "logicWiring.js", logicWiringListJs);// 返回图片url json文件
//            List<AppDesignDetailImageInfo> designDetailImageUrls = getImagesInfo(reqVO);
//            String previewJs = "let  previewImageUrls = " + JSONUtil.toJsonStr(designDetailImageUrls);
//            FileUtils.object2JsonFile(fullPath + "preview.js", previewJs);// 返回分组 json文件List<Object> groupList = new ArrayList<>();String groupListJs = "let  groupList = " + cn.hutool.json.JSONUtil.toJsonStr(groupList);FileUtils.object2JsonFile(fullPath + "group.js", groupListJs);} catch (Exception e) {}}private void downloadFiles(List<ImageInfo> designDetailImages, File target) {try {//将输出流转换成Zip输出流for (ImageInfo detailImageInfo : designDetailImages) {downloadFile(target, detailImageInfo);}} catch (IOException e) {}}}

import cn.hutool.core.util.ObjectUtil;
import cn.hutool.json.JSONUtil;
import com.google.gson.Gson;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.http.MediaType;
import org.springframework.util.MimeTypeUtils;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.commons.CommonsMultipartFile;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.StringJoiner;
import java.util.concurrent.TimeUnit;/*** 文件处理工具类** @author*/
@Slf4j
public class FileUtils {/*** 字符常量:斜杠 {@code '/'}*/public static final char SLASH = '/';/*** 字符常量:反斜杠 {@code '\\'}*/public static final char BACKSLASH = '\\';public static String FILENAME_PATTERN = "[a-zA-Z0-9_\\-\\|\\.\\u4e00-\\u9fa5]+";/*** 输出指定文件的byte数组** @param filePath 文件路径* @param os       输出流* @return*/public static void writeBytes(String filePath, OutputStream os) throws IOException {FileInputStream fis = null;try {File file = new File(filePath);if (!file.exists()) {throw new FileNotFoundException(filePath);}fis = new FileInputStream(file);byte[] b = new byte[1024];int length;while ((length = fis.read(b)) > 0) {os.write(b, 0, length);}} catch (IOException e) {throw e;} finally {if (os != null) {try {os.close();} catch (IOException e1) {e1.printStackTrace();}}if (fis != null) {try {fis.close();} catch (IOException e1) {e1.printStackTrace();}}}}/*** 删除文件** @param filePath 文件* @return*/public static boolean deleteFile(String filePath) {boolean flag = false;File file = new File(filePath);// 路径为文件且不为空则进行删除if (file.isFile() && file.exists()) {file.delete();flag = true;}return flag;}/*** 文件名称验证** @param filename 文件名称* @return true 正常 false 非法*/public static boolean isValidFilename(String filename) {return filename.matches(FILENAME_PATTERN);}/*** 检查文件是否可下载** @param resource 需要下载的文件* @return true 正常 false 非法*/public static boolean checkAllowDownload(String resource) {// 禁止目录上跳级别if (StringUtils.contains(resource, "..")) {return false;}// 检查允许下载的文件规则if (ArrayUtils.contains(MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION, FileTypeUtils.getFileType(resource))) {return true;}// 不在允许下载的文件规则return false;}/*** 下载文件名重新编码** @param request  请求对象* @param fileName 文件名* @return 编码后的文件名*/public static String setFileDownloadHeader(HttpServletRequest request, String fileName) throws UnsupportedEncodingException {final String agent = request.getHeader("USER-AGENT");String filename = fileName;if (agent.contains("MSIE")) {// IE浏览器filename = URLEncoder.encode(filename, "utf-8");filename = filename.replace("+", " ");} else if (agent.contains("Firefox")) {// 火狐浏览器filename = new String(fileName.getBytes(), "ISO8859-1");} else if (agent.contains("Chrome")) {// google浏览器filename = URLEncoder.encode(filename, "utf-8");} else {// 其它浏览器filename = URLEncoder.encode(filename, "utf-8");}return filename;}/*** 返回文件名** @param filePath 文件* @return 文件名*/public static String getName(String filePath) {if (null == filePath) {return null;}int len = filePath.length();if (0 == len) {return filePath;}if (isFileSeparator(filePath.charAt(len - 1))) {// 以分隔符结尾的去掉结尾分隔符len--;}int begin = 0;char c;for (int i = len - 1; i > -1; i--) {c = filePath.charAt(i);if (isFileSeparator(c)) {// 查找最后一个路径分隔符(/或者\)begin = i + 1;break;}}return filePath.substring(begin, len);}/*** 获取文件名,不带后缀** @param filePath* @return*/public static String getFilename(String filePath) {String name = getName(filePath);if (null == name) {return null;}if (name.contains(".")) {int end = name.indexOf(".");return name.substring(0, end);}return name;}/*** 获取文件后缀 如:.zip** @param filePath* @return*/public static String getFilenameSuffix(String filePath) {String name = getName(filePath);if (null == name) {return null;}if (name.contains(".")) {int end = name.indexOf(".");return name.substring(end);}return name;}/*** 是否为Windows或者Linux(Unix)文件分隔符<br>* Windows平台下分隔符为\,Linux(Unix)为/** @param c 字符* @return 是否为Windows或者Linux(Unix)文件分隔符*/public static boolean isFileSeparator(char c) {return SLASH == c || BACKSLASH == c;}/*** 下载文件名重新编码** @param response     响应对象* @param realFileName 真实文件名* @return*/public static void setAttachmentResponseHeader(HttpServletResponse response, String realFileName) throws UnsupportedEncodingException {String percentEncodedFileName = percentEncode(realFileName);StringBuilder contentDispositionValue = new StringBuilder();contentDispositionValue.append("attachment; filename=").append(percentEncodedFileName).append(";").append("filename*=").append("utf-8''").append(percentEncodedFileName);response.setHeader("Content-disposition", contentDispositionValue.toString());response.setHeader("download-filename", percentEncodedFileName);}/*** 百分号编码工具方法** @param s 需要百分号编码的字符串* @return 百分号编码后的字符串*/public static String percentEncode(String s) throws UnsupportedEncodingException {String encode = URLEncoder.encode(s, StandardCharsets.UTF_8.toString());return encode.replaceAll("\\+", "%20");}/*** 获取路径下所有文件名和文件路径* 以,分割** @param dirPath 目录路径* @return hashMap name和url*/public static HashMap<String, String> getMapPath(String dirPath) {HashMap<String, String> pathMap = new HashMap<String, String>();File dirFile = new File(dirPath);String[] fileName = dirFile.list();StringJoiner joiner = new StringJoiner(",");for (String name : fileName) {joiner.add(dirPath + name);}pathMap.put("name", String.join(",", fileName));pathMap.put("url", joiner.toString());return pathMap;}public static boolean deleteAllFile(String dir) {File dirFile = new File(dir);// 如果dir对应的文件不存在,或者不是一个目录,则退出if ((!dirFile.exists()) || (!dirFile.isDirectory())) {return false;}boolean flag = true;// 删除文件夹中的所有文件包括子文件夹File[] files = dirFile.listFiles();for (int i = 0; i < files.length; i++) {// 删除子文件if (files[i].isFile()) {flag = deleteFileFlag(files[i].getAbsolutePath());if (!flag) {break;}}// 删除子文件夹else if (files[i].isDirectory()) {flag = deleteAllFile(files[i].getAbsolutePath());if (!flag) {break;}}}if (!flag) {return false;}// 删除当前文件夹if (dirFile.delete()) {return true;} else {return false;}}/*** 删除文件返回bool** @param fileName* @return boolean*/public static boolean deleteFileFlag(String fileName) {File file = new File(fileName);// 如果文件路径只有单个文件if (file.exists() && file.isFile()) {if (file.delete()) {return true;} else {return false;}} else {return false;}}/*** json文件转json对象** @param data 文件流* @return json对象*/public static Map readJsonFile(byte[] data) {Gson gson = new Gson();String json = "";try {Reader reader = new InputStreamReader(new ByteArrayInputStream(data), StandardCharsets.UTF_8);int ch = 0;StringBuilder buffer = new StringBuilder(1024);while ((ch = reader.read()) != -1) {buffer.append((char) ch);}reader.close();json = buffer.toString();return gson.fromJson(json, Map.class);} catch (IOException e) {log.error("json文件转json对象失败,原因是e={}", e.getMessage());return Collections.emptyMap();}}/*** Object 转换为 json 文件** @param finalPath finalPath 是绝对路径 + 文件名,请确保欲生成的文件所在目录已创建好* @param content   需要被转换的 content*/public static void object2JsonFile(String finalPath, String content) {try {OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(finalPath), StandardCharsets.UTF_8);osw.write(content);osw.flush();osw.close();} catch (IOException e) {e.printStackTrace();}}/*** 将java对象转成json文件返回给前端** @param object   转换为 json* @param fileName json文件名称* @param response 结果*/public static void object2JsonFile(Object object, String fileName, HttpServletResponse response) {try {response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8"));//获取文件的网络输入流byte[] bytes = JSONUtil.toJsonStr(object).getBytes(StandardCharsets.UTF_8);InputStream fis = new ByteArrayInputStream(bytes);byte[] buffer = new byte[1024 * 5];int r;while ((r = fis.read(buffer)) != -1) {response.getOutputStream().write(buffer, 0, r);}} catch (IOException e) {e.printStackTrace();}}/*** 获取封装得MultipartFile** @param inputStream inputStream* @param fileName    fileName* @return MultipartFile*/private MultipartFile getMultipartFile(InputStream inputStream, String fileName) {FileItem fileItem = createFileItem(inputStream, fileName);//CommonsMultipartFile是feign对multipartFile的封装,但是要FileItem类对象return new CommonsMultipartFile(fileItem);}/*** FileItem类对象创建** @param inputStream inputStream* @param fileName    fileName* @return FileItem*/public FileItem createFileItem(InputStream inputStream, String fileName) {FileItemFactory factory = new DiskFileItemFactory(16, null);String textFieldName = "file";FileItem item = factory.createItem(textFieldName, MediaType.MULTIPART_FORM_DATA_VALUE, true, fileName);int bytesRead = 0;byte[] buffer = new byte[8192];OutputStream os = null;//使用输出流输出输入流的字节try {os = item.getOutputStream();while ((bytesRead = inputStream.read(buffer, 0, 8192)) != -1) {os.write(buffer, 0, bytesRead);}inputStream.close();} catch (IOException e) {log.error("Stream copy exception", e);throw new IllegalArgumentException("文件上传失败");} finally {if (os != null) {try {os.close();} catch (IOException e) {log.error("Stream close exception", e);}}if (inputStream != null) {try {inputStream.close();} catch (IOException e) {log.error("Stream close exception", e);}}}return item;}public static int execSh(String bashCommand) {log.info("开始执行shell命令bashCommand={}", bashCommand);int status = 0;try {Runtime runtime = Runtime.getRuntime();String[] bash = {"/bin/bash", "-c", bashCommand};Process exec = runtime.exec(bash);status = exec.waitFor();if (status != 0) {return 1;}} catch (IOException | InterruptedException e) {log.error("执行shell命令bashCommand={}失败,原因是e={}", bashCommand, e.getMessage());}return status;}/*** 执行Shell脚本 0成功 1失败*/public static boolean execSh(String bashCommand, long time, TimeUnit timeUnit) {try {log.info("开始执行shell命令bashCommand={}", bashCommand);Runtime runtime = Runtime.getRuntime();String[] bash = {"/bin/bash", "-c", bashCommand};Process exec = runtime.exec(bash);return exec.waitFor(time, timeUnit);} catch (IOException | InterruptedException e) {log.error("执行shell命令bashCommand={}失败,原因是e={}", bashCommand, e.getMessage());}return false;}public static void deleteTempFiles(File file2, String descDir) {File file1 = new File(descDir);//删除zip解压的数据if (ObjectUtil.isNotEmpty(file1) && file1.exists()) {log.info("file1={}", file1.getPath());deleteFile(file1);}//删除zip文件//删除zip文件if (ObjectUtil.isNotEmpty(file2) && file2.exists()) {log.info("file2={}", file2.getPath());deleteFile(file2);}}public static void deleteFile(File file) {if (file == null) {log.info("deleteFile结果file=null");return;}if (file.isFile()) {boolean delete = file.delete();log.info("删除结果file={},result={}", file.getPath(), delete);} else if (file.isDirectory()) {for (File sub : file.listFiles()) {deleteFile(sub);}file.delete();}}/*** 根据byte数组,生成文件** @param bfile    文件数组* @param filePath 文件存放路径* @param fileName 文件名称*/public static File byte2File(byte[] bfile, String filePath, String fileName) {BufferedOutputStream bos = null;FileOutputStream fos = null;File file = null;try {File dir = new File(filePath);if (!dir.exists() && !dir.isDirectory()) {//判断文件目录是否存在dir.mkdirs();}file = new File(filePath + fileName);fos = new FileOutputStream(file);bos = new BufferedOutputStream(fos);bos.write(bfile);} catch (Exception e) {log.error("byte数组,生成文件失败,原因是e={}", e.getMessage());} finally {try {if (bos != null) {bos.close();}if (fos != null) {fos.close();}} catch (Exception e) {log.error(">>>>   byte2File  error" + e.getMessage());e.printStackTrace();}}return file;}public static byte[] base64StrToBytes(String base64Str) {byte[] bts = org.apache.tomcat.util.codec.binary.Base64.decodeBase64(base64Str);for (int k = 0; k < bts.length; ++k) {//调整异常数据if (bts[k] < 0) {bts[k] += 256;}}return bts;}}
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;/*** @Author:* @Description: 校验分享链接入参*/@Data
@ApiModel("入参")
public class ImageInfo {@ApiModelProperty(name = "id", value = "id", required = true)private String id;@ApiModelProperty(value = "资源路径")private String pathUrl;@ApiModelProperty(value = "文件名称")private String filename;}
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;/*** 入参** @author **** @since 2023/02/16*/@Data
@ApiModel(入参")
public class InfoReqVO {@ApiModelProperty(value = "id")private Long id;@ApiModelProperty(value = "版本号")private Integer version;
}
http://www.dinnco.com/news/35695.html

相关文章:

  • 嵌入式软件能干一辈子seo营销工具
  • 用php开发wap网站东莞网络推广招聘
  • 青羊区电商型网站建设设计兰州网站seo诊断
  • 网站开发与维护的内容wordpress seo教程
  • 网站线框图怎样做中国营销传播网官网
  • 南阳做玉器网站社群营销平台有哪些
  • 免费建站系统有哪些搜索引擎主要包括三个部分
  • asp做网站步骤百度推广价格表
  • 黄冈公司网站建设平台网络软文推广网站
  • 还有专门给别人做性奴的网站seo专业培训费用
  • 做网站必须租服务器吗百家号关键词排名优化
  • 阿里云网站建设一次付费太原seo优化
  • 看国外网站如何做科普企业培训内容包括哪些内容
  • 百度免费邮箱注册关键词seo排名优化
  • 昌乐网站建设网站需要改进的地方
  • 佛山市网站建设公司河南企业网站建设
  • 珠海做网站报价安装百度一下
  • 做穿越火线的网站资料百度官网网站
  • 做贷款行业哪些网站能发布广告推广引流吸引人的标题
  • 51制作工厂网站在线观看无需选择搜索引擎优化的定义是什么
  • 做淘口令的网站seo查询工具
  • 互联业务登录页 网站搜索引擎推广简称
  • 商务网站规划设计要点沈阳专业seo关键词优化
  • 做网站仓库报表系统昆明seo案例
  • 排名好的手机网站建设全国教育培训机构平台
  • 淄博做网站的网络营销的概念是什么
  • 商城网站建设分为几块厦门seo排名收费
  • 2008iis 网站 打不开总裁班课程培训
  • 网站备案 域名不是自己的东台网络推广
  • 企业做网站需要什么seo优化是怎么优化的