我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android下的POS打印机调用的简单实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android下的POS打印机调用的简单实现

本文基于GP58系列,它可以兼容ESC/POS指令集,对EPSON的打印机通用.

Android下的设备调试,如果设备提供了驱动,按照厂家的驱动调试即可;设备未提供驱动,只能按照通用的方法进行调试。这里采用的是调用USB接口来控制打印机输出。

1.首先获取USB管理器


public UsbAdmin(Context context) { 
    mUsbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE); 
    mPermissionIntent = PendingIntent.getBroadcast(context, 0, new Intent(ACTION_USB_PERMISSION), 0); 
    IntentFilter filter = new IntentFilter(ACTION_USB_PERMISSION); 
    context.registerReceiver(mUsbReceiver, filter); 
  }

使用一个延迟意图来接收usb接入时的广播,当广播接收到时,说明有新的设备接入。

添加一个boardcast action

代码如下:
private static final String ACTION_USB_PERMISSION = "com.android.example.USB_PERMISSION";


private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() { 
    public void onReceive(Context context, Intent intent) { 
      String action = intent.getAction(); 
      if (ACTION_USB_PERMISSION.equals(action)) { 
        synchronized (this) { 
          UsbDevice device = (UsbDevice) intent.getParcelableExtra(UsbManager.EXTRA_DEVICE); 
          if (intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false)) { 
            if (device != null) { 
              setDevice(device); 
            } else { 
              Closeusb(); 
             // mDevice = device; 
            } 
          } else { 
            Log.d(TAG, "permission denied for device " + device); 
          } 
        } 
      } 
    } 
  }; 

取到usb设备的引用,android系统会询问你是否允许设备访问,默认为false;当允许了访问之后,会判断USB的引用是否为null,如果不为空则会调用setDevice来创建一个Connection,否则会关闭本次连接。

在setDevice中,我们可以获取设备的功能集(UsbInterface),也可以获取通信通道(UsbEndpoint),同时也创建了host与device的连接用来传输数据。


private void setDevice(UsbDevice device) { 
    if (device != null) { 
      UsbInterface intf = null; 
      UsbEndpoint ep = null; 
      int InterfaceCount = device.getInterfaceCount(); 
      int j; 
      mDevice = device; 
      for (j = 0; j < InterfaceCount; j++) { 
        int i; 
        intf = device.getInterface(j); 
        Log.i(TAG, "接口是:" + j + "类是:" + intf.getInterfaceClass()); 
        if (intf.getInterfaceClass() == 7) { 
          int UsbEndpointCount = intf.getEndpointCount(); 
          for (i = 0; i < UsbEndpointCount; i++) { 
            ep = intf.getEndpoint(i); 
            Log.i(TAG, "端点是:" + i + "方向是:" + ep.getDirection() + "类型是:" + ep.getType()); 
            if (ep.getDirection() == 0 && ep.getType() == UsbConstants.USB_ENDPOINT_XFER_BULK) { 
              Log.i(TAG, "接口是:" + j + "端点是:" + i); 
              break; 
            } 
          } 
          if (i != UsbEndpointCount) { 
            break; 
          } 
        } 
      } 
      if (j == InterfaceCount) { 
        Log.i(TAG, "没有打印机接口"); 
        return; 
      } 
      mEndpointIntr = ep; 
        UsbDeviceConnection connection = mUsbManager.openDevice(device); 
        if (connection != null && connection.claimInterface(intf, true)) { 
          Log.i(TAG, "打开成功! "); 
          mConnection = connection; 
        } else { 
          Log.i(TAG, "打开失败! "); 
          mConnection = null; 
        } 
      } 
  } 

2.在相关的类中新建一个UsbAdmin,调用openUsb,这里首先是走了上面的setDevice()方法,获取到了设备的引用,当连接通道建立时列出所有USB设备,当设备的引用不存在时同样列出所有的USB设备,并且都请求获取USB权限。


public void openUsb() { 
    if (mDevice != null) { 
      setDevice(mDevice); 
      if (mConnection == null) { 
        HashMap<String, UsbDevice> deviceList = mUsbManager.getDeviceList(); 
        Iterator<UsbDevice> deviceIterator = deviceList.values().iterator(); 
        while (deviceIterator.hasNext()) { 
          UsbDevice device = deviceIterator.next(); 
          mUsbManager.requestPermission(device, mPermissionIntent); 
        } 
      } 
    } else { 
      HashMap<String, UsbDevice> deviceList = mUsbManager.getDeviceList(); 
      Iterator<UsbDevice> deviceIterator = deviceList.values().iterator(); 
      while (deviceIterator.hasNext()) { 
        UsbDevice device = deviceIterator.next(); 
        mUsbManager.requestPermission(device, mPermissionIntent); 
      } 
    } 
  } 

3.当上面两部都走完了之后,我们就可以发送指令来控制已经建立连接的打印机了,这里我们使用的是标准的ESC/POS指令集,为硬件默认,贴出代码,这里的指令集采用的是十进制表示形式,也可以替换成十六进制。


public class printerCmdUtils { 
   
  public static final byte ESC = 27;//换码 
  public static final byte FS = 28;//文本分隔符 
  public static final byte GS = 29;//组分隔符 
  public static final byte DLE = 16;//数据连接换码 
  public static final byte EOT = 4;//传输结束 
  public static final byte ENQ = 5;//询问字符 
  public static final byte SP = 32;//空格 
  public static final byte HT = 9;//横向列表 
  public static final byte LF = 10;//打印并换行(水平定位) 
  public static final byte CR = 13;//归位键 
  public static final byte FF = 12;//走纸控制(打印并回到标准模式(在页模式下) ) 
  public static final byte CAN = 24;//作废(页模式下取消打印数据 ) 
//------------------------打印机初始化----------------------------- 
   
  public static byte[] init_printer() 
  { 
    byte[] result = new byte[2]; 
    result[0] = ESC; 
    result[1] = 64; 
    return result; 
  } 
//------------------------换行----------------------------- 
   
  public static byte[] nextLine(int lineNum) 
  { 
      byte[] result = new byte[lineNum]; 
      for(int i=0;i<lineNum;i++) 
      { 
        result[i] = LF; 
      } 
      return result; 
  } 
//------------------------下划线-----------------------------   
   
  public static byte[] underlineWithOneDotWidthOn() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 45; 
    result[2] = 1; 
    return result; 
  } 
   
  public static byte[] underlineWithTwoDotWidthOn() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 45; 
    result[2] = 2; 
    return result; 
  } 
   
  public static byte[] underlineOff() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 45; 
    result[2] = 0; 
    return result; 
  } 
//------------------------加粗----------------------------- 
   
  public static byte[] boldOn() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 69; 
    result[2] = 0xF; 
    return result; 
  } 
   
  public static byte[] boldOff() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 69; 
    result[2] = 0; 
    return result; 
  } 
//------------------------对齐----------------------------- 
   
  public static byte[] alignLeft() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 97; 
    result[2] = 0; 
    return result; 
  } 
   
  public static byte[] alignCenter() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 97; 
    result[2] = 1; 
    return result; 
  } 
   
  public static byte[] alignRight() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 97; 
    result[2] = 2; 
    return result; 
  } 
   
  public static byte[] set_HT_position( byte col ) 
  { 
    byte[] result = new byte[4]; 
    result[0] = ESC; 
    result[1] = 68; 
    result[2] = col; 
    result[3] = 0; 
    return result; 
  } 
//------------------------字体变大----------------------------- 
   
  public static byte[] fontSizeSetBig(int num) 
  { 
      byte realSize = 0; 
      switch (num) 
      { 
      case 1: 
        realSize = 0;break; 
      case 2: 
        realSize = 17;break; 
      case 3: 
        realSize = 34;break; 
      case 4: 
        realSize = 51;break; 
      case 5: 
        realSize = 68;break; 
      case 6: 
        realSize = 85;break; 
      case 7: 
        realSize = 102;break; 
      case 8: 
        realSize = 119;break; 
      } 
      byte[] result = new byte[3]; 
      result[0] = 29; 
      result[1] = 33; 
      result[2] = realSize; 
      return result; 
  } 
//------------------------字体变小----------------------------- 
   
  public static byte[] fontSizeSetSmall(int num) 
  { 
      byte[] result = new byte[3]; 
      result[0] = ESC; 
      result[1] = 33; 
    return result; 
  } 
//------------------------切纸-----------------------------   
   
  public static byte[] feedPaperCutAll() 
  { 
      byte[] result = new byte[4]; 
     result[0] = GS; 
     result[1] = 86; 
     result[2] = 65; 
     result[3] = 0; 
     return result; 
  } 
   
  public static byte[] feedPaperCutPartial() 
  { 
      byte[] result = new byte[4]; 
     result[0] = GS; 
     result[1] = 86; 
     result[2] = 66; 
     result[3] = 0; 
     return result; 
  } 
//------------------------切纸----------------------------- 
  public static byte[] byteMerger(byte[] byte_1, byte[] byte_2){  
    byte[] byte_3 = new byte[byte_1.length+byte_2.length];  
    System.arraycopy(byte_1, 0, byte_3, 0, byte_1.length);  
    System.arraycopy(byte_2, 0, byte_3, byte_1.length, byte_2.length);  
    return byte_3;  
  }  
  public static byte[] byteMerger(byte[][] byteList){  
      int length = 0; 
    for(int i=0;i<byteList.length;i++) 
    { 
        length += byteList[i].length; 
    } 
    byte[] result = new byte[length]; 
    int index = 0; 
    for(int i=0;i<byteList.length;i++) 
    { 
        byte[] nowByte = byteList[i]; 
        for(int k=0;k<byteList[i].length;k++) 
        { 
          result[index] = nowByte[k]; 
          index++; 
        } 
    } 
    return result;  
  }  
} 

4.在以上都完成之后,就可以把你需要的字符串转换成byte数组并调用sendCommand方法来进行打印了


@SuppressLint("NewApi") 
  public boolean sendCommand(byte[] Content) { 
    boolean Result; 
    synchronized (this) { 
      int len = -1; 
      if (mConnection != null) { 
        len = mConnection.bulkTransfer(mEndpointIntr, Content, Content.length, 10000); 
      } 
      if (len < 0) { 
        Result = false; 
        Log.i(TAG, "发送失败! " + len); 
      } else { 
        Result = true; 
        Log.i(TAG, "发送" + len + "字节数据"); 
      } 
    } 
    return Result; 

代码如下:
len = mConnection.bulkTransfer(mEndpointIntr, Content, Content.length, 10000); 

这一步仅仅加了同步锁,并未开启一个新的线程去处理,在本机上没有问题,但上面的USB通信机制的文章有提到要放到异步线程,这里需要注意。

您可能感兴趣的文章:Android实现PDF预览打印功能Android gradle插件打印时间戳的方法详解Android编程实现计算两个日期之间天数并打印所有日期的方法Android中如何安全地打印日志详解Mac 下 Android Studio 不打印日志的解决办法Android jni调试打印char阵列的实例详解Android 蓝牙连接 ESC/POS 热敏打印机打印实例(ESC/POS指令篇)Android 蓝牙连接 ESC/POS 热敏打印机打印实例(蓝牙连接篇)Android打印机--小票打印格式及模板设置实例代码Android进阶——安卓调用ESC/POS打印机打印实例Android手机通过蓝牙连接佳博打印机的实例代码Android实现系统打印功能


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android下的POS打印机调用的简单实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android下的POS打印机调用的简单实现

本文基于GP58系列,它可以兼容ESC/POS指令集,对EPSON的打印机通用. Android下的设备调试,如果设备提供了驱动,按照厂家的驱动调试即可;设备未提供驱动,只能按照通用的方法进行调试。这里采用的是调用USB接口来控制打印机输出
2022-06-06

Android进阶——安卓调用ESC/POS打印机打印实例

前言前一段时间由于工作需要,要研究一下安卓程序调用打印机打印小票,并且要求不能使用蓝牙调用,研究了一下,可以利用socket连接,来实现打印功能。写了个Demo,分享一下。工具:一台打印机(芯烨XP-80XX),一台安卓测试机开发环境:An
2022-06-06

简单实用的Android studio 调试技巧

说到android studio的调试,很多人可能会说,这有什么可讲的不就是一个断点调试么,刚开始我也是这么认为的,直到我了解之后,才发现,调试原来可以玩的这么牛。下面我分别一一做介绍。条件断点(Conditional Breakpoint
2022-06-06

Android实现简单的下拉刷新pulltorefresh

网上下拉刷新的DEMO很多,但是总有各种不满意的地方,有些会下拉卡住,有些回弹不流畅,有些性能太低会各种卡顿,有些emptyView无法下拉...... 自己写的才是最合适自己的,代码很简单,也很容易修改,稍微阅读下代码就能改出自己需要的
2022-06-06

android动态壁纸调用的简单实例

调用后动态壁纸其实是显示在Activity的后面,而Activity则是透明显示,这样就可以看到下面的动态壁纸,如果Activity不是透明的则什么也看不到。 代码中有用到两个接口 IWallpaperService mService; I
2022-06-06

Android实现下载工具的简单代码

下载应该是每个App都必须的一项功能,不采用第三方框架的话,就需要我们自己去实现下载工具了。如果我们自己实现可以怎么做呢? 首先如果服务器文件支持断点续传,则我们需要实现的主要功能点如下: 多线程、断点续传下载 下载管理:开始、暂停、继续、
2022-06-06

Android简单实现屏幕下方Tab菜单的方法

本文实例讲述了Android简单实现屏幕下方Tab菜单的方法。分享给大家供大家参考,具体如下: 看到很多热门的Android程序(如:新浪微博、腾讯微博、京东商城、淘宝、当当等等)使用选项卡风格作为程序界面的主框架结构,而Android的选
2022-06-06

Android视频加水印之FFmpeg的简单应用实例

视频处理是Android开发中常见的需求。像是视频加水印,视频格式转换,视频截图等等…… FFmpeg是处理视频时常用到的工具,一般情况可以用FFmpeg源码自己编译,也可以使用网上别人编译好的包,这里我使用的是
2022-06-07

Python模块怎么实现简单的调用

这篇文章给大家分享的是有关Python模块怎么实现简单的调用的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。python可以做什么Python是一种编程语言,内置了许多有效的工具,Python几乎无所不能,该语言通
2023-06-14

Mybatis调用MySQL存储过程的简单实现

1.存储过程的简介我们常用的操作数据库语言SQL语句在执行的时候需要要先编译,然后执行,而存储过程(Stored Procedure)是一组为了完成特定功能的SQL语句集,经编译后存储在数据库中,用户通过指定存储过程的名字并给定参数(如果该
2023-05-31

Android中Glide实现超简单的图片下载功能

本文介绍了Glide实现超简单的图片下载功能,具体步骤如下:添加依赖compile 'com.github.bumptech.glide:glide:3.7.0' 添加权限2022-06-06

Android使用GridView实现日历的简单功能

简单的日历实现,只是显示了每一个月,没有显示当天和记事这些功能主要是计算月初是周几,月末是周几,然后相应的显示上一月多少天和下一月多少天。 先看一下关于日期的用到的几个工具类public static int ge
2022-06-06

JNI实现最简单的JAVA调用C/C++代码

JNI,是Java Native Interface的简称,中文是“Java本地调用”。通过这种技术可以做到以下两点: Java程序中的函数可以调用Native语言写的函数,Native一般指的是C/C++编写的函数。 Native程序
2023-05-31

Android数据库(SQLite)的简单使用——增、删、查改功能的简单实现

记录一下Android数据库的增删查改的简单使用 话不多说,先献上你们最爱的效果图~ 这边我用的是一个ListView来展示数据库里的数据准备工作: 先写一个类继承SQLiteOpenHelper,因为SQLiteOpenHelper.ja
2022-06-06

android中AutoCompleteTextView的简单用法(实现搜索历史)

网上有不少教程,那个提示框字符集都是事先写好的,例如用一个String[] 数组去包含了这些数据,但是,我们也可以吧用户输入的作为历史记录保存下面先上我写的代码:import android.app.Activity;import andr
2022-06-06

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录