Android实现打开本地pdf文件的两种方式

2025-04-14 16:50

本文主要是介绍Android实现打开本地pdf文件的两种方式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Android实现打开本地pdf文件的两种方式》在现代应用中,PDF格式因其跨平台、稳定性好、展示内容一致等特点,在Android平台上,如何高效地打开本地PDF文件,不仅关系到用户体验,也直接影响...

一、项目概述

在现代应用中,PDF格式因其跨平台、稳定性好、展示内容一致等特点,广泛应用于电子书、报表、合同、资料阅读等场景。在Android平台上,如何高效地打开本地PDF文件,不仅关系到用户体验,也直接影响到应用的功能丰富度。在实际项目中,实现打开本地PDF文件主要有以下两种方式:

  1. 外部调用方式
    利用Intent调用系统已安装的PDF阅读器程序,将本地PDF文件传递给该程序进行展示。这种方式开发成本低、实现简单,缺点在于需要依赖第三方PDF阅读器,且用户体验会因不同阅读器而异。

  2. 内嵌显示方式
    集成第三方PDF阅读库(如AndroidPdfViewer、PdfRenderer、MuPDF等),在自己的应用中直接解析并展示PDF文件。这种方式能够给用户提供统一、定制化的阅读体验,但需要考虑库的兼容性、性能、内存占用等问题。

本项目旨在通过详细介绍上述两种实现方式的原理、优缺点、实现步骤及具体代码示例,帮助开发者全面掌握在Android平台上打开本地PDF文件的关键技术。文章内容主要分为以下部分:

  1. 项目背景与应用场景

  2. 相关知识介绍

  3. 实现思路与方案对比

  4. 代码整合与详细注释

  5. 代码解读与关键技术解析

  6. 项目总结与扩展讨论

  7. 实践建议与未来展望

通过对项目整体架构、关键模块以及细节处理的介绍,本文不仅能够帮助初学者迅速上手实现打开本地PDF文件的功能,也能为有经验的开发者提供参考,便于在实际项目中进行二次开发和技术扩展。

二、相关知识

在实现打开本地PDF文件之前,需要掌握以下几方面的相关知识:

2.1 PDF文件基本概述

  • PDF格式简介
    PDF(Portable Document Format)是一种便携式文档格式,由Adobe公司于1993年推出,其特点是文档内容与格式固定,能在不同设备上正确显示。常用于电子书、报表、说明书、合同、发票等场景。

  • PDF阅读原理
    PDF文件的内部结构包括文本、图像、矢量图形、表单、附件等,解析PDF文件需要实现页面的渲染、文本和图像提取、矢量图形绘制等功能。Android平台上,可以利用系统自带的PdfRenderer(API21及以上)或第三方库来实现PDF渲染。

2.2 Android 文件访问与存储权限

  • 文件读取权限访问本地PDF文件通常需要读取外部存储,因此需要在AndroidManifest.xml中声明相关权限,例如

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
  • 同时在Android 6.0(API 23)及以上版本下还需要在运行时动态申请权限。

  • 文件路径与FileProvider
    为了安全地访问文件,尤其在Android 7.0(API 24)及以后版本,为了解决File URI暴露的问题,需要使用FileProvider来生成content:// URI,将本地PDF文件的路径转换为可供其他应用访问的URI。

2.3 调用系统PDF阅读器

  • Intent调用机制
    Android利用Intent机制允许应用间通信。当需要打开PDF文件时,可以构造一个ACTION_VIEW类型的Intent,将文件的URI和MIME类型(通常为"application/pdf")传递给系统,系统会自动选择一个已安装的PDF阅读器来处理。

  • 常见问题
    需要注意的是:

    • 如果系统中没有安装PDF阅读器,Intent调用可能会失败,此时需要进行异常捕获并给予用户提示;

    • 文件URI传递需要借助FileProvider,否则在Android 7.0以上版本会出现FileUriExposedException。

2.4 第三方PDF阅读库

  • AndroidPdfViewer
    目前较为流行的一款开源PDF阅读库,基于PdfiumAndroid实现PDF渲染,具有良好的性能和交互体验。集成方法简单,只需要在Gradle中添加依赖,然后在布局中引用PDFView控件即可进行渲染和翻页处理。

  • PdfRenderer
    Android系统从API21开始提供PdfRenderer类,可直接用来渲染PDF页面,但功能较为基础,支持旋转、缩放和页面切换,但交互体验需要开发者自行扩展。

  • 其他开源库
    除了以上两种,还有MuPDF、PdfBox-Android等库,不同库在性能、体积、功能扩展方面各有侧重,开发者可根据实际项目需求进行选择。

2.5 动态权限与FileProvider配置

  • 动态权限
    在Android 6.0及以上版本,读取外部存储权限属于危险权限,需在应用运行时请求用户授权。需要使用ActivityCompat.requestPermissions()方法请求权限,并在onRequestPermissionsResult()方法中接收用户的授权结果。

  • FileProvider配置
    为了使用FileProvider,需在AndroidManifest.xml中声明FileProvider,以及在res/xml目录下创建file_paths.xml文件配置允许访问的目录,从而安全地共享文件URI。

三、项目实现思路

本项目实现打开本地PDF文件主要提供两种实现方案,具体思路如下:

3.1 外部调用方式:利用Intent打开PDF

  1. 权限申请
    在AndroidManifest.xml中声明READ_EXTERNAL_STORAGE权限,并在Activity中动态请求权限。

  2. 文件访问与URI转换
    获取本地PDF文件的路径(可以保存在SD卡或内部存储中),使用FileProvider将File对象转换为content:// URI。

  3. 构造Intent
    创建一个ACTION_VIEW类型的Intent,将生成的URI及MIME类型"application/pdf"传递给Intent,并设置FLAG_GRANT_READ_URI_PERMISSION标志。

  4. 启动Activity
    调用startActivity(intent)启动系统PDF阅读器。若系统中未安装PDF阅读器,则捕获ActivityNotFoundException并提示用户下载安装PDF阅读器。

  5. 异常处理
    对于文件不存在、权限未授权、文件URI非法等情况,给予用户相应提示,确保应用稳定性。

3.2 内嵌显示方式:利用第三方PDF阅读库

  1. 集成第三方库
    在Gradle中添加例如AndroidPdfViewer依赖(如:"com.github.barteksc:android-pdf-viewerandroid:3.2.0-beta.1"),保证版本兼容当前项目要求。

  2. 设计布局
    在布局文件中添加PDFView控件,用于显示PDF文档。

  3. 读取PDF文件
    在Activity或Fragment中,通过文件路径或FileProvider获取PDF文件的InputStream,并将其传递给PDFView控件进行解析和渲染。

  4. 交互设计
    第三方库一般内置手势操作(翻页、缩放、拖动等),开发者可根据需求配置手势控制及显示选项,例如是否显示目录、书签、页码等。

  5. 错误处理与性能优化
    考虑大文件加载、页面缓存、内存溢www.chinasem.cn出等问题,并根据库的文档进行优化设置,同时在出现异常时给予用户错误提示。

3.3 对比与选择

  • 外部调用方式
    优点:实现简单,不需要在APP内维护PDF渲染逻辑,依赖系统或第三方阅读器;缺点:用户体验不统一,依赖外部应用。

  • 内嵌显示方式
    优点:可以定制统一的用户界面与交互体验,对PDF的控制更加灵活,支持更多自定义功能;缺点:集成成本较高,需关注性能、兼容性及库的维护更新。

在实际项目中,可根据项目定位与用户需求选择适合的方案;若仅为临时预览PDF文件,可优先考虑Intent方式;如应用重点为PDF阅读体验,则建议集成第三方PDF阅读库。

四、整合代码

下面提供两套完整示例代码,分别对应外部调用方式与内嵌显示方式,并附有详细注释说明。

4.1 外部调用方式

4.1.1 AndroidManifest.xml 配置

在Manifest中声明权限,并配置FileProvider(注意,需同时创建file_paths.xml)。

<!-- AndroidManifest.xml -->
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.pdfopener">
 
    <!-- 申请读取外部存储权限 -->
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
 
    <application
        android:allowBackup="true"
        android:label="PDFOpener"
        android:icon="@mipmap/ic_launcher"
        android:roundIcon="@mipmap/ic_launcher_round"
        编程android:theme="@style/AppTheme">
        
        <!-- 主Activity -->
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/>
                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>
        </activity>
        
        <!-- 配置FileProvider -->
        <provider
            android:name="androidx.core.content.FileProvider"
            android:authorities="${applicationId}.fileprovider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_paths"/>
        </provider>
        
    </application>
</manifest>

4.1.2 file_paths.xml 文件(放在 res/xml/ 目录下)

<!-- res/xml/file_paths.xml -->
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <!-- 允许访问外部存储中指定目录下的所有文件 -->
    <external-path name="external_files" path="."/>
</paths>

4.1.3 MainActivity.java(调用外部PDF阅读器)

package com.example.pdfopener;
 
import android.Manifest;
import android.content.ActivityNotFoundException;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.widget.Button;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.FileProvider;
import java.io.File;
 
/**
 * MainActivity
 *
 * 该Activity展示如何利用Intent从本地打开PDF文件,
 * 使用FileProvider获取content:// URI,并调用系统PDF阅读器。
 */
public class MainActivity extends AppCompatActivity {
 
    private static final int REQUEST_READ_STORAGE = 100;
    private Button btnOpenPDF;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main_pdf);
 
        btnOpenPDF = findViewById(R.id.btn_open_pdf);
 
        // 检查READ_EXTERNAL_STORAGE权限
        if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)
                != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
                    REQUEST_READ_STORAGE);
        }
 
        btnOpenPDF.setOnClickListener(v -> {
            // 此处假设PDF文件位于外部存储的Download目录下,文件名为sample.pdf
            File pdfFile = new File(Environment.getExternalStoragePublicDirectory(
                    Environment.DIRECTORY_DOWNLOADS), "sample.pdf");
 
            if (!pdfFile.exists()) {
                Toast.makeText(MainActivity.this, "PDF文件不存在", Toast.LENGTH_SHORT).show();
                return;
            }
 
            // 通过FileProvider获取content URI
            Uri pdfUri = FileProvider.getUriForFile(MainActivity.this,
                    getApplicationContext().getPackageName() + ".fileprovider", pdfFile);
 
            // 创建Intent,设置类型为application/pdf
            Intent intent = new Intent(Intent.ACTION_VIEW);
            intent.setDataAndType(pdfUri, "application/pdf");
            // 授权临时读取权限
            intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
 
            try {
                startActivity(intent);
            } catch (ActivityNotFoundException e) {
                Toast.makeText(MainActivity.this, "未找到可以打开PDF的应用,请安装PDF阅读器", Toast.LENGTH_LONG).show();
            }
        });
    }
 
    /**
     * 动态请求权限回调
     */
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,
                                           @NonNull int[] grantResults) {
        if (requestCode == REQUEST_READ_STORAGE) {
            if (!(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED)) {
                Toast.makeText(this, "需要授予存储读取权限", Toast.LENGTH_LONG).show();
            }
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }
}

4.1.4 activity_main_pdf.xml 布局文件

<!-- res/layout/activity_main_pdf.xml -->
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/main_container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:gravity="center"
    android:padding="16dp">
 
    <Button
        android:id="@+id/btn_open_pdf"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="打开本地PDF文件" />
</LinearLayout>

4.2 内嵌显示方式:集成AndroidPdfViewer库

下面介绍如何在应用中内嵌显示PDF文件,使用开源库AndroidPdfViewer来实现。该库依赖于PdfiumAndroid实现高效PDF解析与展示。

4.2.1 Gradle依赖配置

在项目的app模块build.gradle中添加如下依赖:

dependencies {
    // 其他依赖...
    implementation 'com.github.barteksc:android-pdf-viewer:3.2.0-beta.1'
}

同时在项目根目录的build.gradle中确保加入jitpack仓库:

allprojects {
    repositories {
        google()
        jcenter()
        maven { url 'https://jitpack.io' }
    }
}

4.2.2 布局文件(activity_pdf_viewer.xml)

在布局中添加PDFView控件:

<!-- res/layout/activity_pdf_viewer.xml -->
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/pdf_container"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
 
    <com.github.barteksc.pdfviewer.PDFView
        android:id="@+id/pdfView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</RelativeLayout>

4.2.3 PDFViewerActivity.java

创建一个Activity来加载并展示PDF文件。此示例从本地存储加载PDF文件,类似于外部调用方式,也可结合FileProvider安全读取文件。

package com.example.pdfopener;
 
import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.os.Environment;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import com.github.barteksc.pdfviewer.PDFView;
import java.io.File;
 http://www.chinasem.cn
/**
 * PDFViewerActivity
 *
 * 该Activity展示如何利用AndroidPdphpfViewer库在应用内部加载和展示本地PDF文件。
 */
public class PDFViewerActivity extends AppCompatActivity {
 
    private static final int REQUEST_READ_STORAGE = 101;
    private PDFView pdfView;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_pdf_viewer);
 
        pdfView = findViewById(R.id.pdfView);
 
        if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)
                != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
                    REQUEST_READ_STORAGE);
        } else {
            loadPdf();
        }
    }
 
    private void loadPdf() {
        // 假设PDF文件位于Download目录下,文件名为sample.pdf
        File pdfFile = new File(Environment.getExternalStoragePublicDirectory(
                Environment.DIRECTORY_DOWNLOADS), "sample.pdf");
 
        if (!pdfFile.exists()) {
            Toast.makeText(PDFViewerActivity.this, "PDF文件不存在", Toast.LENGTH_SHORT).show();
            return;
        }
 
        // 使用PDFView加载PDF文件
        pdfView.fromFile(pdfFile)
                .enableSwipe(true)          // 支持滑动翻页
                .swipeHorizontal(false)     // false为竖直滑动
                .enableDoubletap(true)      // 支持双击缩放
                .defaultPage(0)             // 默认第一页
                .load();
    }
 
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,
                                           @NonNull int[] grantResults) {
        if (requestCode == REQUEST_READ_STORAGE) {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                loadPdf();
            } else {
                Toast.makeText(this, "需要读取存储权限以打开PDF文件", Toast.LENGTH_LONG).show();
            }
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }
}

并在对应布局中添加此按钮。

五、代码解读

下面详细解读以上代码中的核心实现逻辑及关键点。

5.1 外部调用方式

  • 权限申请
    在MainActivity中首先检查并动态申请READ_EXTERNAL_STORAGE权限,确保能够访问外部存储文件。

  • FileProvider配置
    通过在Manifest中配置FileProvider和在res/xml/file_paths.xml中定义允许访问目录,保证将本地文件路径转换为安全的content URI,从而在Android 7.0及以上版本避免FileUriExposedException。

  • Intent构造与启动
    构造ACTION_VIEW的Intent时,通过setDataAndType()设置文件的URI和MIME类型为"application/pdf",并设置FLAG_GRANT_READ_URI_PERMISSION标志,确保目标应用具有读取文件的权限;调用startActivity()启动系统PDF阅读器。

  • 异常捕捉
    若系统中没有安装可打开PDF文件的应用,则捕获ActivityNotFoundException,并提示用户下载安装PDF阅读器。

5.2 内嵌显示方式

  • 第三方库集成
    利用Gradle引入AndroidPdfViewer库,参考其文档通过PDFView控件加载PDF文件。使用pdfView.fromFile(file).load()方法实现文件读取与页面渲染。

  • 交互体验配置
    PDFView控件内置多种手势操作,支持翻页、缩放、双击等。开发者可通过配置enableSwipe()、swipeHorizontal()、enableDoubletap()等方法调整交互效果,提升用户阅读体验。

  • 错误处理
    同样需要检查文件是否存在,若不存在则提示用户,并处理动态权限申请,确保应用在各版本系统上均能正常运行。

5.3 权限与兼容性

  • 动态权限处理
    结合ActivityCompat.requestPermissions()实现动态权限申请,对于没有被授权的情况给予提示。

  • FileProvider使用
    防止直接使用file://URI在新版本系统中导致异常,通过FileProvider转换为content://URI,确保数据安全传递。

5.4 两种方案优缺点

  • 外部调用方式
    优点:实现简单、依赖系统应用,无需引入庞大库;缺点:用户体验受限于其他PDF阅读器,对交互与界面定制能力不足。

  • 内嵌显示方式
    优点:可自定义界面与操作,整合于应用内部,用户体验统一;缺点:需要集成第三方库,可能引入依赖和性能调优问题。

六、项目总结

本项目详细介绍了在Android平台上实现打开本地PDF文件的两种常见方法及关键技术:

  1. 外部调用方式
    利用Intent与FileProvider调用系统已安装的PDF阅读器打开本地PDF文件,开发流程简单、实现成本低,但用户体验依赖于第三方应用。

  2. 内嵌显示方式
    集成第三方PDF阅读库(如AndroidPdfViewer),在应用内直接展示PDF文件,提供定制化的阅读体验,适合对PDF阅读有较高要求的项目。

通过对相关知识(如PDF基本概念、文件存储与权限管理、Intent调用机制、FileProvider配置等)的介绍,再结合详细的实现思路与逐步代码解析,本项目为开发者提供了一整套完整的实现方案。代码部分附有详尽注释,帮助读者逐行了解各个模块的功能和实现细节。

6.1 技术亮点

  • 动态权限申请与FileProvider配置,确保在最新Android版本下安全访问本地文件。

  • 利用Intent机制调用系统应用,展示最简便的文件打开方式。

  • 集成AndroidPdfViewer库,实现内嵌PDF阅读,提供良好的用户交互和定制化扩展能力。

6.2 应用场景

  • 企业内部文档查看工具、电子书阅读器、合同管理系统等需要展示PDF文件的应用。

  • 需要对用户阅读体验进行统一定制和控制的场景,建议采用内嵌显示方式。

  • 简单预览或临时文件查看时,可以快速使用Intent方式调用系统阅读器。

6.3 开发建议

  • 在开发前务必检查目标设备上的文件路径和权限,避免因权限问题引起的应用崩溃。

  • 考虑多种错误处理情况,设计友好的用户提示信息,如文件不存在、权限未授权或没有PDF阅读器安装等。

  • 对于内嵌显示方式,注意第三方库的版本更新与兼容性测试,确保在不同Android版本与设备上均能流畅运行。

  • 根据项目实际需要选择适合的方案,并在用户体验和开发成本间做出平衡。

七、实践建议与未来展望

在未来的开发中,可以从以下几方面进一步扩展和优化该功能:

  1. 高级交互功能

    • 增加目录导航、书签管理、全文搜索等功能,提升PDF阅读体验。

    • 支持夜间模式、字体调整、页面旋转等个性化功能,使阅读更加便捷舒适。

  2. 性能优化

    • 优化大文件的加载速度与内存管理,采用分页加载或缓存策略。

    • 对PdfRenderer等系统API进行封装,适配低于API21的设备,提升兼容性。

  3. 数据保护与安全

    • 在读取敏感文档时增加加密解密机制,保护用户隐私数据不被非法泄露。

    • 使用自定义FileProvider配置更精细的访问控制,确保应用文件安全共享。

  4. 与在线服务整合

    • 支持从服务器获取PDF文件并缓存到本地,自动检测文档更新。

    • 建立云端文档管理系统,实现跨设备同步和在线批注功能。

  5. 多平台扩展

    • 将部分功能模块封装为独立库,在其他平台(如桌面系统、IOS)中实现类似功能。

    • 探索利用Jetpack Compose实现声明式PDF阅读界面,结合最新的Android开发趋势。

八、结语

本文详细介绍了在Android平台上实现打开本地PDF文件的两种主要方法:一种是通过Intent调用系统PDF阅读器打开PDF文件,另一种是集成第三方PDF阅读库在应用内部展示PDF文档。全文内容从项目概述、相关理论、实现思路,到完整代码、注释解读、实践建议和未来展望,全面深入地解析了整个实现过程。

无论您是初学者还是有经验的开发者,通过本文都能系统了解如何在Android中处理文件权限、使用FileProvider、构造Intent以及集成第三方库从而实现PDF文件的阅读功能。希望本文能为您的博客撰写和项目开发提供充分的参考与帮助,在实际工作中不断优化用户体验,实现更高质量、更专业的应用。

以上就是Android实现打开本地pdf文件的两种方式的详细内容,更多关于Android打开本地pdf文件的资料请关注China编程(www.chinasem.cn)其它相关文章!

这篇关于Android实现打开本地pdf文件的两种方式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1154212

相关文章

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

OpenCV图像形态学的实现

《OpenCV图像形态学的实现》本文主要介绍了OpenCV图像形态学的实现,包括腐蚀、膨胀、开运算、闭运算、梯度运算、顶帽运算和黑帽运算,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起... 目录一、图像形态学简介二、腐蚀(Erosion)1. 原理2. OpenCV 实现三、膨胀China编程(

通过Spring层面进行事务回滚的实现

《通过Spring层面进行事务回滚的实现》本文主要介绍了通过Spring层面进行事务回滚的实现,包括声明式事务和编程式事务,具有一定的参考价值,感兴趣的可以了解一下... 目录声明式事务回滚:1. 基础注解配置2. 指定回滚异常类型3. ​不回滚特殊场景编程式事务回滚:1. ​使用 TransactionT

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Spring Shell 命令行实现交互式Shell应用开发

《SpringShell命令行实现交互式Shell应用开发》本文主要介绍了SpringShell命令行实现交互式Shell应用开发,能够帮助开发者快速构建功能丰富的命令行应用程序,具有一定的参考价... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定义S

SpringBatch数据写入实现

《SpringBatch数据写入实现》SpringBatch通过ItemWriter接口及其丰富的实现,提供了强大的数据写入能力,本文主要介绍了SpringBatch数据写入实现,具有一定的参考价值,... 目录python引言一、ItemWriter核心概念二、数据库写入实现三、文件写入实现四、多目标写入

Android Studio 配置国内镜像源的实现步骤

《AndroidStudio配置国内镜像源的实现步骤》本文主要介绍了AndroidStudio配置国内镜像源的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、修改 hosts,解决 SDK 下载失败的问题二、修改 gradle 地址,解决 gradle

SpringSecurity JWT基于令牌的无状态认证实现

《SpringSecurityJWT基于令牌的无状态认证实现》SpringSecurity中实现基于JWT的无状态认证是一种常见的做法,本文就来介绍一下SpringSecurityJWT基于令牌的无... 目录引言一、JWT基本原理与结构二、Spring Security JWT依赖配置三、JWT令牌生成与

SpringBoot实现微信小程序支付功能

《SpringBoot实现微信小程序支付功能》小程序支付功能已成为众多应用的核心需求之一,本文主要介绍了SpringBoot实现微信小程序支付功能,文中通过示例代码介绍的非常详细,对大家的学习或者工作... 目录一、引言二、准备工作(一)微信支付商户平台配置(二)Spring Boot项目搭建(三)配置文件

Spring中配置ContextLoaderListener方式

《Spring中配置ContextLoaderListener方式》:本文主要介绍Spring中配置ContextLoaderListener方式,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录Spring中配置ContextLoaderLishttp://www.chinasem.cntene