- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
服务器代码
class UDPServer {
public static void main(String args[]) throws Exception
{
DatagramSocket serverSocket = new DatagramSocket(9876);
byte[] receiveData = new byte[1024];
byte[] sendData = new byte[1024];
while(true) {
DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);
serverSocket.receive(receivePacket);
System.out.println("RECEIVED: ");
sendData = receivePacket.getData();
InetAddress IPAddress = receivePacket.getAddress();
int port = receivePacket.getPort();
DatagramPacket sendPacket =
new DatagramPacket(sendData, sendData.length, IPAddress, port);
serverSocket.send(sendPacket);
}
}
}
和客户
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
import java.net.UnknownHostException;
import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.AudioTrack;
import android.media.MediaRecorder;
import android.util.Log;
public class AudioCall {
private static final String LOG_TAG = "AudioCall";
private static final int SAMPLE_RATE = 8000; // Hertz
private static final int SAMPLE_INTERVAL = 20; // Milliseconds
private static final int SAMPLE_SIZE = 2; // Bytes
private static final int BUF_SIZE = SAMPLE_INTERVAL * SAMPLE_INTERVAL * SAMPLE_SIZE * 2; //Bytes
private InetAddress address; // Address to call
private int port = 9876; // Port the packets are addressed to
private boolean mic = false; // Enable mic?
private boolean speakers = false; // Enable speakers?
public AudioCall(InetAddress address) {
this.address = address;
}
public void startCall() {
startMic();
startSpeakers();
}
public void endCall() {
Log.i(LOG_TAG, "Ending call!");
muteMic();
muteSpeakers();
}
public void muteMic() {
mic = false;
}
public void muteSpeakers() {
speakers = false;
}
public void startMic() {
// Creates the thread for capturing and transmitting audio
mic = true;
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
// Create an instance of the AudioRecord class
Log.i(LOG_TAG, "Send thread started. Thread id: " + Thread.currentThread().getId());
int a = AudioRecord.getMinBufferSize(SAMPLE_RATE, AudioFormat.CHANNEL_IN_MONO, AudioFormat.ENCODING_PCM_16BIT)*10;
System.out.println("test: " + a);
AudioRecord audioRecorder = new AudioRecord (MediaRecorder.AudioSource.MIC, SAMPLE_RATE,
AudioFormat.CHANNEL_IN_MONO, AudioFormat.ENCODING_PCM_16BIT,
AudioRecord.getMinBufferSize(SAMPLE_RATE, AudioFormat.CHANNEL_IN_MONO, AudioFormat.ENCODING_PCM_16BIT)*10);
int bytes_read = 0;
int bytes_sent = 0;
//byte[] buf = new byte[BUF_SIZE]; // 1600
//byte buf[BUF_SIZE] = new byte[BUF_SIZE];
try {
// Create a socket and start recording
Log.i(LOG_TAG, "Packet destination: " + address.toString());
DatagramSocket socket = new DatagramSocket();
audioRecorder.startRecording();
System.out.println("We can send now: " + mic);
DatagramSocket clientSocket = new DatagramSocket();
while(mic) {
// #############
InetAddress IPAddress = InetAddress.getByName("212.109.192.201");
byte[] sendData = new byte[1024];
bytes_read = audioRecorder.read(sendData, 0, 1024);
DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length, IPAddress, 9876);
clientSocket.send(sendPacket);
// ##############
// Capture audio from the mic and transmit it
//bytes_read = audioRecorder.read(buf, 0, BUF_SIZE);
//audioRecorder.read(buf, 0, BUF_SIZE);
// byte[] buf = new byte[BUF_SIZE];
// DatagramPacket packet = new DatagramPacket(buf, 1024, IPAddress, 9876);
//DatagramPacket packet = new DatagramPacket(buf, bytes_read, address, port);
// socket.send(packet);
bytes_sent += bytes_read;
Log.i(LOG_TAG, "Total bytes sent: " + bytes_sent);
// Thread.sleep(SAMPLE_INTERVAL, 0);
}
// Stop recording and release resources
audioRecorder.stop();
audioRecorder.release();
socket.disconnect();
socket.close();
mic = false;
return;
}
// catch(InterruptedException e) {
//
// Log.e(LOG_TAG, "InterruptedException: " + e.toString());
// mic = false;
// }
catch(SocketException e) {
Log.e(LOG_TAG, "SocketException: " + e.toString());
mic = false;
}
catch(UnknownHostException e) {
Log.e(LOG_TAG, "UnknownHostException: " + e.toString());
mic = false;
}
catch(IOException e) {
Log.e(LOG_TAG, "IOException: " + e.toString());
mic = false;
}
}
});
thread.start();
}
public void startSpeakers() {
// Creates the thread for receiving and playing back audio
if(!speakers) {
speakers = true;
Thread receiveThread = new Thread(new Runnable() {
@Override
public void run() {
// Create an instance of AudioTrack, used for playing back audio
Log.i(LOG_TAG, "Receive thread started. Thread id: " + Thread.currentThread().getId());
AudioTrack track = new AudioTrack(AudioManager.STREAM_MUSIC, SAMPLE_RATE, AudioFormat.CHANNEL_OUT_MONO,
AudioFormat.ENCODING_PCM_16BIT, BUF_SIZE, AudioTrack.MODE_STREAM);
track.play();
try {
// Define a socket to receive the audio
DatagramSocket socket = new DatagramSocket(port);
byte[] buf = new byte[BUF_SIZE];
System.out.println("We can listen now: " + speakers);
while(speakers) {
// Play back the audio received from packets
DatagramPacket packet = new DatagramPacket(buf, BUF_SIZE);
socket.receive(packet);
Log.i(LOG_TAG, "Packet received: " + packet.getLength());
track.write(packet.getData(), 0, BUF_SIZE);
}
// Stop playing back and release resources
socket.disconnect();
socket.close();
track.stop();
track.flush();
track.release();
speakers = false;
return;
}
catch(SocketException e) {
Log.e(LOG_TAG, "SocketException: " + e.toString());
speakers = false;
}
catch(IOException e) {
Log.e(LOG_TAG, "IOException: " + e.toString());
speakers = false;
}
}
});
receiveThread.start();
}
}
}
在本地主机上一切正常,但 UDP 服务器在公共(public)服务器上不接收,并且在某些情况下它只接收前 4 个数据包然后停止。服务器操作系统是ubuntu 14.0版本,无防火墙。我注意到,当UDP数据包的字节值为负值时,它会接收UDP数据包,但当字节值为正值时,它不会接收UDP数据包。 Android list 一切都很好,它具有互联网和其他所需的权限。我想创建一个应用程序,它像一个 channel 一样创建,并将音频广播到该 channel 。任何客户端都可以通过收听特定 channel 来收听音频。问题是它应该是实时的并且具有搜索功能,因此当您实时收听时,可以返回之前收听几秒钟,甚至从头开始收听。请问有什么建议吗?我将非常感谢您的回复
最佳答案
正如评论中所述,您描述的情况是不可能的。然而这个:
while(true)
{
DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);
serverSocket.receive(receivePacket);
System.out.println("RECEIVED: ");
sendData = receivePacket.getData();
InetAddress IPAddress = receivePacket.getAddress();
int port = receivePacket.getPort();
DatagramPacket sendPacket =
new DatagramPacket(sendData, sendData.length, IPAddress, port);
serverSocket.send(sendPacket);
}
不正确。无论实际收到什么,您总是以 1024 字节进行回复。由于接收到的数据包已经包含了自己的长度和源地址,因此可以大大简化为:
while(true)
{
DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);
serverSocket.receive(receivePacket);
System.out.println("RECEIVED: ");
serverSocket.send(receivePacket);
}
在客户端中,您不需要在循环内不断获取相同的 InetAddress
以及以下内容:
bytes_read = audioRecorder.read(sendData, 0, 1024);
DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length, IPAddress, 9876);
应该是:
bytes_read = audioRecorder.read(sendData, 0, 1024);
if (bytesRead == -1)
break;
DatagramPacket sendPacket = new DatagramPacket(sendData, bytesRead, IPAddress, 9876);
关于当字节值为正时,Java UDP 服务器未接收数据包。在本地主机上工作正常,但在外部 IP 服务器上工作不正常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35465125/
SQLite、Content provider 和 Shared Preference 之间的所有已知区别。 但我想知道什么时候需要根据情况使用 SQLite 或 Content Provider 或
警告:我正在使用一个我无法完全控制的后端,所以我正在努力解决 Backbone 中的一些注意事项,这些注意事项可能在其他地方更好地解决......不幸的是,我别无选择,只能在这里处理它们! 所以,我的
我一整天都在挣扎。我的预输入搜索表达式与远程 json 数据完美配合。但是当我尝试使用相同的 json 数据作为预取数据时,建议为空。点击第一个标志后,我收到预定义消息“无法找到任何内容...”,结果
我正在制作一个模拟 NHL 选秀彩票的程序,其中屏幕右侧应该有一个 JTextField,并且在左侧绘制弹跳的选秀球。我创建了一个名为 Ball 的类,它实现了 Runnable,并在我的主 Draf
这个问题已经有答案了: How can I calculate a time span in Java and format the output? (18 个回答) 已关闭 9 年前。 这是我的代码
我有一个 ASP.NET Web API 应用程序在我的本地 IIS 实例上运行。 Web 应用程序配置有 CORS。我调用的 Web API 方法类似于: [POST("/API/{foo}/{ba
我将用户输入的时间和日期作为: DatePicker dp = (DatePicker) findViewById(R.id.datePicker); TimePicker tp = (TimePic
放宽“邻居”的标准是否足够,或者是否有其他标准行动可以采取? 最佳答案 如果所有相邻解决方案都是 Tabu,则听起来您的 Tabu 列表的大小太长或您的释放策略太严格。一个好的 Tabu 列表长度是
我正在阅读来自 cppreference 的代码示例: #include #include #include #include template void print_queue(T& q)
我快疯了,我试图理解工具提示的行为,但没有成功。 1. 第一个问题是当我尝试通过插件(按钮 1)在点击事件中使用它时 -> 如果您转到 Fiddle,您会在“内容”内看到该函数' 每次点击都会调用该属
我在功能组件中有以下代码: const [ folder, setFolder ] = useState([]); const folderData = useContext(FolderContex
我在使用预签名网址和 AFNetworking 3.0 从 S3 获取图像时遇到问题。我可以使用 NSMutableURLRequest 和 NSURLSession 获取图像,但是当我使用 AFHT
我正在使用 Oracle ojdbc 12 和 Java 8 处理 Oracle UCP 管理器的问题。当 UCP 池启动失败时,我希望关闭它创建的连接。 当池初始化期间遇到 ORA-02391:超过
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve
引用这个plunker: https://plnkr.co/edit/GWsbdDWVvBYNMqyxzlLY?p=preview 我在 styles.css 文件和 src/app.ts 文件中指定
为什么我的条形这么细?我尝试将宽度设置为 1,它们变得非常厚。我不知道还能尝试什么。默认厚度为 0.8,这是应该的样子吗? import matplotlib.pyplot as plt import
当我编写时,查询按预期执行: SELECT id, day2.count - day1.count AS diff FROM day1 NATURAL JOIN day2; 但我真正想要的是右连接。当
我有以下时间数据: 0 08/01/16 13:07:46,335437 1 18/02/16 08:40:40,565575 2 14/01/16 22:2
一些背景知识 -我的 NodeJS 服务器在端口 3001 上运行,我的 React 应用程序在端口 3000 上运行。我在 React 应用程序 package.json 中设置了一个代理来代理对端
我面临着一个愚蠢的问题。我试图在我的 Angular 应用程序中延迟加载我的图像,我已经尝试过这个2: 但是他们都设置了 src attr 而不是 data-src,我在这里遗漏了什么吗?保留 d
我是一名优秀的程序员,十分优秀!