热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android手机通过蓝牙连接佳博打印机的实例代码

这篇文章主要介绍了Android手机通过蓝牙连接佳博打印机的实例代码,非常不错具有参考借鉴价值,需要的朋友可以参考下

所使用的打印机为佳博打印机,支持蓝牙、wifi、usb我所使用的是通过蓝牙来连接。

在网上找到一个佳博官方针对安卓开发的App源码,但是各种的跳转,没有看太懂,所以又去问度娘,找到了一个不错的文章

Android对于蓝牙开发从2.0版本的sdk才开始支持,而且模拟器不支持,测试至少需要两部手机,所以制约了很多技术人员的开发。

1. 首先,要操作蓝牙,先要在AndroidManifest.xml里加入权限

// 管理蓝牙设备的权限 

// 使用蓝牙设备的权限 

2.打开蓝牙

获得蓝牙适配器(android.bluetooth.BluetoothAdapter),检查该设备是否支持蓝牙,如果支持,就打开蓝牙。

// 检查设备是否支持蓝牙 
adapter = BluetoothAdapter.getDefaultAdapter(); 
if (adapter == null) 
{ 
// 设备不支持蓝牙 
} 
// 打开蓝牙 
if (!adapter.isEnabled()) 
{ 
Intent intent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE); 
// 设置蓝牙可见性,最多300秒 
intent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION,300);
context.startActivity(intent); 
}

3.获取已配对的蓝牙设备(android.bluetooth.BluetoothDevice)

首次连接某蓝牙设备需要先配对,一旦配对成功,该设备的信息会被保存,以后连接时无需再配对,所以已配对的设备不一定是能连接的。

BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter(); 
Set devices = adapter.getBondedDevices(); 
for(int i=0; i

4.搜索周围的蓝牙设备

适配器搜索蓝牙设备后将结果以广播形式传出去,所以需要自定义一个继承广播的类,在onReceive方法中获得并处理蓝牙设备的搜索结果。

// 设置广播信息过滤 
IntentFilter intentFilter = new IntentFilter(); 
intentFilter.addAction(BluetoothDevice.ACTION_FOUND); 
intentFilter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED); 
intentFilter.addAction(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED); 
intentFilter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED); 
// 注册广播接收器,接收并处理搜索结果 
context.registerReceiver(receiver, intentFilter); 
// 寻找蓝牙设备,android会将查找到的设备以广播形式发出去 
adapter.startDiscovery(); 
自定义广播类
private BroadcastReceiver receiver = new BroadcastReceiver() { 
@Override 
public void onReceive(Context context, Intent intent) { 
String action = intent.getAction(); 
if (BluetoothDevice.ACTION_FOUND.equals(action)) { 
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); 
System.out.println(device.getName()); 
} 
} 
}

5.蓝牙设备的配对和状态监视

private BroadcastReceiver receiver = new BroadcastReceiver() { 
@Override 
public void onReceive(Context context, Intent intent) { 
String action = intent.getAction(); 
if (BluetoothDevice.ACTION_FOUND.equals(action)) { 
// 获取查找到的蓝牙设备 
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); 
System.out.println(device.getName()); 
// 如果查找到的设备符合要连接的设备,处理 
if (device.getName().equalsIgnoreCase(name)) { 
// 搜索蓝牙设备的过程占用资源比较多,一旦找到需要连接的设备后需要及时关闭搜索 
adapter.cancelDiscovery(); 
// 获取蓝牙设备的连接状态 
cOnnectState= device.getBondState(); 
switch (connectState) { 
// 未配对 
case BluetoothDevice.BOND_NONE: 
// 配对 
try { 
Method createBOndMethod= BluetoothDevice.class.getMethod("createBond"); 
createBondMethod.invoke(device); 
} catch (Exception e) { 
e.printStackTrace(); 
} 
break; 
// 已配对 
case BluetoothDevice.BOND_BONDED: 
try { 
// 连接 
connect(device); 
} catch (IOException e) { 
e.printStackTrace(); 
} 
break; 
} 
} 
} else if(BluetoothDevice.ACTION_BOND_STATE_CHANGED.equals(action)) { 
// 状态改变的广播 
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); 
if (device.getName().equalsIgnoreCase(name)) { 
cOnnectState= device.getBondState(); 
switch (connectState) { 
case BluetoothDevice.BOND_NONE: 
break; 
case BluetoothDevice.BOND_BONDING: 
break; 
case BluetoothDevice.BOND_BONDED: 
try { 
// 连接 
connect(device); 
} catch (IOException e) { 
e.printStackTrace(); 
} 
break; 
} 
} 
} 
} 
}

6.蓝牙设备的连接

private void connect(BluetoothDevice device) throws IOException { 
// 固定的UUID 
final String SPP_UUID = "00001101-0000-1000-8000-00805F9B34FB"; 
UUID uuid = UUID.fromString(SPP_UUID); 
BluetoothSocket socket = device.createRfcommSocketToServiceRecord(uuid); 
socket.connect(); 
} 
private void connect(BluetoothDevice device) throws IOException { 
// 固定的UUID 
final String SPP_UUID = "00001101-0000-1000-8000-00805F9B34FB"; 
UUID uuid = UUID.fromString(SPP_UUID); 
BluetoothSocket socket = device.createRfcommSocketToServiceRecord(uuid); 
socket.connect(); 
}

1.BluetoothAdapter 顾名思义,蓝牙适配器,直到我们建立bluetoothSocket连接之前,都要不断操作它

BluetoothAdapter里的方法很多,常用的有以下几个:

cancelDiscovery() 根据字面意思,是取消发现,也就是说当我们正在搜索设备的时候调用这个方法将不再继续搜索

disable()关闭蓝牙

enable()打开蓝牙,这个方法打开蓝牙不会弹出提示,更多的时候我们需要问下用户是否打开,一下这两行代码同样是打开蓝牙,不过会提示用户:

Intemtenabler=new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);

startActivityForResult(enabler,reCode);//同startActivity(enabler);

getAddress()获取本地蓝牙地址

getDefaultAdapter()获取默认BluetoothAdapter,实际上,也只有这一种方法获取BluetoothAdapter

getName()获取本地蓝牙名称

getRemoteDevice(String address)根据蓝牙地址获取远程蓝牙设备

getState()获取本地蓝牙适配器当前状态(感觉可能调试的时候更需要)

isDiscovering()判断当前是否正在查找设备,是返回true

isEnabled()判断蓝牙是否打开,已打开返回true,否则,返回false

listenUsingRfcommWithServiceRecord(String name,UUID uuid)根据名称,UUID创建并返回BluetoothServerSocket,这是创建BluetoothSocket服务器端的第一步

startDiscovery()开始搜索,这是搜索的第一步

2.BluetoothDevice看名字就知道,这个类描述了一个蓝牙设备

createRfcommSocketToServiceRecord(UUIDuuid)根据UUID创建并返回一个BluetoothSocket

这个方法也是我们获取BluetoothDevice的目的——创建BluetoothSocket

这个类其他的方法,如getAddress(),getName(),同BluetoothAdapter

3.BluetoothServerSocket如果去除了Bluetooth相信大家一定再熟悉不过了,既然是Socket,方法就应该都差不多,

这个类一种只有三个方法

两个重载的accept(),accept(inttimeout)两者的区别在于后面的方法指定了过时时间,需要注意的是,执行这两个方法的时候,直到接收到了客户端的请求(或是过期之后),都会阻塞线程,应该放在新线程里运行!

还有一点需要注意的是,这两个方法都返回一个BluetoothSocket,最后的连接也是服务器端与客户端的两个BluetoothSocket的连接

close()这个就不用说了吧,翻译一下——关闭!

4.BluetoothSocket,跟BluetoothServerSocket相对,是客户端

一共5个方法,不出意外,都会用到

close(),关闭

connect()连接

getInptuStream()获取输入流

getOutputStream()获取输出流

getRemoteDevice()获取远程设备,这里指的是获取bluetoothSocket指定连接的那个远程蓝牙设备

根据这篇文章只是更改了最后的蓝牙连接部分(因为佳博打印机有自己的SDK,SDK中有自己的连接方法)将下载的DEMO中的gprintersdkv22.jar和xUtils-2.6.14.jar两个包导入,之后的连接方法如下:

/**
*连接
*/
private voidconnect() {
intrel =0;
try{//使用端口1,4代表模式为蓝牙模式,蓝牙地址,最后默认为0
rel = mGpService.openPort(1,4,adressData.get(loction),0);
}catch(RemoteException e) {
e.printStackTrace();
}
GpCom.ERROR_CODE r = GpCom.ERROR_CODE.values()[rel];
if(r != GpCom.ERROR_CODE.SUCCESS) {
if(r == GpCom.ERROR_CODE.DEVICE_ALREADY_OPEN) {
//开启成功
}else{
UIUtils.showMessage(GpCom.getErrorText(r));
}
}else{
ProgressDialogUtil.dismiss(BuleToothActivity.this);
UIUtils.showMessage("失败");
}
}

不要忘记的就是mGpService是佳博SDK的一个自己的服务,要先获取,并与Activity进行绑定;

protected voidonCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_buletooth);
connection();
}
private voidconnection() {
cOnn=newPrinterServiceConnection();
Intent intent =newIntent(this, GpPrintService.class);
this.bindService(intent,conn, Context.BIND_AUTO_CREATE);// bindService
}
classPrinterServiceConnectionimplementsServiceConnection {
@Override
public void onServiceDisconnected(ComponentName name) {
mGpService=null;
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
mGpService= GpService.Stub.asInterface(service);
}
}

还要在清单文件中去配置服务







做完这些之后就大功告成,可以连接到打印机了,不过还是不能打印,还要进行一些打印的配置,由于我的项目的原因,连接和打印并没有在同一个界面,所以继续去研究佳博源码,发现如果前面链接已经成功的话,在另一个界面只需就行一个配置就可一直接打印打印也代码如下:

public classPrintActivityextendsBaseActivityimplementsView.OnClickListener{
@Bind(R.id.print_print)
Buttonprint;
privateGpServicemGpService=null;
privatePrinterServiceCOnnectionconn=null;
private static final int MAIN_QUERY_PRINTER_STATUS=0xfe;
private static final int REQUEST_PRINT_LABEL=0xfd;
private static final int REQUEST_PRINT_RECEIPT=0xfc;
private int mTotalCopies=0;
@Override
protected void initHandler() {
handler=newHandler() {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
}
};
}
privateBroadcastReceivermBroadcastReceiver=newBroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
Log.d("TAG", action);
// GpCom.ACTION_DEVICE_REAL_STATUS 为广播的IntentFilter
if(action.equals(GpCom.ACTION_DEVICE_REAL_STATUS)) {
//业务逻辑的请求码,对应哪里查询做什么操作
intrequestCode = intent.getIntExtra(GpCom.EXTRA_PRINTER_REQUEST_CODE, -1);
//判断请求码,是则进行业务操作
if(requestCode ==MAIN_QUERY_PRINTER_STATUS) {
intstatus = intent.getIntExtra(GpCom.EXTRA_PRINTER_REAL_STATUS,16);
String str;
if(status == GpCom.STATE_NO_ERR) {
str ="打印机正常";
}else{
str ="打印机 ";
if((byte) (status & GpCom.STATE_OFFLINE) > 0) {
str +="脱机";
}
if((byte) (status & GpCom.STATE_PAPER_ERR) > 0) {
str +="缺纸";
}
if((byte) (status & GpCom.STATE_COVER_OPEN) > 0) {
str +="打印机开盖";
}
if((byte) (status & GpCom.STATE_ERR_OCCURS) > 0) {
str +="打印机出错";
}
if((byte) (status & GpCom.STATE_TIMES_OUT) > 0) {
str +="查询超时";
}
}
Toast.makeText(getApplicationContext(),"打印机:"+1+"状态:"+ str, Toast.LENGTH_SHORT)
.show();
}else if(requestCode ==REQUEST_PRINT_RECEIPT) {
intstatus = intent.getIntExtra(GpCom.EXTRA_PRINTER_REAL_STATUS,16);
if(status == GpCom.STATE_NO_ERR) {
sendReceipt();
}else{
Toast.makeText(PrintActivity.this,"query printer status error", Toast.LENGTH_SHORT).show();
}
}
}
}
};
@Override
protected void initTitle() {
mTitleTextMiddle.setText("打印");
}
private void connection() {
cOnn=newPrinterServiceConnection();
Intent intent =newIntent(this, GpPrintService.class);
this.bindService(intent,conn, Context.BIND_AUTO_CREATE);// bindService
}
classPrinterServiceConnectionimplementsServiceConnection {
@Override
public void onServiceDisconnected(ComponentName name) {
Log.i("ServiceConnection","onServiceDisconnected() called");
mGpService=null;
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
mGpService= GpService.Stub.asInterface(service);
}
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_print);
connection();
//注册实时状态查询广播
registerReceiver(mBroadcastReceiver,newIntentFilter(GpCom.ACTION_DEVICE_REAL_STATUS));
/**
*票据模式下,可注册该广播,在需要打印内容的最后加入addQueryPrinterStatus(),在打印完成后会接收到
* action为GpCom.ACTION_DEVICE_STATUS的广播,特别用于连续打印,
*可参照该sample中的sendReceiptWithResponse方法与广播中的处理
**/
registerReceiver(mBroadcastReceiver,newIntentFilter(GpCom.ACTION_RECEIPT_RESPONSE));
/**
*标签模式下,可注册该广播,在需要打印内容的最后加入addQueryPrinterStatus(RESPONSE_MODE mode)
*,在打印完成后会接收到,action为GpCom.ACTION_LABEL_RESPONSE的广播,特别用于连续打印,
*可参照该sample中的sendLabelWithResponse方法与广播中的处理
**/
registerReceiver(mBroadcastReceiver,newIntentFilter(GpCom.ACTION_LABEL_RESPONSE));
}
@Override
protected void initView() {
print.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch(v.getId()) {
caseR.id.print_print:
if(mGpService==null) {
UIUtils.showMessage("服务正在开启");
}else{
try{
inttype =mGpService.getPrinterCommandType(1);
if(type == GpCom.ESC_COMMAND) {
mGpService.queryPrinterStatus(1,1000,REQUEST_PRINT_RECEIPT);
}else{
Toast.makeText(this,"Printer is not receipt mode", Toast.LENGTH_SHORT).show();
}
}catch(RemoteException e1) {
e1.printStackTrace();
}
}
break;
}
}
private void sendReceipt() {
EscCommand esc =newEscCommand();
esc.addInitializePrinter();
esc.addPrintAndFeedLines((byte)3);
esc.addSelectJustification(EscCommand.JUSTIFICATION.CENTER);//设置打印居中
esc.addSelectPrintModes(EscCommand.FONT.FONTA, EscCommand.ENABLE.OFF, EscCommand.ENABLE.ON, EscCommand.ENABLE.ON, EscCommand.ENABLE.OFF);//设置为倍高倍宽
esc.addText("asdfkldsjgfsdl\n");//打印文字
esc.addPrintAndLineFeed();
/*打印文字 */
esc.addSelectPrintModes(EscCommand.FONT.FONTA, EscCommand.ENABLE.OFF, EscCommand.ENABLE.OFF, EscCommand.ENABLE.OFF, EscCommand.ENABLE.OFF);//取消倍高倍宽
esc.addSelectJustification(EscCommand.JUSTIFICATION.LEFT);//设置打印左对齐
esc.addText("dfkdsgklfds\n");//打印文字
// esc.addText("Welcome to use SMARNET printer!\n"); //打印文字
// /*打印繁体中文需要打印机支持繁体字库 */
// String message = "佳博智匯票據打印機\n";
// // esc.addText(message,"BIG5");
// esc.addText(message, "GB2312");
esc.addPrintAndLineFeed();
/*绝对位置具体详细信息请查看GP58编程手册 */
esc.addText("商品名称");
esc.addSetHorAndVerMotionUnits((byte)7, (byte)0);
esc.addSetAbsolutePrintPosition((short)6);
esc.addText("订单号");
esc.addSetAbsolutePrintPosition((short)10);
esc.addText("状态");
esc.addPrintAndLineFeed();
esc.addText("苹果");
esc.addSetHorAndVerMotionUnits((byte)7, (byte)0);
esc.addSetAbsolutePrintPosition((short)6);
esc.addText("12345");
esc.addSetAbsolutePrintPosition((short)10);
esc.addText("正常");
esc.addPrintAndLineFeed();
esc.addText("果粒橙300ml");
esc.addSetHorAndVerMotionUnits((byte)7, (byte)0);
esc.addSetAbsolutePrintPosition((short)6);
esc.addText("3545456");
esc.addSetAbsolutePrintPosition((short)10);
esc.addText("正常");
esc.addPrintAndLineFeed();
// /*打印图片 */
// esc.addText("Print bitmap!\n"); //打印文字
// Bitmap b = BitmapFactory.decodeResource(getResources(), R.drawable.gprinter);
// esc.addRastBitImage(b, 384, 0); //打印图片
// /*打印一维条码 */
// esc.addText("Print code128\n"); //打印文字
// esc.addSelectPrintingPositionForHRICharacters(EscCommand.HRI_POSITION.BELOW);//
// //设置条码可识别字符位置在条码下方
// esc.addSetBarcodeHeight((byte) 60); //设置条码高度为60点
// esc.addSetBarcodeWidth((byte) 1); //设置条码单元宽度为1
// esc.addCODE128(esc.genCodeB("SMARNET")); //打印Code128码
// esc.addPrintAndLineFeed();
/*
* QRCode命令打印此命令只在支持QRCode命令打印的机型才能使用。在不支持二维码指令打印的机型上,则需要发送二维条码图片
*/
esc.addText("商家二维码\n");//打印文字
esc.addSelectErrorCorrectionLevelForQRCode((byte)0x31);//设置纠错等级
esc.addSelectSizeOfModuleForQRCode((byte)6);//设置qrcode模块大小
esc.addStoreQRCodeData("dfgdgs");//设置qrcode内容
esc.addPrintQRCode();//打印QRCode
esc.addPrintAndLineFeed();
/*打印文字 */esc.addSelectJustification(EscCommand.JUSTIFICATION.CENTER);//设置打印左对齐
esc.addText("Completed!\r\n");//打印结束
//开钱箱
esc.addGeneratePlus(LabelCommand.FOOT.F5, (byte)255, (byte)255);
esc.addPrintAndFeedLines((byte)8);
Vector datas = esc.getCommand();//发送数据
byte[] bytes = GpUtils.ByteTo_byte(datas);
String sss = Base64.encodeToString(bytes, Base64.DEFAULT);
intrs;
try{
rs =mGpService.sendEscCommand(1, sss);
GpCom.ERROR_CODE r = GpCom.ERROR_CODE.values()[rs];
if(r != GpCom.ERROR_CODE.SUCCESS) {
Toast.makeText(getApplicationContext(), GpCom.getErrorText(r), Toast.LENGTH_SHORT).show();
}
}catch(RemoteException e) {
//TODO Auto-generated catch block
e.printStackTrace();
}
}
}

至此大功告成,打印出东西之后感觉好爽

以上所述是小编给大家介绍的Android手机通过蓝牙连接佳博打印机的实例代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!


推荐阅读
  • 在 CentOS 6.5 系统上部署 VNC 服务器的详细步骤与配置指南
    在 CentOS 6.5 系统上部署 VNC 服务器时,首先需要确认 VNC 服务是否已安装。通常情况下,VNC 服务默认未安装。可以通过运行特定的查询命令来检查其安装状态。如果查询结果为空,则表明 VNC 服务尚未安装,需进行手动安装。此外,建议在安装前确保系统的软件包管理器已更新至最新版本,以避免兼容性问题。 ... [详细]
  • SSAS入门指南:基础知识与核心概念解析
    ### SSAS入门指南:基础知识与核心概念解析Analysis Services 是一种专为决策支持和商业智能(BI)解决方案设计的数据引擎。该引擎能够为报告和客户端应用提供高效的分析数据,并支持在多维数据模型中构建高性能的分析应用。通过其强大的数据处理能力和灵活的数据建模功能,Analysis Services 成为了现代 BI 系统的重要组成部分。 ... [详细]
  • 本文探讨了如何有效地构建和优化微信公众平台账号,涵盖了用户信息管理、内容创作与发布、互动策略及数据分析等方面。通过合理设置用户信息字段,如用户名、昵称、密码、真实姓名和性别等,确保账号的安全性和用户体验。同时,文章还介绍了如何利用微信公众平台的各项功能,提升用户参与度和品牌影响力。 ... [详细]
  • 本文探讨了资源访问的学习路径与方法,旨在帮助学习者更高效地获取和利用各类资源。通过分析不同资源的特点和应用场景,提出了多种实用的学习策略和技术手段,为学习者提供了系统的指导和建议。 ... [详细]
  • Spring框架的核心组件与架构解析 ... [详细]
  • 本文详细解析了微信服务端示例类的功能与应用。其中,`ClientResponseHandler` 类主要用于处理微信支付所需的响应数据,而 `TenpayHttpClient` 则是对 HTTP 请求(包括 GET 和 POST 方法)进行了封装,以便在内部调用时更加便捷和高效。这些工具类在实际开发中起到了关键作用,开发者无需深入了解其底层实现细节,即可轻松集成微信支付功能。 ... [详细]
  • 本指南旨在帮助Swoole初学者快速掌握异步并发编程的基本概念和实践方法。通过实例演示,我们将使用Swoole PHP扩展构建一个简单的客户端与服务器模型,并实现基本的通信功能。首先,我们将从客户端的实现入手(文件名为:client.php)。 ... [详细]
  • Jeecg开源社区正式启动第12届架构技术培训班,现已开放报名。本次培训采用师徒制模式,深入探讨Java架构技术。类似于大学导师指导研究生的方式,特别适合在职人员。导师将为学员布置课题,提供丰富的视频资料,并进行一对一指导,帮助学员高效学习和完成任务。我们的教学方法注重实践与理论结合,旨在培养学员的综合技术能力。 ... [详细]
  • Docker入门指南:初探容器化技术
    Docker入门指南:初探容器化技术摘要:Docker 是一个使用 Go 语言开发的开源容器平台,旨在实现应用程序的构建、分发和运行的标准化。通过将应用及其依赖打包成轻量级的容器,Docker 能够确保应用在任何环境中都能一致地运行,从而提高开发和部署的效率。本文将详细介绍 Docker 的基本概念、核心功能以及如何快速上手使用这一强大的容器化工具。 ... [详细]
  • 本文详细介绍了如何安全地手动卸载Exchange Server 2003,以确保系统的稳定性和数据的完整性。根据微软官方支持文档(https://support.microsoft.com/kb833396/zh-cn),在进行卸载操作前,需要特别注意备份重要数据,并遵循一系列严格的步骤,以避免对现有网络环境造成不利影响。此外,文章还提供了详细的故障排除指南,帮助管理员在遇到问题时能够迅速解决,确保整个卸载过程顺利进行。 ... [详细]
  • 润乾报表JNDI配置异常分析与解决方法
    在使用润乾报表时,通过JNDI连接数据源是常见的配置方式。本文详细分析了JNDI配置异常的原因,并提供了具体的解决方法。重点讨论了JNDI名称在三个关键位置的配置:1. Tomcat或应用服务器的数据源配置(以Tomcat为例);2. 润乾报表中的数据源配置;3. 应用程序代码中的JNDI名称引用。通过这些步骤,可以有效解决JNDI配置异常问题,确保报表系统的稳定运行。 ... [详细]
  • Python 实战:异步爬虫(协程技术)与分布式爬虫(多进程应用)深入解析
    本文将深入探讨 Python 异步爬虫和分布式爬虫的技术细节,重点介绍协程技术和多进程应用在爬虫开发中的实际应用。通过对比多进程和协程的工作原理,帮助读者理解两者在性能和资源利用上的差异,从而在实际项目中做出更合适的选择。文章还将结合具体案例,展示如何高效地实现异步和分布式爬虫,以提升数据抓取的效率和稳定性。 ... [详细]
  • 本文探讨了Android系统中支持的图像格式及其在不同版本中的兼容性问题,重点涵盖了存储、HTTP传输、相机功能以及SparseArray的应用。文章详细分析了从Android 10 (API 29) 到Android 11 的存储规范变化,并讨论了这些变化对图像处理的影响。此外,还介绍了如何通过系统升级和代码优化来解决版本兼容性问题,以确保应用程序在不同Android版本中稳定运行。 ... [详细]
  • 利用树莓派畅享落网电台音乐体验
    最近重新拾起了闲置已久的树莓派,这台小巧的开发板已经沉寂了半年多。上个月闲暇时间较多,我决定将其重新启用。恰逢落网电台进行了改版,回忆起之前在树莓派论坛上看到有人用它来播放豆瓣音乐,便萌生了同样的想法。通过一番调试,终于实现了在树莓派上流畅播放落网电台音乐的功能,带来了全新的音乐享受体验。 ... [详细]
  • 在处理遗留数据库的映射时,反向工程是一个重要的初始步骤。由于实体模式已经在数据库系统中存在,Hibernate 提供了自动化工具来简化这一过程,帮助开发人员快速生成持久化类和映射文件。通过反向工程,可以显著提高开发效率并减少手动配置的错误。此外,该工具还支持对现有数据库结构进行分析,自动生成符合 Hibernate 规范的配置文件,从而加速项目的启动和开发周期。 ... [详细]
author-avatar
私受他
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有